文件内容
scripts/_temu_common.py
#!/usr/bin/env python3
"""Shared helpers for LinkFox Temu API skill scripts."""
import json
import os
import sys
from urllib.error import HTTPError, URLError
from urllib.request import Request, urlopen
from _temu_token_store import get_token
BASE_URL = os.environ.get("TEMU_API_BASE_URL") or os.environ.get(
"STORE_API_BASE_URL", "https://tool-gateway.linkfox.com"
)
BASE_URL = BASE_URL.rstrip("/")
PROXY_URL = f"{BASE_URL}/temu/proxy"
FILE_DOWNLOAD_URL = f"{BASE_URL}/temu/fileDownload"
VALID_SITES = frozenset({"cn", "partner", "us", "global", "eu"})
VALID_MANAGEMENT_TYPES = frozenset({"full-managed", "semi-managed"})
# LinkFox 用户 Token(网关鉴权),勿与 Body 中的 Temu accessToken 混淆
LINKFOX_TOKEN_PARAM_KEYS = ("token", "linkfoxToken", "linkfox_token")
def get_linkfox_token(params=None) -> str:
"""
LinkFox 用户鉴权 Token,与 linkfox-amazon-store-auth 一致。
优先级:请求 JSON 中的 token / linkfoxToken > 环境变量 LINKFOXAGENT_API_KEY。
"""
if params:
for key in LINKFOX_TOKEN_PARAM_KEYS:
value = params.get(key)
if value is not None and str(value).strip():
return str(value).strip()
key = os.environ.get("LINKFOXAGENT_API_KEY")
if not key:
print(
"LinkFox user Token not configured. Same as linkfox-amazon-store-auth:\n"
"1. Visit https://yxgb3sicy7.feishu.cn/wiki/GIkkweGghiyzkqkRXQKc2n0Tnre to obtain your Key\n"
"2. export LINKFOXAGENT_API_KEY=your-key-here\n"
" Or pass \"token\" in the JSON parameters of proxy/fileDownload scripts.",
file=sys.stderr,
)
sys.exit(1)
return key
def build_gateway_headers(linkfox_token: str) -> dict:
"""网关鉴权:Authorization(全站通用)+ Token(TEMU_API_SPEC 约定)。"""
return {
"Authorization": linkfox_token,
"Token": linkfox_token,
"Content-Type": "application/json",
"User-Agent": "LinkFox-Skill/1.0",
}
def load_json_arg(argv: list) -> dict:
if len(argv) < 2:
return {}
try:
return json.loads(argv[1])
except json.JSONDecodeError as e:
print(f"Invalid parameter format: {e}", file=sys.stderr)
sys.exit(1)
def require_text(params: dict, key: str, label=None) -> str:
value = params.get(key)
if value is None or not str(value).strip():
print(f"Error: '{label or key}' is required.", file=sys.stderr)
sys.exit(1)
return str(value).strip()
def validate_site(site: str) -> str:
if site not in VALID_SITES:
print(
f"Error: invalid site '{site}'. Must be one of: {', '.join(sorted(VALID_SITES))}",
file=sys.stderr,
)
sys.exit(1)
return site
def validate_management_type(management_type: str) -> str:
if management_type not in VALID_MANAGEMENT_TYPES:
print(
"Error: invalid managementType. Must be: full-managed, semi-managed",
file=sys.stderr,
)
sys.exit(1)
return management_type
def call_temu_api(
url: str,
body: dict,
timeout: int = 60,
linkfox_params=None,
) -> dict:
"""调用 Temu 网关接口;必须先具备 LinkFox 用户 Token。"""
linkfox_token = get_linkfox_token(linkfox_params)
data = json.dumps(body, ensure_ascii=False).encode("utf-8")
req = Request(
url,
data=data,
headers=build_gateway_headers(linkfox_token),
method="POST",
)
try:
with urlopen(req, timeout=timeout) as response:
return json.loads(response.read().decode("utf-8"))
except HTTPError as e:
raw = e.read().decode("utf-8") if e.fp else ""
try:
return json.loads(raw)
except json.JSONDecodeError:
return {"error": f"HTTP {e.code}: {e.reason}", "details": raw}
except URLError as e:
return {"error": f"Connection failed: {e.reason}"}
def is_linkfox_auth_error(result: dict) -> bool:
msg = str(result.get("message") or result.get("error") or "")
return "无法识别当前用户" in msg or "重新登录" in msg
def resolve_access_token(params: dict) -> str:
"""Temu 店铺 accessToken:直接传入或从本地 storeKey 读取。"""
if params.get("accessToken"):
return str(params["accessToken"]).strip()
store_key = params.get("storeKey")
if not store_key:
print(
"Error: provide Temu 'accessToken' or 'storeKey' (+ site, managementType).",
file=sys.stderr,
)
sys.exit(1)
site = validate_site(require_text(params, "site"))
management_type = validate_management_type(require_text(params, "managementType"))
token_purpose = str(params.get("tokenPurpose", "default")).strip() or "default"
token = get_token(str(store_key).strip(), site, management_type, token_purpose)
if not token:
print(
f"Error: no Temu token for storeKey={store_key}, site={site}, "
f"managementType={management_type}, tokenPurpose={token_purpose}. "
"Run temu_token_guide.py and save_temu_access_token.py first.",
file=sys.stderr,
)
sys.exit(1)
return token
def parse_nested_body(result: dict) -> dict:
"""If gateway returns a JSON string in body, parse it into temuBody."""
body = result.get("body")
if isinstance(body, str) and body.strip():
try:
result["temuBody"] = json.loads(body)
except json.JSONDecodeError:
pass
return result