
本文详解如何在 PHP 后端正确解析 Shopify 官方 CDN 上的 currencies.js 文件,提取其中的 rates 对象,并规避因 JavaScript 语法导致的 JSON 解析失败问题。
本文详解如何在 php 后端正确解析 shopify 官方 cdn 上的 `currencies.js` 文件,提取其中的 `rates` 对象,并规避因 javascript 语法导致的 json 解析失败问题。
Shopify 提供的 https://cdn.shopify.com/s/javascripts/currencies.js 并非标准 JSON 接口,而是一个导出 JavaScript 对象的脚本文件(如 var Currency = { rates: { USD: 1, EUR: 0.92, ... }, convert: function(...) { ... } };)。直接使用 json_decode() 解析其响应体必然返回 null——因为 JS 对象字面量 ≠ JSON 字符串。
要成功提取 rates,核心思路是:将 JS 变量声明转换为合法 JSON 字符串,再交由 json_decode() 处理。以下是一套经过验证、鲁棒性较强的 PHP 处理流程:
$response = Http::get('https://cdn.shopify.com/s/javascripts/currencies.js');
$res = $response->body();
// 步骤 1:剥离变量声明和函数定义,保留纯对象结构
$res = trim(str_replace("var Currency=", "", $res));
$res = str_replace(",convert:function(D,R,S){return D*this.rates[R]/this.rates[S]}};", "}", $res);
$res = str_replace("//# sourceMappingURL=/s/javascripts/currencies.js.map", "", $res);
// 步骤 2:修复键名与字符串值的引号格式(JS 允许无引号键名,JSON 不允许)
$res = str_replace('{', '{"', $res);
$res = str_replace(':', '":"', $res);
$res = str_replace(',', '","', $res);
$res = str_replace('}', '"}');
$res = str_replace('"rates":"', '"rates":', $res); // 修正 rates 键值对格式
// 步骤 3:清理换行、HTML 实体及多余引号
$res = str_replace("\n", "", $res);
$res = str_replace('"', '"', $res);
$res = trim($res);
$res = str_replace('"{', '{', $res);
$res = str_replace('}"', '}', $res);
$res = $res . "}"; // 补全结尾大括号(防截断)
// 步骤 4:解析为关联数组
$enc = json_decode($res, true);
if (json_last_error() !== JSON_ERROR_NONE) {
throw new RuntimeException('Failed to parse currencies.js: ' . json_last_error_msg());
}
// ✅ 成功获取汇率数组
$rates = $enc['rates'] ?? [];
// 示例:安全获取并转换 VED 汇率(确保数值类型)
$ved_rate = isset($rates['VED']) ? floatval($rates['VED']) : 0.0;
echo "VES rate: {$ved_rate}"; // 输出类似:VES rate: 36.521⚠️ 重要注意事项:
- 稳定性风险:该脚本结构属 Shopify 内部实现,未提供官方 API 保障,未来可能变更格式或路径,不建议用于生产环境的关键业务逻辑;
- 推荐替代方案:优先使用 Shopify Admin API 的 /admin/api/{version}/currencies.json(需权限)或接入权威第三方汇率服务(如 exchangerate-api.com、Open Exchange Rates);
- 缓存必做:CDN 文件虽静态,但频繁请求易被限流;务必本地缓存(如 Redis 或文件),设置合理 TTL(例如 1 小时);
- 错误防御:始终检查 json_last_error() 和 $enc['rates'] 是否存在,避免未定义索引警告;
- 字符编码:确保 HTTP 请求响应编码为 UTF-8,必要时添加 mb_convert_encoding($res, 'UTF-8', 'auto')。
总结:技术上可行 ≠ 架构上推荐。本文提供的解析逻辑可作为临时调试或轻量场景的兜底方案,但长期应迁移到受支持、可监控、有 SLA 的汇率数据源。










