如何验证MY服务器上的Android应用内结算交易?

时间:2011-12-05 00:41:13

标签: php android in-app-billing

我制作了一个Android应用,可以使用in-app-billing购买商品。购买商品后,可以轻松地在Android电子市场和手机之间同步交易 - 即可在应用中使用。但是,我需要我的服务器知道购买。提供特定于应用程序的数据的决定应该在我的服务器上进行,而不是在客户端应用程序中进行。

E.g。

  1. 用户从Android电子市场购买商品X.
  2. 交易数据Y发送给客户。
  3. 客户端将Y发送到我的服务器。
  4. 客户端要求服务器为X提供内容。
  5. 如果Y有效,服务器会提供内容 。如何实现这一目标?
  6. 问:如何验证来自Android客户端(可能源自Google服务器)的交易数据是不是假的?即黑客没有生成数据。

    Google服务器 - > Android客户端 - >我的服务器 - > Android客户端

    也许这更像是一个PHP问题而不是其他任何问题。究竟我的服务器脚本(PHP)应该怎么做才能验证检索到的数据是否真实?

3 个答案:

答案 0 :(得分:20)

使用openssl_verify($ data,$ signature,$ key)

变量$ data和$ signature应该使用https从android客户端发送到你的php服务器。该交易包含这两个项目。在您确认客户端上的交易之前将其发送到您的服务器。(请参阅此处的文档 - http://developer.android.com/guide/market/billing/billing_integrate.html

变量$ key是您的发布商帐户中提供的Google公开密钥,来自Licensing&应用内结算面板。复制公钥并在PHP代码中使用它,最好使用您在服务器上安装的配置文件而不是实际的PHP代码。

如果openssl_verify调用成功,则应将订单号存储在服务器上,并确保它们是唯一的,以便无法重播。请注意,单个数据收据和签名对可以包含许多订单号,但通常只有一个订单。

答案 1 :(得分:11)

我们使用AndroidBillingLibrary

将其作为项目安装在Eclipse中,让您的项目将其作为库导入。

我们实施了BillingController.IConfiguration,类似于

import net.robotmedia.billing.BillingController;

public class PhoneBillingConfiguration implements BillingController.IConfiguration{
    @Override
    public byte[] getObfuscationSalt() {
        return new byte[] {1,-2,3,4,-5,6,-7,theseshouldallberandombyteshere,8,-9,0};
    }

    @Override
    public String getPublicKey() {
        return "superlongstringhereIforgothowwemadethis";
    }
}

然后,对于我们的申请,我们扩展了Application

public class LocalizedApplication extends Application {

    @Override
    public void onCreate() {
        super.onCreate();

//      BillingController.setDebug(true);
        BillingController.setConfiguration(new PhoneBillingConfiguration());
    }
}

AndroidManifest包含此(以及所有其他内容)

<application
    android:icon="@drawable/icon"
    android:label="@string/app_name"
    android:name=".LocalizedApplication"   <!-- use your specific Application  -->
    android:largeHeap="true"
    android:hardwareAccelerated="true"
    >

    <!-- For billing -->
    <service android:name="net.robotmedia.billing.BillingService" />
        <receiver android:name="net.robotmedia.billing.BillingReceiver">
        <intent-filter>
            <action android:name="com.android.vending.billing.IN_APP_NOTIFY" />
            <action android:name="com.android.vending.billing.RESPONSE_CODE" />
            <action android:name="com.android.vending.billing.PURCHASE_STATE_CHANGED" />
        </intent-filter>
    </receiver>

我们实施了ISignatureValidator

public class PhoneSignatureValidator implements ISignatureValidator {
    private final String TAG = this.getClass().getSimpleName();
    private PhoneServerLink mServerLink;


    private BillingController.IConfiguration configuration;

    public PhoneSignatureValidator(Context context, BillingController.IConfiguration configuration, String our_product_sku) {
        this.configuration = configuration;
        mServerLink = new PhoneServerLink(context);
        mServerLink.setSku(our_product_sku);
    }


    @Override
    public boolean validate(String signedData, String signature) {
        final String publicKey;
        if (configuration == null || TextUtils.isEmpty(publicKey = configuration.getPublicKey())) {
            Log.w(BillingController.LOG_TAG, "Please set the public key or turn on debug mode");
            return false;
        }
        if (signedData == null) {
            Log.e(BillingController.LOG_TAG, "Data is null");
            return false;
        }
        // mServerLink will talk to your server
        boolean bool = mServerLink.validateSignature(signedData, signature);
        return bool;
    }

}

上面的最后几行称你的班级实际上会与你的服务器通话。

我们的PhoneServerLink开头是这样的:

public class PhoneServerLink implements GetJSONListener {

    public PhoneServerLink(Context context) {
        mContext = context;
    }

    public boolean validateSignature(String signedData, String signature) {
        return getPurchaseResultFromServer(signedData, signature, false);
    }

    private boolean getPurchaseResultFromServer(String signedData, String signature,  boolean async) {  
            // send request to server using whatever protocols you like 
    }

}

答案 2 :(得分:4)

交易数据使用特定于您应用的私钥进行签名。还有一个nonce来防止重放(即多次发送相同,有效的数据)。如果您验证nonce是唯一的并且签名在您的服务器上有效,那么您可以合理地确定它不是假的。检查this Google IO presentation的IAB部分以进行讨论。