feat(mcp-macro): fetch_cot_tff async fetcher with cache
This commit is contained in:
@@ -5,6 +5,16 @@ from typing import Any
|
||||
|
||||
import httpx
|
||||
from mcp_common.http import async_client
|
||||
from mcp_macro.cot import parse_disagg_row, parse_tff_row
|
||||
from mcp_macro.cot_contracts import (
|
||||
ALL_DISAGG_SYMBOLS,
|
||||
ALL_TFF_SYMBOLS,
|
||||
CFTC_BASE_URL,
|
||||
DISAGG_DATASET_ID,
|
||||
SYMBOL_TO_CFTC_CODE_DISAGG,
|
||||
SYMBOL_TO_CFTC_CODE_TFF,
|
||||
TFF_DATASET_ID,
|
||||
)
|
||||
|
||||
FRED_BASE = "https://api.stlouisfed.org/fred/series/observations"
|
||||
FINNHUB_CALENDAR = "https://finnhub.io/api/v1/calendar/economic"
|
||||
@@ -609,3 +619,48 @@ async def fetch_market_overview() -> dict[str, Any]:
|
||||
_MARKET_CACHE["data"] = out
|
||||
_MARKET_CACHE["ts"] = now
|
||||
return out
|
||||
|
||||
|
||||
_COT_TTL = 3600.0 # 1h
|
||||
_COT_CACHE: dict[tuple[str, str, int], dict[str, Any]] = {}
|
||||
_COT_CACHE_TS: dict[tuple[str, str, int], float] = {}
|
||||
|
||||
|
||||
async def fetch_cot_tff(symbol: str, lookback_weeks: int = 52) -> dict[str, Any]:
|
||||
"""Fetch COT TFF report per simbolo equity/financial. Returns ASC by date."""
|
||||
import time
|
||||
|
||||
symbol = symbol.upper()
|
||||
if symbol not in SYMBOL_TO_CFTC_CODE_TFF:
|
||||
return {"error": "unknown_symbol", "available": ALL_TFF_SYMBOLS}
|
||||
|
||||
key = (symbol, "tff", lookback_weeks)
|
||||
now = time.monotonic()
|
||||
if key in _COT_CACHE and (now - _COT_CACHE_TS[key]) < _COT_TTL:
|
||||
return _COT_CACHE[key]
|
||||
|
||||
code = SYMBOL_TO_CFTC_CODE_TFF[symbol]
|
||||
url = f"{CFTC_BASE_URL}/{TFF_DATASET_ID}.json"
|
||||
async with async_client(timeout=10.0) as client:
|
||||
resp = await client.get(
|
||||
url,
|
||||
params={
|
||||
"cftc_contract_market_code": code,
|
||||
"$order": "report_date_as_yyyy_mm_dd DESC",
|
||||
"$limit": str(lookback_weeks),
|
||||
},
|
||||
)
|
||||
if resp.status_code != 200:
|
||||
return {"symbol": symbol, "report_type": "tff", "rows": [], "error": "cftc_unavailable"}
|
||||
raw_rows = resp.json() or []
|
||||
parsed = [parse_tff_row(r) for r in raw_rows]
|
||||
parsed.sort(key=lambda r: r["report_date"]) # ASC by date
|
||||
out = {
|
||||
"symbol": symbol,
|
||||
"report_type": "tff",
|
||||
"rows": parsed,
|
||||
"data_timestamp": datetime.now(UTC).isoformat(),
|
||||
}
|
||||
_COT_CACHE[key] = out
|
||||
_COT_CACHE_TS[key] = now
|
||||
return out
|
||||
|
||||
Reference in New Issue
Block a user