如何实现应用内支付零掉单?

在电子商务的快速发展中,用户体验成为了衡量平台服务质量的重要标准。当用户完成商品购买后,开发者需要及时发放相关权益。但在实际应用场景中,由于网络波动、进程被中止等异常因素,可能会出现支付状态无法及时确认的情况,从而无法及时发放权益,即出现"掉单"情况。

掉单不仅会严重影响用户支付体验,还可能引发退款、投诉等,容易产生负面舆情,从而增加开发者和运营人员额外的操作成本。

HarmonyOS SDK应用内支付服务(IAP Kit)为了确保权益发放,需要在以下2种场景检查用户是否存在已购未发货的商品,以防止掉单情况。

1.应用启动时。

2.购买请求(createPurchase)返回iap.IAPErrorCode.PRODUCT_OWNEDiap.IAPErrorCode.SYSTEM_ERROR时

如果存在已购未发货商品,则发放相关权益,然后向IAP Kit确认发货,完成购买。

业务流程

开发步骤

1.应用客户端向IAP Kit发起queryPurchases请求,获取用户已购买但未确认发货的订单信息。

2.在请求参数QueryPurchasesParameter中指定对应的productType,同时指定queryType为iap.PurchaseQueryType.UNFINISHED。当接口请求成功时,IAP Kit将返回一个QueryPurchaseResult对象,该对象包含承载了订单信息的PurchaseData的列表。

3.对purchaseData.jwsPurchaseOrder进行解码验签。建议应用客户端将purchaseData发送至应用服务器,在应用服务器执行此操作。

4.验证成功可得到对应的PurchaseOrderPayload的JSON字符串,如果PurchaseOrderPayload.purchaseOrderRevocationReasonCode为空,则代表购买成功,需要进行补发货处理。

5.建议先检查此笔订单权益的发放状态,未发放则发放权益,成功后记录PurchaseOrderPayload等信息,用于后续检查权益发放状态。

6.发货成功后,应用需调用finishPurchase接口确认发货,以此通知IAP服务器更新商品的发货状态,完成购买流程。

7.发起请求时,需在请求参数FinishPurchaseParameter中携带PurchaseOrderPayload中的productType、purchaseToken、purchaseOrderId。

请求成功后,IAP服务器会将相应商品标记为已发货状态。对于消耗型商品,IAP服务器会将相应商品重新设置为可购买状态,用户即可再次购买该商品。对于非消耗型商品,用户购买后永久拥有,无法再次购买该商品。

typescript 复制代码
import { iap } from '@kit.IAPKit';
import { common } from '@kit.AbilityKit';
import { BusinessError } from '@kit.BasicServicesKit';
// JWTUtil为自定义类,可参见示例代码
import { JWSUtil } from '../common/JWSUtil';

queryPurchases() {
  const param: iap.QueryPurchasesParameter = {
    productType: iap.ProductType.CONSUMABLE,
    queryType: iap.PurchaseQueryType.UNFINISHED
  };
  const context: common.UIAbilityContext = getContext(this) as common.UIAbilityContext;
  iap.queryPurchases(context, param).then((res: iap.QueryPurchaseResult) => {
    console.info('Succeeded in querying purchases.');
    const purchaseDataList: string[] = res.purchaseDataList;
    if (purchaseDataList === undefined || purchaseDataList.length <= 0) {
      return;
    }
    for (let i = 0; i < purchaseDataList.length; i++) {
      const jwsPurchaseOrder: string = JSON.parse(purchaseDataList[i]).jwsPurchaseOrder;
      if (!jwsPurchaseOrder) {
        continue;
      }
      const purchaseStr = JWSUtil.decodeJwtObj(jwsPurchaseOrder);
      // 需自定义PurchaseOrderPayload类,包含的信息请参见PurchaseOrderPayload
      const purchaseOrderPayload = JSON.parse(purchaseStr) as PurchaseOrderPayload;
      // 处理发货
      // ...
      // 发货成功后向IAP Kit发送finishPurchase请求,确认发货,完成购买
      this.finishPurchase(purchaseOrderPayload);
    }
  }).catch((err: BusinessError) => {
    // 请求失败
    console.error(`Failed to query purchases. Code is ${err.code}, message is ${err.message}`);
  });
}

finishPurchase(purchaseOrder: PurchaseOrderPayload) {
  const finishPurchaseParam: iap.FinishPurchaseParameter = {
    productType: purchaseOrder.productType,
    purchaseToken: purchaseOrder.purchaseToken,
    purchaseOrderId: purchaseOrder.purchaseOrderId
  };
  const context: common.UIAbilityContext = getContext(this) as common.UIAbilityContext;
  iap.finishPurchase(context, finishPurchaseParam).then(() => {
    // 请求成功
    console.info('Succeeded in finishing purchase.');
  }).catch((err: BusinessError) => {
    // 请求失败
    console.error(`Failed to query purchases. Code is ${err.code}, message is ${err.message}`);
  });
}

单机应用权益发放(非消耗型商品)

用户在购买非消耗型商品后,将永久拥有该商品的权益。应用需要在用户购买非消耗型商品后,始终为其发放相关权益。

