Android 华为应用内支付接入及问题

这段时间接入了华为应用内支付,遇到一些问题,特此记录哈!

华为支付主要支持的商品类型包括:消耗型商品、非消耗型商品、订阅型商品三种

  1. 消耗型商品:使用一次后即消耗掉,随使用减少,需要再次购买的商品。例:游戏货币,游戏道具等。
  2. 非消耗型商品:一次性购买,永久拥有,无需消耗。例:游戏中额外的游戏关卡、应用中无时限的高级会员等。
  3. 订阅型商品:用户购买后在一段时间内允许访问增值功能或内容,周期结束后自动续期购买下一期的服务。例:应用中有时限的高级会员,如视频月度会员。
  • 具体对接可参考流程官方文档
  • 大概流程就是去华为后台用包名/签名信息注册应用,生成appid和app secret这些,然后就是去“产品运营 > 商品管理”里面配置商品
  • 代码层面就集成sdk和添加混淆等逻辑,如果需要后台执行消耗商品逻辑这些就需要对接服务端的集成开发,否则客户端自己消耗商品

问题点:

  • 如何进行沙盒测试
  1. 需要在AppGallery Connect中的“用户与访问”中添加测试帐号,这些测试帐号必须都是真实的华为帐号,沙盒测试帐号添加完成之后需要30min~1h才能生效,实测10min钟左右就生效了。
  2. 开发的app如果此前没有在华为应用商店AppGallery Connect上架过版本,只需要确保测试包的versionCode大于0即可; 如果已有上架的版本,则测试包的versionCode需要大于上架版本的versionCode。
  3. 满足上述条件即可在手机的hms core里面登录测试账号,然后调用支付,理论上就可看到沙盒测试的弹框和文案提示,然后支付时付钱流程就被跳过了,如下图:


    1.jpg

    2.jpg
  • 沙盒测试能力未生效
  1. 检查测试帐号是否正确,登录的账号是否是已添加且生效的测试账号
  2. 解开apk包检查app的versionCode版本是否大于线上版本的versionCode
  3. 去手机设置的应用管理里面搜索"hms code",点开"应用的其他设置"进入hms core的设置界面进行版本更新,保证更新到最新版本
  4. 上面一步还可以替换成去对应应用商店搜索"hms code"执行应用更新,或者有打开按钮也可以进入到设置界面,如果本机的应用商店不满足,可以下载华为的应用商店apk就可以执行hms core的打开设置页操作
  5. 这个设置界面还可以进行订阅型的修改和暂停及取消等更多操作,可以熟练操作便于排查问题
  • 我遇到的问题是消耗型可以进入沙盒模式,但是订阅型一直不进入沙盒模式,后来提交了华为工单,回复加华为的技术开发QQ,他提示我一直测的订阅产品还在有效期内,再执行可能不得进沙盒,建议我新建一个订阅型产品重新测看,我试了一把,建了个新订阅型,还是没进沙盒,但是过了一段时间,突然进去就是沙盒模式了,我只能怀疑是新建了订阅型产品,要沙盒模式生效可能需要半小时左右,虽然技术老兄也没肯定我的猜想!

  • 订单消耗这个问题

  1. 初始化启动需要调用obtainOwnedPurchases查询priceType = 0(消耗型商品)是否有未消耗的订单,否则支付会返回60051(ORDER_PRODUCT_OWNED)无法再次支付,需要先消耗掉(可选择客户端自己调用consumeOwnedPurchase或者后台调用接口消耗)
  2. 支付中如果返回了60051需要先把订单消耗掉再重新发起支付,支付成功后也要及时消耗掉才行
  • 华为登录/支付回调都走的onActivityResult,不方便的问题
  1. 针对这个问题,参考微信sdk的操作,可以设计一个透明的activity,然后回调操作完成后记得finish掉即可,主要设置哈activity的主题即可
  2. 附上透明设置相关:
class XXHuaweiActivity: AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        val mainFy = FrameLayout(this)
        mainFy.setBackgroundColor(Color.TRANSPARENT)
        setContentView(mainFy)
    }

  override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
        super.onActivityResult(requestCode, resultCode, data)
        ......
  }
  
}
<application>
        <activity
            android:name=".XXHuaweiActivity"
            android:theme="@style/Transparent" />
    </application>
<resources>
    <style name="Transparent" parent="@style/Theme.AppCompat.NoActionBar">
        <item name="android:windowBackground">#00000000</item>
        <item name="android:windowIsTranslucent">true</item>
        <item name="android:windowNoTitle">true</item>
    </style>
</resources>

附上消耗型商品的客户端补单消耗流程代码

