Java服务端实现Google Pay支付功能

Hello大家好,鉴于之前第一篇的博客关于Google pay支付,可能很多读者对整体功能不太清晰,然后最近对代码进行了重构,希望接下来这篇能够帮助读者们更快捷、更方便的集成Google Pay功能,好了话不多说,直接进入正题:

Google支付所需配置

可参考官网,创建应用、添加权限、生成json文件,这里我就不详细多说了,这种配置获取可根据官网、或者网上找的方法一大堆。像有些读者出现调用API时出现400的错误,基本都是配置错误导致的,主要是权限没配置好。

谷歌支付回调

谷歌开发这通知功能(Pub/Sub),这个我也不详细多说了,可参考官网。(如果不懂可以看之前的博客有参考地址)

好的接下来咱们直接进入代码层面,先说代码层面之前,我先给大家梳理下流程:
支付流程图
下单逻辑读者们应该没啥问题吧,注意看里面的订单处理(1)和订单处理(2),分为客户端主动回调请求订单处理和Google回调服务端通知订单处理,订单处理(1)相信流程大家都没问题,但是订单处理(2),大家想想,如果Google回调通知订单,服务端怎么获取该用户发起的系统订单号呢?注意:Google下单时时支持字段透传的,也就是说,在步骤③的时候吧步骤②返回的系统订单号放入,那么这个Google支付完成后,回调Google将会透传这个字段给服务端,这个下面代码会讲(各位读者不要急)~,那这样这个回调通知是不是就产生了Google订单和系统订单关联关系了!

接下来看代码实现,具体下单逻辑我就不讲述了,这是读者们实现的,以下会讲解怎么集成Google Pay API。

代码集成实现

1、所需依赖
Google pay SDK依赖和认证依赖:

<!-- google pay依赖(使用最新版本)可取maven仓库查看,这边是GitHub发布的新版本 -->
        <dependency>
            <groupId>com.google.apis</groupId>
            <artifactId>google-api-services-androidpublisher</artifactId>
            <version>v3-rev20230815-2.0.0</version>
        </dependency>
<!-- google client依赖 https://mvnrepository.com/artifact/com.google.auth/google-auth-library-oauth2-http -->
        <dependency>
            <groupId>com.google.auth</groupId>
            <artifactId>google-auth-library-oauth2-http</artifactId>
            <version>1.19.0</version>
        </dependency>

2、Client对象创建

	    //存放多个实例的 AndroidPublisher Map
    public static Map<String, AndroidPublisher> ANDROID_PUBLISHER_MAP = new ConcurrentHashMap<>();
	
	/**
     * 必须实现对象创建逻辑,可能会存在空
     *
     * @param appName 应用唯一标识(单例模式创建)
     * @return
     */
    @Override
    public synchronized AndroidPublisher getAuthInstance(String appName) throws PaymentCoreException {
        AndroidPublisher androidPublisher = ANDROID_PUBLISHER_MAP.get(appName);
        if (androidPublisher != null) {
            return androidPublisher;
        }
        //工厂策略模式读取,这个是我存放的Google配置信息,大家可忽略,可自己实现
        GooglePayConfigEntity googlePayConfig = paymentConfigFactory.getPaymentConfig(ConstantUtil.GOOGLE_PAYMENT_CONFIG_IMPL, appName);
        InputStream input = null;
        try {
            //获取用户凭证 json文件流
            input = FileUtil.getImageStreamByUrl(googlePayConfig.getJsonFileUrl());
            //生成Credentials对象
            GoogleCredentials credentials = GoogleCredentials.fromStream(input);
            HttpTransport transport = GoogleNetHttpTransport.newTrustedTransport();
            HttpCredentialsAdapter httpCredentialsAdapter = new HttpCredentialsAdapter(credentials);
            //创建 AndroidPublisher对象
            androidPublisher = new AndroidPublisher.Builder(
                    transport,
                    GsonFactory.getDefaultInstance(),
                    httpCredentialsAdapter).build();
        } catch (Exception e) {
            throw new PaymentCoreException("[Google] client instance create exception: " + e.getMessage());
        } finally {
            FileUtil.closeStream(input);
        }
        if (androidPublisher != null) {
            //放入对象容器
            ANDROID_PUBLISHER_MAP.put(appName, androidPublisher);
        }
        return androidPublisher;
    }

