Files
seed-mcp/scrape/sources/bayer_seeds.py
T
justin ac40e05734
Image rebuild (skip scrape) / build (push) Failing after 7s
seed-mcp scaffold: clone docs-mcp-template, customize for crop_seed PRODUCT_NAME
Sibling project to crop-chem-docs, same MCP-template lineage. Corpus is
seed/hybrid varieties across 6 vendors instead of pesticide labels.

What's customized vs. the template:
- CLAUDE.md: vendor matrix, build priority, Pioneer fallback policy,
  canonical sidecar schema (per-crop), Golden Harvest disease-scale
  reversal gotcha, no-IPv6 / HTTPS-clone note
- README.md: vendor coverage table, tool list, phase status
- Dockerfile: PRODUCT_NAME=crop_seed default, sources.json (not
  bundles.json), HYBRID_SEARCH=true, OLLAMA_URL + RERANK_URL Docker
  DNS defaults (same llama-rerank sidecar as crop-chem-docs)
- .gitea/workflows/refresh.yml: monthly cron (seed catalogs move
  slowly), 5 GREEN scraper steps, corpus-YYYY.MM.DD tag for Drawbar
  pinning, continue-on-error on GC step
- .gitea/workflows/image-only.yml: paths filter + cancel-in-progress
  concurrency group
- scripts/registry_gc.py: lifted from crop-chem-docs (correct Gitea
  packages API URL + UA header to bypass CF block on default
  Python-urllib UA)
- sources.json: catalog of 6 sources + scope_filter + per-source
  schema notes + Pioneer-exclusion rationale
- scrape/runner.py: dispatcher with --all = GREEN-only
- scrape/sources/{bayer_seeds,golden_harvest,nk,agripro,becks_pfr,
  becks_products}.py: stub modules with implementation notes
- docs_mcp/server.py: PRODUCT_NAME default → crop_seed,
  PRODUCT_DOCS_URL → repo URL

Pioneer is intentionally NOT a source. ToS bans automation; dealer
locator is login-gated. The MCP returns a curated fallback lesson
directing the user to pioneer.com.

Next phases:
- Phase 1: implement bayer_seeds (lift-and-shift from crop-chem-docs
  Bayer scraper; same __NEXT_DATA__ infra)
- Phase 7: curate eval/queries.jsonl
- Phase 11: lessons.md with Pioneer fallback + disease-scale notes

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-25 12:28:49 -04:00

57 lines
1.9 KiB
Python

"""Bayer seeds scraper — DEKALB (corn) + Asgrow (soy) + WestBred (wheat).
Source: ``cropscience.bayer.us`` — same Next.js + ``__NEXT_DATA__``
infrastructure used by crop-chem-docs' Bayer crop-protection scraper.
That scraper is the reference; this one lifts ~80% of its plumbing
and adapts the per-product field mapping for seed schema.
Catalog index pages:
/corn/dekalb/seed-catalog
/soybeans/asgrow/seed-catalog
/wheat/westbred/seed-catalog
Each catalog page is a Next.js route; the per-variety data lives in
``__NEXT_DATA__.props.pageProps.{whatever}``. The buildId in the
script tag rotates — fetch the index page first, extract the
buildId, then fetch the per-variety JSON.
Output layout:
corpus/bayer_seeds/<source_key>.md LLM-visible body
corpus/bayer_seeds/<source_key>.json Sidecar metadata
source_key convention: ``<brand>-<product-slug>`` lowercased, e.g.
``dekalb-dkc62-08rib`` or ``asgrow-ag34xf2``.
Sidecar schema (per CLAUDE.md):
source: "bayer_seeds"
source_key: str
vendor: "Bayer"
brand: "DEKALB" | "Asgrow" | "WestBred"
product_name: str
crop: "corn" | "soybeans" | "wheat"
relative_maturity: int | null # corn only
maturity_group: float | null # soy only
wheat_class: str | null # wheat only
trait_stack: list[str]
agronomic_ratings: dict[str, int] # normalized 1-9 (9 = best)
disease_ratings: dict[str, int] # normalized 1-9 (9 = best)
regional_recommendation: list[str]
source_urls: list[str]
fetched_at: str (ISO 8601 UTC)
TODO: implement. Reference: ~/github/crop-chem-docs/scrape/sources/bayer.py
"""
from __future__ import annotations
import sys
def main(argv: list[str] | None = None) -> int:
print("bayer_seeds: not implemented yet — see ~/github/crop-chem-docs/scrape/sources/bayer.py for the reference Next.js extraction pattern",
file=sys.stderr)
return 2
if __name__ == "__main__":
sys.exit(main(sys.argv[1:]))