/**
     * 消耗型商品的客户端补单消耗流程
     * @param iapPublicKey IAP公钥
     */
    fun supplementaryConsumeOrderProcess(activity: Activity, iapPublicKey: String, callBack: ConsumePurchaseCallBack) {
        printLog("supplementaryConsumeOrderProcess start")
        val context = WeakReference(activity)
        val ownedPurchasesReq = OwnedPurchasesReq()// 构造一个OwnedPurchasesReq对象
        ownedPurchasesReq.priceType = 0// priceType: 0:消耗型商品; 1:非消耗型商品; 2:订阅型商品
        // 获取调用接口的Activity对象,调用obtainOwnedPurchases接口获取所有已购但未发货的消耗型商品的购买信息
        val task = Iap.getIapClient(context.get()).obtainOwnedPurchases(ownedPurchasesReq)
        task.addOnSuccessListener { result ->
            printLog("supplementaryConsumeOrderProcess $result")
            // 获取接口请求成功的结果
            if (result?.inAppPurchaseDataList != null) {
                for (i in result.inAppPurchaseDataList.indices) {
                    val inAppPurchaseData = result.inAppPurchaseDataList[i]
                    val inAppSignature = result.inAppSignature[i]
                    printLog("supplementaryConsumeOrderProcess inAppPurchaseData:$inAppPurchaseData inAppSignature:$inAppSignature")
                    // 使用应用的IAP公钥验证inAppPurchaseData的签名数据
                    // 如果验签成功,必须校验InAppPurchaseData中的productId、price、currency等信息的一致性
                    // 验证一致后,确认每个商品的购买状态。确认商品已支付后,检查此前是否已发过货,未发货则进行发货操作。发货成功后执行消耗操作
                    try {
                        val inAppPurchaseDataBean = InAppPurchaseData(inAppPurchaseData)
                        val purchaseState = inAppPurchaseDataBean.purchaseState
                        var checkFlag = false
                        val orderInfoBean = OrderInfoBean(inAppPurchaseDataBean.productId, inAppPurchaseDataBean.price, inAppPurchaseDataBean.currency)
                        printLog("supplementaryConsumeOrderProcess OrderInfoBean $orderInfoBean")
                        if (CheckInAppPurchaseData.checkSuccessOrder(inAppPurchaseData, inAppSignature, iapPublicKey, orderInfoBean)) {
                            printLog("supplementaryConsumeOrderProcess checkSuccessOrder success")
                            checkFlag = true
                        } else {
                            printLog("supplementaryConsumeOrderProcess checkSuccessOrder fail")
                        }
                        if (checkFlag && purchaseState == InAppPurchaseData.PurchaseState.PURCHASED) {//已购买 //直接消耗
                            // 构造一个ConsumeOwnedPurchaseReq对象
                            val req = ConsumeOwnedPurchaseReq()
                            var purchaseToken: String? = ""
                            try {
                                // 从购买信息inAppPurchaseData中获取purchaseToken。inAppPurchaseData可从一次支付请求或者请求obtainOwnedPurchases接口获取
                                val inAppPurchaseDataBean = InAppPurchaseData(inAppPurchaseData)
                                purchaseToken = inAppPurchaseDataBean.purchaseToken
                                printLog("supplementaryConsumeOrderProcess purchaseToken:$purchaseToken")
                            } catch (e: JSONException) {
                                e.printStackTrace()
                            }
                            req.purchaseToken = purchaseToken
                            // 调用consumeOwnedPurchase接口
                            val task = Iap.getIapClient(context.get()).consumeOwnedPurchase(req)
                            task.addOnSuccessListener {// 获取接口请求成功时的结果信息
                                printLog("ConsumeOwnedPurchaseReq Success ${it.returnCode}")
                                callBack.consumePurchaseResult(it)
                            }.addOnFailureListener { e ->
                                if (e is IapApiException) {
                                    val status = e.status
                                    val returnCode = e.statusCode
                                    printLog("ConsumeOwnedPurchaseReq fail status:$status  returnCode:$returnCode")
                                } else {
                                    // 其他外部错误
                                }
                                printLog("ConsumeOwnedPurchaseReq fail $e")
                                callBack.consumePurchaseFail(e.message)
                            }
                        }
                    } catch (e: JSONException) {
                        e.printStackTrace()
                    }
                }
            }
        }.addOnFailureListener { e ->
            if (e is IapApiException) {
                val status: Status = e.status
                val returnCode = e.statusCode
                printLog("ConsumeOwnedPurchaseReq status:$status  returnCode:$returnCode")
            } else {
                // 其他外部错误
            }
            printLog("ConsumeOwnedPurchaseReq ${e.toString()}")
        }
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容