3、以下是各种API处理方法:

    /**
     * 一次性(单次)购买订单对象获取
     *
     * @param appName       应用唯一标识
     * @param packageName   包名称
     * @param productId     商品Id
     * @param purchaseToken 谷歌购买凭据
     * @return
     */
    public ProductPurchase getOneTimeGoogleOrder(String appName, String packageName, String productId, String purchaseToken) throws PaymentCoreException {
        AndroidPublisher authInstance = this.getAuthInstance(appName);
        if (authInstance == null) {
            throw new PaymentCoreException("[Google] AndroidPublisher instance is empty");
        }
        ProductPurchase products = null;
        try {
            products = authInstance.purchases().products().get(packageName, productId, purchaseToken).execute();
            log.info("Products order info: {}", JSONUtil.toJsonStr(products));
        } catch (IOException e) {
            throw new PaymentCoreException("[Google] Get google products order info exception: " + e.getMessage());
        }
        return products;
    }

	/**
     * 订阅购买订单获取(V2版本),最新订阅请使用V2,官方对V1已经不维护了,但是可以用哈!
     *
     * @param appName       应用唯一标识
     * @param packageName   包名称
     * @param purchaseToken 购买凭据
     * @return
     */
    public SubscriptionPurchaseV2 getSubscriptionGoogleOrderV2(String appName, String packageName, String purchaseToken) throws PaymentCoreException {
        AndroidPublisher authInstance = this.getAuthInstance(appName);
        if (authInstance == null) {
            throw new PaymentCoreException("[Google] AndroidPublisher instance is empty");
        }
        SubscriptionPurchaseV2 subscriptionsV2 = null;
        try {
            subscriptionsV2 = authInstance.purchases().subscriptionsv2().get(packageName, purchaseToken).execute();
            log.info("SubscriptionsV2 order info: {}", JSONUtil.toJsonStr(subscriptionsV2));
        } catch (IOException e) {
            throw new PaymentCoreException("[Google] Get google subscriptions v2 order info exception: " + e.getMessage());
        }
        return subscriptionsV2;
    }

	/**
     * 取消订阅方法
     *
     * @param appName        应用唯一标识
     * @param packageName    包名称
     * @param subscriptionId 订阅Id,其实就是商品Id
     * @param purchaseToken  购买凭据
     * @return
     */
    public void cancelSubscription(String appName, String packageName, String subscriptionId, String purchaseToken) throws PaymentCoreException {
        AndroidPublisher authInstance = this.getAuthInstance(appName);
        if (authInstance == null) {
            throw new PaymentCoreException("[Google] AndroidPublisher instance is empty");
        }
        try {
            authInstance.purchases().subscriptions().cancel(packageName, subscriptionId, purchaseToken).execute();
            log.info("Cancel subscription success");
        } catch (IOException e) {
            throw new PaymentCoreException("[Google] Cancel google subscriptions exception: " + e.getMessage());
        }
    }

    /**
     * 订阅退款方法
     *
     * @param appName        应用唯一标识
     * @param packageName    包名称
     * @param subscriptionId 订阅Id,其实就是商品Id
     * @param purchaseToken  购买凭据
     * @return
     */
    public void refundSubscription(String appName, String packageName, String subscriptionId, String purchaseToken) throws PaymentCoreException {
        AndroidPublisher authInstance = this.getAuthInstance(appName);
        if (authInstance == null) {
            throw new PaymentCoreException("[Google] AndroidPublisher instance is empty");
        }
        try {
            authInstance.purchases().subscriptions().refund(packageName, subscriptionId, purchaseToken).execute();
            log.info("Refund subscription success");
        } catch (IOException e) {
            throw new PaymentCoreException("[Google] Refund google subscriptions exception: " + e.getMessage());
        }
    }

    /**
     * 订阅撤销方法,但是不建议我们实现,让客户自己在Google play商店操作订阅的状态
     *
     * @param appName        应用唯一标识
     * @param packageName    包名称
     * @param subscriptionId 订阅Id,其实就是商品Id
     * @param purchaseToken  购买凭据
     * @return
     */
    public void revokeSubscription(String appName, String packageName, String subscriptionId, String purchaseToken) throws PaymentCoreException {
        AndroidPublisher authInstance = this.getAuthInstance(appName);
        if (authInstance == null) {
            throw new PaymentCoreException("[Google] AndroidPublisher instance is empty");
        }
        try {
            authInstance.purchases().subscriptions().revoke(packageName, subscriptionId, purchaseToken).execute();
            log.info("Revoke subscription success");
        } catch (IOException e) {
            throw new PaymentCoreException("[Google] Revoke google subscriptions exception: " + e.getMessage());
        }
        return;
    }

    /**
     * 订阅购买订单获取(旧版本),跟V2一样的逻辑, 但是google官网说明方法已经废弃,不维护
     *
     * @param appName        应用唯一标识
     * @param packageName    包名称
     * @param subscriptionId 订阅Id,其实就是商品Id
     * @param purchaseToken  购买凭据
     * @return
     */
    public SubscriptionPurchase getSubscriptionGoogleOrder(String appName, String packageName, String subscriptionId, String purchaseToken) throws PaymentCoreException {
        AndroidPublisher authInstance = this.getAuthInstance(appName);
        if (authInstance == null) {
            throw new PaymentCoreException("[Google] AndroidPublisher instance is empty");
        }
        SubscriptionPurchase subscriptions = null;
        try {
            subscriptions = authInstance.purchases().subscriptions().get(packageName, subscriptionId, purchaseToken).execute();
            log.info("Subscriptions order info: {}", JSONUtil.toJsonStr(subscriptions));
        } catch (IOException e) {
            throw new PaymentCoreException("[Google] Get google subscriptions order info exception: " + e.getMessage());
        }
        return subscriptions;
    }