请在以下场景获取用户已购非消耗型商品的信息,并发放相关权益。

1.应用启动时。

2.购买请求(createPurchase)返回iap.IAPErrorCode.PRODUCT_OWNED或iap.IAPErrorCode.SYSTEM_ERROR时。

开发步骤

1.应用客户端向IAP Kit发起queryPurchases请求,获取用户已购非消耗型商品的订单状态信息。

在请求参数QueryPurchasesParameter中指定productType为iap.ProductType.NONCONSUMABLE,同时指定queryType为iap.PurchaseQueryType.CURRENT_ENTITLEMENT。当接口请求成功时,IAP Kit将返回一个QueryPurchaseResult对象,该对象包含承载了订单信息的PurchaseData的列表。

2.对每个PurchaseData.jwsPurchaseOrder进行解码验签。

3.验证成功可得到对应的PurchaseOrderPayload的JSON字符串,需要发放相关权益。

4.发放权益后,应用需调用finishPurchase接口确认发货,以此通知IAP服务器更新商品的发货状态,完成购买流程。

发起请求时,需在请求参数FinishPurchaseParameter中携带PurchaseOrderPayload中的productType、purchaseToken、purchaseOrderId。 请求成功后,IAP服务器会将相应商品标记为已发货。对于非消耗型商品,用户购买后永久拥有,无法再次购买该商品。

typescript 复制代码
import { iap } from '@kit.IAPKit';
import { common } from '@kit.AbilityKit';
import { BusinessError } from '@kit.BasicServicesKit';
// JWTUtil为自定义类,可参见示例代码
import { JWSUtil } from '../common/JWSUtil';

queryPurchases() {
  const param: iap.QueryPurchasesParameter = {
    productType: iap.ProductType.NONCONSUMABLE,
    queryType: iap.PurchaseQueryType.CURRENT_ENTITLEMENT
  };
  const context: common.UIAbilityContext = getContext(this) as common.UIAbilityContext;
  iap.queryPurchases(context, param).then((res: iap.QueryPurchaseResult) => {
    console.info('Succeeded in querying purchases.');
    const purchaseDataList: string[] = res.purchaseDataList;
    if (purchaseDataList === undefined || purchaseDataList.length <= 0) {
      return;
    }
    for (let i = 0; i < purchaseDataList.length; i++) {
      const jwsPurchaseOrder: string = JSON.parse(purchaseDataList[i]).jwsPurchaseOrder;
      if (!jwsPurchaseOrder) {
        continue;
      }
      // 对jwsPurchaseOrder进行解码验签
      const purchaseStr = JWSUtil.decodeJwtObj(jwsPurchaseOrder);
      // 需自定义PurchaseOrderPayload类,包含的信息请参见PurchaseOrderPayload
      const purchaseOrderPayload = JSON.parse(purchaseStr) as PurchaseOrderPayload;
      // 处理权益发放
      // ...
      // 发放权益后向IAP Kit发送finishPurchase请求,确认发货,完成购买
      if (purchaseOrderPayload && purchaseOrderPayload.finishStatus !== '1') {
        this.finishPurchase(purchaseOrderPayload);
      }
    }
  }).catch((err: BusinessError) => {
    // 请求失败
    console.error(`Failed to query purchases. Code is ${err.code}, message is ${err.message}`);
  });
}

finishPurchase(purchaseOrder: PurchaseOrderPayload) {
  const finishPurchaseParam: iap.FinishPurchaseParameter = {
    productType: purchaseOrder.productType,
    purchaseToken: purchaseOrder.purchaseToken,
    purchaseOrderId: purchaseOrder.purchaseOrderId
  };
  const context: common.UIAbilityContext = getContext(this) as common.UIAbilityContext;
  iap.finishPurchase(context, finishPurchaseParam).then(() => {
    // 请求成功
    console.info('Succeeded in finishing purchase.');
  }).catch((err: BusinessError) => {
    // 请求失败
    console.error(`Failed to query purchases. Code is ${err.code}, message is ${err.message}`);
  });
}

了解更多详情>>

访问应用内支付服务联盟官网

获取权益发放的开发指导文档

相关推荐
__Benco2 分钟前
OpenHarmony - 小型系统内核(LiteOS-A)(六)
人工智能·harmonyos
xq95272 小时前
鸿蒙next 获取验证码输入框 重磅来袭
harmonyos
别说我什么都不会2 小时前
【仓颉三方库】 数据解析——yaml4cj
harmonyos
zhangmeng3 小时前
一文带你读懂鸿蒙Stage模型开发运行期和编译期概念
harmonyos·arkts·arkui
梁下轻语的秋缘7 小时前
HarmonyOs学习 环境配置后 实验1:创建项目Hello World
学习·华为·harmonyos
少年的云河月7 小时前
OpenHarmony Camera开发指导(五):相机预览功能(ArkTS)
harmonyos·openharmony·camera·相机开发
Hello_MyDream7 小时前
鸿蒙语言基础
华为·harmonyos
杯莫停丶7 小时前
对象池模式在uniapp鸿蒙APP中的深度应用
uni-app·harmonyos·鸿蒙
simple_lau9 小时前
浅谈鸿蒙多线程
harmonyos·arkts·arkui