
代码
/**
* 服务器二次验证代码
* @param ¥receipt
* @param bool ¥isSandbox
* @return array|void
* @throws \Exception
*/
function getReceiptData(¥receipt, ¥isSandbox = false) {
if (¥isSandbox) {
¥endpoint = 'https://sandbox.itunes.apple.com/verifyReceipt';
} else {
¥endpoint = 'https://buy.itunes.apple.com/verifyReceipt';
}
//¥receipt = str_replace(' ',+, ¥receipt);
¥postData = json_encode([receipt-data => ¥receipt]);
//¥postData = '{receipt-data:'. ¥receipt .'}';
BLog::pay(苹果支付:postData:¥postData);
¥ch = curl_init(¥endpoint);
curl_setopt(¥ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt(¥ch, CURLOPT_POST, true);
curl_setopt(¥ch, CURLOPT_POSTFIELDS, ¥postData);
curl_setopt(¥ch, CURLOPT_SSL_VERIFYPEER, 0); //这两行一定要加,不加会报SSL 错误
curl_setopt(¥ch, CURLOPT_SSL_VERIFYHOST, 0);
¥response = curl_exec(¥ch);
¥errno = curl_errno(¥ch);
¥errmsg = curl_error(¥ch);
curl_close(¥ch);
//判断时候出错,抛出异常
if (¥errno != 0) {
return [
'result' => false,
'txtMessage' => ¥errmsg,
];
}
¥data = json_decode(¥response);
BLog::pay(苹果支付:response:¥response);
//此处是看到先人们的指导,又看到apple的官方说法改的。否则会审核不过貌似是审核也会走沙盒测试者,
//此处先判断一次返回的status是否=21007 这数据是从测试环境,但它发送到生产环境中进行验证。它发送到测试环境来代替。
if (¥data->status == 21007) {
¥this->getReceiptData(¥receipt, true);
return;
}
//判断返回的数据是否是对象
if (!is_object(¥data)) {
throw new \Exception('Invalid response data');
}
//判断购买时候成功
if (!isset(¥data->status) || ¥data->status != 0) {
throw new \Exception('Invalid receipt');
}
//返回产品的信息
return (array)¥data->receipt;
}
/**
* ios 支付验证
* @param ¥allData
* @return array|string
*/
public function iosIAPPay(¥allData) {
¥receiptData = ¥allData['receiptData'];
//获取 App 发送过来的数据,设置时候是沙盒状态
¥receipt = ¥receiptData;
¥isSandbox = true;
if ('server' == APP_ENV) {
¥isSandbox = false;
}
//开始执行验证
try {
¥info = ¥this->getReceiptData(¥receipt, ¥isSandbox);
¥package_name = ¥info['bid'] ?? ;
if (!¥package_name) {
¥package_name = ¥info['bundle_id'] ?? ;
}
if (!¥package_name) {
return [
'result' => false,
'txtMessage' => 'bid或bundle_id缺失',
];
}
¥product_id = ¥info['product_id'] ?? ;
if (!¥product_id) {
¥product_id = ¥info['in_app'][0]->product_id ?? ;
}
if (!¥product_id) {
return [
'result' => false,
'txtMessage' => 'product_id缺失',
];
}
¥transaction_id = ¥info['transaction_id'] ?? ;
if (!¥transaction_id) {
¥transaction_id = ¥info['in_app'][0]->transaction_id ?? ;
}
if (!¥transaction_id) {
return [
'result' => false,
'txtMessage' => 'transaction_id缺失',
];
}
¥productInfo = DB::table('ios_payment_config')
->where('package_name', ¥package_name)
->where('product_id', ¥product_id)
->first();
if (!¥productInfo) {
BLog::pay(苹果支付:无该产品: . json_encode(¥info));
return [
'result' => false,
'txtMessage' => '无该产品',
];
}
¥user_open_id = ¥info['download_id'] ?? ;
¥res = ¥this->doPay(¥allData['userId'], ¥allData['orderId'], ¥transaction_id, ¥user_open_id);
BLog::pay(苹果支付:支付结果: . json_encode(¥info) . json_encode(¥res));
return ¥res;
} catch (\Exception ¥e) {
return [
'result' => false,
'txtMessage' => ¥e->getMessage(),
];
}
}
常见错误码
/**
* 21000 App Store不能读取你提供的JSON对象
* 21002 receipt-data域的数据有问题
* 21003 receipt无法通过验证
* 21004 提供的shared secret不匹配你账号中的shared secret
* 21005 receipt服务器当前不可用
* 21006 receipt合法,但是订阅已过期。服务器接收到这个状态码时,receipt数据仍然会解码并一起发送
* 21007 receipt是Sandbox receipt,但却发送至生产系统的验证服务
* 21008 receipt是生产receipt,但却发送至Sandbox环境的验证服务
*/
遇到的坑
回包结构的修改
新结构:
{
receipt: {
receipt_type: ProductionSandbox,
adam_id: 0,
app_item_id: 0,
bundle_id: 申请苹果支付时的串号 固定的值,
application_version: 24,
download_id: 0,
version_external_identifier: 0,
receipt_creation_date: 2022-************Etc/GMT,
receipt_creation_date_ms: 1643************000,
receipt_creation_date_pst: 2022-0a************/Los_Angeles,
request_date: 2022-02-2************Etc/GMT,
request_date_ms: 164************7,
request_date_pst: 2022-************ngeles,
original_purchase_date: 201************tc/GMT,
original_purchase_date_ms: 13************00,
original_purchase_date_pst: 2013************geles,
original_application_version: 1.0,
in_app: [ //变成了数组
{
quantity: 1,
product_id: 10000,#产品ID
transaction_id: , ##交易单号
original_transaction_id: ,
purchase_date: 2022************Etc/GMT,
purchase_date_ms: 164************0,
purchase_date_pst: 2022-************les,
original_purchase_date: 2022************GMT,
original_purchase_date_ms: 164************0,
original_purchase_date_pst: 202************les,
is_trial_period: false,
in_app_ownership_type: PURCHASED#交易状态
},
{
quantity: 1,
product_id: 10000,#产品ID
transaction_id: , ##交易单号
original_transaction_id: ,
purchase_date: 2022************Etc/GMT,
purchase_date_ms: 164************0,
purchase_date_pst: 2022-************les,
original_purchase_date: 2022************GMT,
original_purchase_date_ms: 164************0,
original_purchase_date_pst: 202************les,
is_trial_period: false,
in_app_ownership_type: PURCHASED#交易状态
},
]
},
environment: Sandbox, //环境
status: 0
}
老版本是一个一维数组,没有in_app
post请求的坑,报21003
第一种,有空格,使用字符串替换
¥receipt = str_replace(' ',+, ¥receipt);
1.
第二种,json不对,直接拼字符串,不要json_encode
¥postData = '{receipt-data:'. ¥receipt .'}';
1.
第三种,苹果那边的问题,什么都没改,第一天好的,第二天报错(沙盒报错,线上好的),第三天好了。
当你停下来休息的时候,不要忘记别人还在奔跑!
扫描二维码与小二CMS创始人沟通:
小二CMS专注于高端网站定制、系统开发、商城开发、外贸网站建设、公众号开发、小程序开发、网站优化推广、安全运维等技术领域。是高端定制网站领域著名服务商!
上一篇: 修改DNS解决个别网站无法访问的问题
下一篇: css控制文字加白色描边的方法
本文提供打造高效互动社区论坛网站的全方位指南与实用技巧,涵盖定位规划、功能设计、用户运营、内容管理与技术优化,并结合小二CMS的强大功能,帮助团队快速构建高粘性、高活跃的论坛平台。
本文系统讲解医院网站优化排名的具体流程,涵盖关键词研究、站内优化、内容建设、技术SEO及数据监测,并结合小二CMS的强大功能,助力医疗机构提升搜索引擎可见度与患者访问转化率。
本文深入解析模板网站建设与定制网站设计的核心区别,结合小二CMS的功能优势,帮助企业明确建站需求与投入方向,找到兼顾效率、成本与个性化的最佳方案,提升品牌竞争力。
本文聚焦合肥企业在数字时代的建站需求,解析如何借助小二CMS快速打造高性能官网,融合本地化营销与SEO策略,提升品牌形象与商业转化,开启全新增长篇章。
本文详解网页设计中的字体规范指南,结合“小二CMS”实践案例,解析如何通过科学排版、合理字阶与色彩搭配提升可读性与视觉美感,助力打造卓越用户体验。
针对物流与快递行业对高效获客、透明服务及数字化管理的需求,本文提出基于小二CMS的专业网站建设解决方案。通过在线下单、实时运单跟踪、智能客服等核心功能模块,结合小二CMS的可视化内容管理、多端适配与数据整合能力,助力物流快递企业构建集品牌展示、业务办理与客户运营于一体的数字化门户,实现服务效率提升与客户体验优化。
集团企业门户网站是展示企业综合实力、实现多层级协同与品牌统一的核心数字阵地。本文针对集团企业跨地域、多业务线、多层级的管理特点,提出基于小二CMS的定制化解决方案。通过多站点统一管理、分级权限管控、品牌内容聚合分发等核心能力,小二CMS助力集团企业构建“总部中枢+分支协同”的数字化门户体系,实现品牌形象一致、信息高效流转与内外资源整合,为集团战略落地与业务增长提供强力支撑。
针对初创型中小企业资源有限、数字化能力薄弱等痛点,本文提出基于小二CMS的一站式解决方案。从低成本内容管理到精准营销获客,从私域流量沉淀到高效运营协作,小二CMS凭借可视化操作、智能工具集成与灵活部署能力,助力初创企业以最小成本构建数字化竞争力,实现“从0到1”的快速成长。