4、介绍以下重点参数:
appName: 用于SaaS环境区分项目的,一个环境一个appName。
packageName: Google配置的包应用名称。客户端传入。
subscriptionId: Google配置的商品Id,唯一标识。客户端传入。
purchaseToken: 用户购买的凭证。客户端传入。

5、判断订单是否成功:

单次(消耗):

/**
 * @author xxx  应用内商品的消耗状态。可能的值包括:0。尚未消耗 1. 已使用 (Products对象中的值)
 * @version 1.0
 * @date 2023/8/23 20:13
 */
public enum ConsumptionStateEnum {

    NOT_CONSUMED(0, "Yet to be consumed"),
    CONSUMED(1, "Consumed");

    public int code;
    public String msg;

    ConsumptionStateEnum(int code, String msg) {
        this.code = code;
        this.msg = msg;
    }

    public int getCode() {
        return code;
    }

    public String getMsg() {
        return msg;
    }
}

/**
 * @author xxx  订单的购买状态 可能的值包括:0。购买 1. 已取消 2. 待处理 (Products对象中的值)
 * @version 1.0
 * @date 2023/8/23 19:59
 */
public enum PurchaseStateEnum {

    PURCHASED(0, "Purchased"),
    CANCELED(1, "Canceled"),
    PENDING(2, "Pending");

    public int code;
    public String msg;

    PurchaseStateEnum(int code, String msg) {
        this.code = code;
        this.msg = msg;
    }

    public int getCode() {
        return code;
    }

    public String getMsg() {
        return msg;
    }
}

    /**
     * 校验单次购买是否成功
     *
     * @param productPurchase
     * @return
     */
    public boolean checkOneTimeGoogleOrderSuccess(ProductPurchase productPurchase) {
        if (ConsumptionStateEnum.CONSUMED.getCode() == productPurchase.getConsumptionState()
                && PurchaseStateEnum.PURCHASED.getCode() == productPurchase.getPurchaseState()) {
            return true;
        }
        return false;
    }

    /**
     * 校验单次购买是否失败
     *
     * @param productPurchase
     * @return
     */
    public boolean checkOneTimeGoogleOrderFail(ProductPurchase productPurchase) {
        if (PurchaseStateEnum.CANCELED.getCode() == productPurchase.getPurchaseState()) {
            return true;
        }
        return false;
    }

