This repository has been archived on 2026-02-24. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
market/scrape/product/registry.py
giles 6271a715a1
Some checks failed
Build and Deploy / build-and-deploy (push) Has been cancelled
feat: initialize market app with browsing, product, and scraping code
Split from coop monolith. Includes:
- Market/browse/product blueprints
- Product sync API
- Suma scraping pipeline
- Templates for market, browse, and product views
- Dockerfile and CI workflow for independent deployment
2026-02-09 23:16:34 +00:00

21 lines
625 B
Python

from __future__ import annotations
from typing import Callable, Dict, List, Union
Extractor = Callable[[object, str], Dict[str, Union[str, float, int, list, dict, None]]]
REGISTRY: List[Extractor] = []
def extractor(fn: Extractor) -> Extractor:
"""Decorator to register an extractor."""
REGISTRY.append(fn)
return fn
def merge_missing(dst: dict, src: dict) -> None:
"""
Merge src into dst. Only write keys that are missing or empty in dst.
"Empty" means None, "", [], {}.
"""
for k, v in (src or {}).items():
if k not in dst or dst[k] in (None, "", [], {}):
dst[k] = v