Search code examples
androidandroid-studioin-app-purchasein-app-billing

In app billing consumable item


I am trying to implement Google In-App Billing for selling consumable items (coins). I tested it with a non-consumable item and it worked fine. But I can't make it consumable. Every time I test it, I can only buy it once ! Here is my code :

public class MainActivity extends AppCompatActivity { IabHelper mHelper;

boolean verifyDeveloperPayload(Purchase p) {
    String payload = p.getDeveloperPayload();

/*
 * TODO: verify that the developer payload of the purchase is correct. It will be
 * the same one that you sent when initiating the purchase.
 *
 * WARNING: Locally generating a random string when starting a purchase and
 * verifying it here might seem like a good approach, but this will fail in the
 * case where the user purchases an item on one device and then uses your app on
 * a different device, because on the other device you will not have access to the
 * random string you originally generated.
 *
 * So a good developer payload has these characteristics:
 *
 * 1. If two different users purchase an item, the payload is different between them,
 *    so that one user's purchase can't be replayed to another user.
 *
 * 2. The payload must be such that you can verify it even when the app wasn't the
 *    one who initiated the purchase flow (so that items purchased by the user on
 *    one device work on other devices owned by the user).
 *
 * Using your own server to store and verify developer payloads across app
 * installations is recommended.
 */

    return true;
}
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    String base64EncodedPublicKey = "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAlJQQTbdM6zP0585Ar0YKZAYQish29+AkZpdu4fGUO3WLoVm9UPOSNMMBmo8odzQbcVZdlKUfocohg/52qoQk9crVIhdHJM+O1GK9+hJSdVkZo0PWW5+1sJSCQ7cw0NTxIdDQVSYT0WWC2zkn8Fpxyz1N9pGHh21jxbviDYcnh1gyK+mCLt6jWcVxKl8BYgC0SS7K9F+7kHy+B/GG8ZSl2xhcCqlid/8cEjqH7yvMPciWA8lHvHB7rGz/nUg/v2ydhmUY6f8Ifh6+ygUu2XrhDU0v8wZ24yKw2Kw4SVZbm5ZmC/DXCgx+hIWVL+/yAFqHJ0ygqwW4aLTuKV6PyDaC1QIDAQAB";

    // compute your public key and store it in base64EncodedPublicKey
    mHelper = new IabHelper(this, base64EncodedPublicKey);
    mHelper.enableDebugLogging(true);

    mHelper.startSetup(new IabHelper.OnIabSetupFinishedListener() {
        public void onIabSetupFinished(IabResult result) {
            if (!result.isSuccess()) {
                // Oh no, there was a problem.
                Log.d("TAG", "Problem setting up In-app Billing: " + result);
                Log.i("TAG", "ERROR");
            }
            // Hooray, IAB is fully set up!
            //check owned items & consum em
            checkOwnedItems();
            //make a test purchase
                makePurchase();
        }
    });

}

private void makePurchase() {
    try {
        mHelper.launchPurchaseFlow(this, "next", 10001,
                mPurchaseFinishedListener, "bGoa+V7g/yqDXvKRqq+JTFn4uQZbPiQJo4pf9RzJ");
    } catch (IabHelper.IabAsyncInProgressException e) {
        e.printStackTrace();
        showToast("oh no error purchase" + String.valueOf(e));
    }
}

private void checkOwnedItems() {
    try {
        mHelper.queryInventoryAsync(mGotInventoryListener);
    } catch (IabHelper.IabAsyncInProgressException e) {
        showToast("Oh no error in check()");
        //complain("Error querying inventory. Another async operation in progress.");
    }
}


// Listener that's called when we finish querying the items and subscriptions we own
IabHelper.QueryInventoryFinishedListener mGotInventoryListener = new IabHelper.QueryInventoryFinishedListener() {
    public void onQueryInventoryFinished(IabResult result, Inventory inventory) {
        Purchase item = inventory.getPurchase("next");
        if (item != null && verifyDeveloperPayload(item)) {
            //Log.d("TAG", "We have gas. Consuming it.");
            try {
                mHelper.consumeAsync(inventory.getPurchase("next"), mConsumeFinishedListener);
            } catch (IabHelper.IabAsyncInProgressException e) {
                // complain("Error consuming gas. Another async operation in progress.");
                showToast("oh no error when consuming");
            }
            return;
        }
    }

};


// Called when consumption is complete
IabHelper.OnConsumeFinishedListener mConsumeFinishedListener = new IabHelper.OnConsumeFinishedListener() {
    public void onConsumeFinished(Purchase purchase, IabResult result) {
        Log.d("TAG", "Consumption finished. Purchase: " + purchase + ", result: " + result);

        // if we were disposed of in the meantime, quit.
        if (mHelper == null) return;

        // We know this is the "gas" sku because it's the only one we consume,
        // so we don't check which sku was consumed. If you have more than one
        // sku, you probably should check...
        if (result.isSuccess()) {
            // successfully consumed, so we apply the effects of the item in our
            // game world's logic, which in our case means filling the gas tank a bit
            //Log.d(TAG, "Consumption successful. Provisioning.");
            //mTank = mTank == TANK_MAX ? TANK_MAX : mTank + 1;
            // saveData();
            // alert("You filled 1/4 tank. Your tank is now " + String.valueOf(mTank) + "/4 full!");
        }
        else {
            //complain("Error while consuming: " + result);
        }
        Log.d("TAG", "End consumption flow.");
    }
};


// Callback for when a purchase is finished
IabHelper.OnIabPurchaseFinishedListener mPurchaseFinishedListener = new IabHelper.OnIabPurchaseFinishedListener() {
    public void onIabPurchaseFinished(IabResult result, Purchase purchase) {
        Log.d("TAG", "Purchase finished: " + result + ", purchase: " + purchase);

        // if we were disposed of in the meantime, quit.
        if (mHelper == null) return;

        if (result.isFailure()) {
            // complain("Error purchasing: " + result);
            return;
        }
        if (!verifyDeveloperPayload(purchase)) {
            // complain("Error purchasing. Authenticity verification failed.");
            return;
        }

        Log.d("TAG", "Purchase successful.");

        if (purchase.getSku().equals("next")) {

            Log.d("TAG", "Purchase is gas. Starting gas consumption.");
            try {
                mHelper.consumeAsync(purchase, mConsumeFinishedListener);
            } catch (IabHelper.IabAsyncInProgressException e) {
                //complain("Error consuming gas. Another async operation in progress.");
                showToast("oh no error when consuming");
                return;
            }
        }
    }
};

@Override
public void onDestroy() {
    super.onDestroy();
    if (mHelper != null) try {
        mHelper.dispose();
    } catch (IabHelper.IabAsyncInProgressException e) {
        e.printStackTrace();
    }
    mHelper = null;
}
private void showToast(String message) {
    Toast.makeText(this, message, Toast.LENGTH_LONG).show();
}

} }

Sorry for my english and thanks.


Solution

  • I got it now : So my first problem was that I called makePurchase right after the checkOwnedItems method --> Throws Exception :can't start async operation (launchPurchaseFlow) because another async operation (refresh inventory) is in progress.Then I used a button to call makePurchase and it still wasn't working... because I had made toasts with success or fail + exception in all listeners I knew that the onIabPurchaseFinished method wasn't called so I googled and found this answer : https://stackoverflow.com/a/14968203/4106137.