订阅

/**
 * @author xxx订阅的确认状态。 (subscriptionsv2)对象中的 acknowledgementState
 * ACKNOWLEDGEMENT_STATE_UNSPECIFIED	未指定的确认状态。
 * ACKNOWLEDGEMENT_STATE_PENDING	订阅尚未确认。
 * ACKNOWLEDGEMENT_STATE_ACKNOWLEDGED	已确认订阅。
 * @version 1.0
 * @date 2023/8/23 20:55
 */
public enum SubscriptionAckStateEnum {

    ACKNOWLEDGEMENT_STATE_UNSPECIFIED("ACKNOWLEDGEMENT_STATE_UNSPECIFIED", "Unspecified acknowledgement state"),
    ACKNOWLEDGEMENT_STATE_PENDING("ACKNOWLEDGEMENT_STATE_PENDING", "The subscription is not acknowledged yet"),
    ACKNOWLEDGEMENT_STATE_ACKNOWLEDGED("ACKNOWLEDGEMENT_STATE_ACKNOWLEDGED", "The subscription is acknowledged");

    public String code;
    public String msg;

    SubscriptionAckStateEnum(String code, String msg) {
        this.code = code;
        this.msg = msg;
    }

    public String getCode() {
        return code;
    }

    public String getMsg() {
        return msg;
    }
}

/**
 * @author xxx订阅的当前状态。 (subscriptionsv2)对象中的 subscriptionState
 * SUBSCRIPTION_STATE_UNSPECIFIED	未指定订阅状态。
 * SUBSCRIPTION_STATE_PENDING	已创建订阅,但在注册期间正在等待付款。在此状态下,所有商品均正在等待付款。
 * SUBSCRIPTION_STATE_ACTIVE	订阅处于有效状态。- (1) 如果订阅是自动续订型方案,则至少有一项是自动续订状态并且未过期。- (2) 如果订阅是预付费方案,至少有一项不会过期。
 * SUBSCRIPTION_STATE_PAUSED	订阅已暂停。只有在订阅是自动续订型方案时,才会显示该状态。在此状态下,所有商品均处于暂停状态。
 * SUBSCRIPTION_STATE_IN_GRACE_PERIOD	订阅处于宽限期。只有在订阅是自动续订型方案时,才会显示该状态。在此状态下,所有商品均处于宽限期。
 * SUBSCRIPTION_STATE_ON_HOLD	订阅已暂停(已暂停)。只有在订阅是自动续订型方案时,才会显示该状态。在此状态下,所有商品均处于保全状态。
 * SUBSCRIPTION_STATE_CANCELED	订阅已取消,但尚未过期。只有在订阅是自动续订型方案时,才会显示该状态。所有商品的 autoRenewEnabled 都设为 false。
 * SUBSCRIPTION_STATE_EXPIRED	订阅已过期。所有内容的过期时间都是过去的时间。
 * @version 1.0
 * @date 2023/8/23 20:37
 */
public enum SubscriptionStateEnum {

    SUBSCRIPTION_STATE_UNSPECIFIED("SUBSCRIPTION_STATE_UNSPECIFIED", "Unspecified subscription state"),
    SUBSCRIPTION_STATE_PENDING("SUBSCRIPTION_STATE_PENDING", "Subscription was created"),
    SUBSCRIPTION_STATE_ACTIVE("SUBSCRIPTION_STATE_ACTIVE", "Subscription is active"),
    SUBSCRIPTION_STATE_PAUSED("SUBSCRIPTION_STATE_PAUSED", "Subscription is paused"),
    SUBSCRIPTION_STATE_IN_GRACE_PERIOD("SUBSCRIPTION_STATE_IN_GRACE_PERIOD", "Subscription is in grace period"),
    SUBSCRIPTION_STATE_ON_HOLD("SUBSCRIPTION_STATE_ON_HOLD", "Subscription is on hold (suspended)"),
    SUBSCRIPTION_STATE_EXPIRED("SUBSCRIPTION_STATE_EXPIRED", "Subscription is expired"),
    SUBSCRIPTION_STATE_CANCELED("SUBSCRIPTION_STATE_CANCELED", "Subscription is canceled but not expired yet");

    public String code;
    public String msg;

    SubscriptionStateEnum(String code, String msg) {
        this.code = code;
        this.msg = msg;
    }

    public String getCode() {
        return code;
    }

    public String getMsg() {
        return msg;
    }

}

    /**
     * 校验订阅订单是否支付成功
     *
     * @param subscriptionsV2
     * @return
     */
    public boolean checkSubscriptionSuccess(SubscriptionPurchaseV2 subscriptionsV2) {
        String acknowledgementState = subscriptionsV2.getAcknowledgementState();
        if (SubscriptionAckStateEnum.ACKNOWLEDGEMENT_STATE_ACKNOWLEDGED.getCode().equals(acknowledgementState)) {
            return true;
        }
        return false;
    }

以上就是集成的代码实现。

回调处理

可看官网文档,但是这边我都给你弄成JavaBean了,也没必要去看(以下忽略getter/setter):

回调实体:

/**
 * @author xxx 谷歌开发者通知对象
 * @version 1.0
 * @date 2023/10/8 16:12
 */
public class DeveloperNotification implements Serializable {
    /**
     * 此通知的版本。最初,此值为“1.0”。此版本与其他版本字段不同。
     */
    private String version;

    /**
     * 与此通知相关的应用的软件包名称(例如“com.some.thing”)。
     */
    private String packageName;

    /**
     * 事件发生的时间戳,以从公元纪年开始计算的毫秒数表示。
     */
    private long eventTimeMillis;

    /**
     * 如果此字段存在,则此通知与订阅相关,并且此字段包含与订阅相关的其他信息。请注意,此字段与 testNotification 和 oneTimeProductNotification 互斥。
     */
    private SubscriptionNotification subscriptionNotification;

    /**
     * 如果此字段存在,则此通知与一次性购买相关,并且此字段包含与购买交易相关的其他信息。请注意,此字段与 testNotification 和 subscriptionProductNotification 互斥。
     */
    private OneTimeProductNotification oneTimeProductNotification;

    /**
     * 如果此字段存在,则此通知与测试发布相关。这些只通过 Google Play 管理中心发送。请注意,此字段与 subscriptionNotification 和 oneTimeProductNotification 互斥。
     */
    private TestNotification testNotification;
}

/**
 * @author xxx 谷歌开发者回调订阅通知信息
 * @version 1.0
 * @date 2023/10/8 15:55
 */
public class SubscriptionNotification implements Serializable {
    /**
     * 通知的版本。最初,此值为“1.0”。此版本与其他版本字段不同。
     */
    private String version;

    /**
     * 订阅的 notificationType 可以具有以下值:
     * (1) SUBSCRIPTION_RECOVERED - 从帐号保留状态恢复了订阅。
     * (2) SUBSCRIPTION_RENEWED - 续订了处于活动状态的订阅。
     * (3) SUBSCRIPTION_CANCELED - 自愿或非自愿地取消了订阅。如果是自愿取消,在用户取消时发送。
     * (4) SUBSCRIPTION_PURCHASED - 购买了新的订阅。
     * (5) SUBSCRIPTION_ON_HOLD - 订阅已进入帐号保留状态(如果已启用)。
     * (6) SUBSCRIPTION_IN_GRACE_PERIOD - 订阅已进入宽限期(如果已启用)。
     * (7) SUBSCRIPTION_RESTARTED - 用户已通过 Play > 帐号 > 订阅恢复了订阅。订阅已取消,但在用户恢复时尚未到期。如需了解详情,请参阅恢复。
     * (8) SUBSCRIPTION_PRICE_CHANGE_CONFIRMED - 用户已成功确认订阅价格变动。
     * (9) SUBSCRIPTION_DEFERRED - 订阅的续订时间点已延期。
     * (10) SUBSCRIPTION_PAUSED - 订阅已暂停。
     * (11) SUBSCRIPTION_PAUSE_SCHEDULE_CHANGED - 订阅暂停计划已更改。
     * (12) SUBSCRIPTION_REVOKED - 用户在到期时间之前已撤消订阅。
     * (13) SUBSCRIPTION_EXPIRED - 订阅已到期。
     */
    private int notificationType;

    /**
     * 购买订阅时向用户设备提供的令牌。
     */
    private String purchaseToken;

    /**
     * 所购买订阅的商品 ID(例如“monthly001”)。
     */
    private String subscriptionId; 
}    

/**
 * @author xxx 谷歌开发者回调单次通知信息
 * @version 1.0
 * @date 2023/10/8 16:09
 */
public class OneTimeProductNotification implements Serializable {
    /**
     * 通知的版本。最初,此值为“1.0”。此版本与其他版本字段不同。
     */
    private String version;

    /**
     * 通知的类型。它可以具有以下值:
     * (1) ONE_TIME_PRODUCT_PURCHASED - 用户成功购买了一次性商品。
     * (2) ONE_TIME_PRODUCT_CANCELED - 用户已取消待处理的一次性商品购买交易。
     */
    private int notificationType;

    /**
     * 购买订阅时向用户设备提供的令牌。
     */
    private String purchaseToken;

    /**
     * 购买的一次性商品的商品 ID(例如“sword_001”)。
     */
    private String sku;
}

/**
 * @author xxx 订阅通知类型,具体描述可参考SubscriptionNotification实体
 * @version 1.0
 * @date 2023/10/8 19:15
 */
public enum SubscriptionNotifyTypeEnum {

    SUBSCRIPTION_RECOVERED(1),
    SUBSCRIPTION_RENEWED(2),
    SUBSCRIPTION_CANCELED(3),
    SUBSCRIPTION_PURCHASED(4),
    SUBSCRIPTION_ON_HOLD(5),
    SUBSCRIPTION_IN_GRACE_PERIOD(6),
    SUBSCRIPTION_RESTARTED(7),
    SUBSCRIPTION_PRICE_CHANGE_CONFIRMED(8),
    SUBSCRIPTION_DEFERRED(9),
    SUBSCRIPTION_PAUSED(10),
    SUBSCRIPTION_PAUSE_SCHEDULE_CHANGED(11),
    SUBSCRIPTION_REVOKED(12),
    SUBSCRIPTION_EXPIRED(13);

    private int type;

    SubscriptionNotifyTypeEnum(int type) {
        this.type = type;
    }

    public int getType() {
        return type;
    }
}    

Google回调接口实现:

/**
     * 谷歌回调通知,订单处理
     *
     * @param body
     */
    @RequestMapping(value = "/notify/payment/google", method = RequestMethod.POST)
    public void googleNotify(@RequestBody(required = false) byte[] body) {
        String bodyStr = new String(body, "utf-8");
        //转成JSON
            /**  参数信息
             *{
             *   "message": {
             *     "attributes": {
             *       "key": "value"
             *     },
             *     "data": "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
             *     "messageId": "136969346945"
             *   },
             *   "subscription": "projects/myproject/subscriptions/mysubscription"
             * }
             */
            JSONObject bodyJson = JSONObject.parseObject(URLDecoder.decode(bodyStr, "utf-8"));
            //base64解码data
            String data = EncryptionUtil.base64Decode(bodyJson.getJSONObject("message").getString("data"));
            DeveloperNotification developerNotification = JSONObject.parseObject(data, DeveloperNotification.class);
			//判断developerNotification对象中有哪个对象
			if (subscriptionNotification != null) {
                //为订阅通知,调用上述的方法获取订单校验即可
            }
            if (oneTimeProductNotification != null) {
                //为单次通知,调用上述的方法获取订单校验即可
            }
    }

订阅订单处理:

注意,里面包含了流程图中订单处理(2)获取系统订单的方式

/**
     * 订阅回调通知
     *
     * @param subscriptionNotification
     * @param packageName
     * @param appName
     */
    private void subscriptionNotify(SubscriptionNotification subscriptionNotification, String packageName, String appName) {
        int notificationType = subscriptionNotification.getNotificationType();
        String purchaseToken = subscriptionNotification.getPurchaseToken();
        //获取订阅订单信息
        try {
            SubscriptionPurchaseV2 subscriptionGoogleOrderV2 = getSubscriptionGoogleOrderV2(appName, packageName, purchaseToken);
            //判断订单状态
            if (!googlePayment.checkSubscriptionSuccess(subscriptionGoogleOrderV2)) {
                return;
            }
            //判断通知类型
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_PURCHASED.getType() == notificationType) {
                //购买了新订阅,如果是上面流程图(2)处理,教你们怎么获取系统订单,让客户端设置相应的属性即可
                ExternalAccountIdentifiers externalAccountIdentifiers = subscriptionGoogleOrderV2.getExternalAccountIdentifiers();
                String obfuscatedExternalAccountId = externalAccountIdentifiers.getObfuscatedExternalAccountId();
                String obfuscatedExternalProfileId = externalAccountIdentifiers.getObfuscatedExternalProfileId();
                String externalAccountId = externalAccountIdentifiers.getExternalAccountId();
                responseVO = newSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_CANCELED.getType() == notificationType) {
                //取消了订阅
                responseVO = cancelSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_RENEWED.getType() == notificationType) {
                //续订了订阅
                //根据现在请求谷歌的googleOrderId获取续订包的订单,获取用户跟productId
                        //续订的谷歌订单是有规则的,比如你第一次购买产生的订单是GPA.123-456-789,那第一次续订的产生的订单号是GPA.123-456-789..0,第二次GPA.123-456-789..1,依次类推。
                        //续订这里还有一个坑,当订阅包降级购买的时候,谷歌并不是通知你购买了新订阅包而是续订,比如你购买的高级订阅包是GPA.123-456-789,当你切换低级的订阅包的时候订单号为GPA.111-222-333..0,你会发现订单号变了,这时候就难办了,拿不到之前用户的信息(谷歌返回的订单参数是没有项目里面用户信息的),经过联调发现,他们两个订单的linkedPurchaseToken是一样的,可以参考上图中参数的介绍
                        //当降级和自动续订的时候都会走这个通知,所以里面有两个逻辑,需要自己去判断,这两者的不同上面都写的很清楚了。
                responseVO = renewSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_ON_HOLD.getType() == notificationType) {
                //订阅进入保留状态,用户拒绝付款
                responseVO = onHoldSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_RECOVERED.getType() == notificationType) {
                //从保留状态恢复了订阅
                responseVO = recoverSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_RESTARTED.getType() == notificationType) {
                //重新注册了订阅,用户已通过“Play”>“帐号”>“订阅”重新激活其订阅(需要选择使用订阅恢复功能)
                responseVO = restartSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_REVOKED.getType() == notificationType) {
                //用户撤销订阅
                responseVO = revokeSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_EXPIRED.getType() == notificationType) {
                //订阅过期
                responseVO = expiredSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_IN_GRACE_PERIOD.getType() == notificationType) {
                //订阅进入宽限期
                responseVO = inGracePeriodSubscription(subscriptionGoogleOrderV2);
            }
            if (SubscriptionNotifyTypeEnum.SUBSCRIPTION_PAUSED.getType() == notificationType) {
                //订阅暂停
                responseVO = pauseSubscription(subscriptionGoogleOrderV2);
            }
        } catch (PaymentCoreException e) {
            log.error("Google get subscription order v2 exception: {}", e.getMessage());
            e.printStackTrace();
        } finally {
            //记录处理结果
        }
    }

所有的代码均已讲解,各位读者对这个支付功能还有什么疑惑呢,在评论区探讨的,码字不易~讲解不易!!!点个赞再走。记住,我还是那个会撩头发的程序猿!!!

转载请标明出处谢谢大家~

文章出处登录后可见!

已经登录?立即刷新

共计人评分,平均

到目前为止还没有投票!成为第一位评论此文章。

(2)
乘风的头像乘风管理团队
上一篇 2023年12月14日
下一篇 2023年12月14日

相关推荐

此站出售,如需请站内私信或者邮箱!