Home | History | Annotate | Download | only in v2
      1 page.title=Implementing Subscriptions  <span style="font-size:16px;">(IAB Version 2)</span>
      2 excludeFromSuggestions=true
      3 @jd:body
      4 
      5 <p class="caution" style=
      6 "background-color:#fffdeb;width:100%;margin-bottom:1em;padding:.5em;">
      7   The <strong>In-app Billing Version 2 API</strong> has been deprecated in
      8   favor of the Version 3 API. If your app is using In-app Billing, please
      9   <strong>make sure that it is using the Version 3 API</strong>. If your app is
     10   still using the Version 2 API, you must <strong>migrate to the Version 3 API
     11   as soon as possible</strong>.<br>
     12   <br>
     13   We plan to turn off the In-app Billing Version 2 service on <strong>January
     14   27, 2015</strong>, after which time users will <strong>no longer be able to
     15   purchase in-app items and subscriptions through the Version 2 API</strong>.
     16   We strongly encourage and recommend you migrate your apps to use Version 3
     17   API by November 2014, to provide ample time for users to update their apps to
     18   the new version.<br>
     19   <br>
     20   For more information, please see the <a href=
     21   "http://support.google.com/googleplay/android-developer/answer/6090268">Help Center
     22   article</a>. For common questions about transitioning your implementation to
     23   In-app Billing Version 3, please see <a href=
     24   "{@docRoot}google/play/billing/billing_overview.html#migration">Migration
     25   Considerations</a>.
     26 </p>
     27     <div id="qv-wrapper" style="margin-top:0;">
     28 <div id="qv">
     29   <h2>In this document</h2>
     30   <ol>
     31         <li><a href="#sample">Sample Application</a></li>
     32         <li><a href="#model">Application Model</a></li>
     33         <li><a href="#token">Purchase Token</a></li>
     34         <li><a href="#version">Checking the In-app Billing API Version</a></li>
     35         <li><a href="#purchase">Purchasing a Subscription</a></li>
     36         <li><a href="#restore">Restoring Transactions</a></li>
     37         <li><a href="#validity">Checking Subscription Validity</a></li>
     38         <li><a href="#viewstatus">Letting Users Cancel or View Status</a></li>
     39         <li><a href="#purchase-state-changes">Recurring Billing and Changes in Purchase State</a></li>
     40         <li><a href="#modifying">Modifying Your App for Subscriptions</a></li>
     41    </ol>
     42 </div>
     43 </div>
     44 
     45 <p>This document is focused on highlighting implementation details that are 
     46 specific to subscriptions with the Version 2 API. To understand how  
     47 subscriptions work, see <a href="{@docRoot}google/play/billing/billing_subscriptions.html">In-app Billing Subscriptions</a>.</p>
     48 
     49 
     50 <h2 id="sample">Sample Application</h2>
     51 
     52 <p>To help you get started with your In-app Billing implementation and
     53 subscriptions, an updated Version of the In-app Billing sample app is available.
     54 You can download the sample app from the Android SDK repository using the
     55 Android SDK Manager. For details, see <a
     56 href="{@docRoot}google/play/billing/v2/billing_integrate.html#billing-download">
     57 Downloading the Sample Application</a>.</p>
     58 
     59 <h2 id="model">Application Model</h2>
     60 
     61 <p>With subscriptions, your app uses the standard In-app Billing application
     62 model, sending billing requests to the Play Store application over interprocess
     63 communication (IPC) and receiving purchase responses from the Play Store app in
     64 the form of asynchronous broadcast intents. Your application does not manage any
     65 network connections between itself and the Google Play server or use any special
     66 APIs from the Android platform.</p>
     67 
     68 <p>Your app also uses the standard In-app Billing components &mdash; a billing
     69 Service for sending requests, a BroadcastReceiver for receiving the responses,
     70 and a security component for verifying that the response was sent by Google
     71 Play. Also recommended are a response Handler for processing notifications,
     72 errors, and status messages, and an observer for sending callbacks to your
     73 application as needed. All of these components and their interactions are
     74 described in full in the <a
     75 href="{@docRoot}google/play/billing/v2/api.html">In-app Billing
     76 Overview</a> and related documents.</p>
     77 
     78 <p>To initiate different types of billing communication with Google Play, your
     79 app will use the standard set of in-app billing requests and receive the same
     80 responses. Inside the requests and responses are two new fields described below.
     81 </p>
     82 
     83 <h2 id="token">Purchase Token</h2>
     84 
     85 <p>Central to the end-to-end architecture for subscriptions is the purchase
     86 token, a string value that uniquely identifies (and associates) a user ID and a
     87 subscription ID. Google Play generates the purchase token when the user
     88 completes the purchase of a subscription product (and payment is approved by
     89 Google Wallet) and then sends it to the purchasing app on the device through the
     90 In-app Billing API. </p>
     91 
     92 <p>At the conclusion of a <code>PURCHASE_REQUEST</code> message flow, your app
     93 can retrieve the purchase token and other transaction details by initiating a
     94 <code>GET_PURCHASE_INFORMATION</code> request. The Bundle returned by the call
     95 contains an JSON array of order objects. In the order corresponding to the
     96 subscription purchase, the token is available in the <code>purchaseToken</code>
     97 field. </p>
     98 
     99 <p>An example of a JSON order object that includes a subscription purchase token
    100 is shown below. </p>
    101 
    102 <pre class="no-pretty-print" style="color:black">{ "nonce" : 1836535032137741465,
    103   "orders" :
    104     [{ "notificationId" : "android.test.purchased",
    105        "orderId" : "12999556515565155651.5565135565155651"
    106        "packageName" : "com.example.dungeons",
    107        "productId" : "android.test.purchased",
    108        "developerPayload" : "bGoa+V7g/yqDXvKRqq+JTFn4uQZbPiQJo4pf9RzJ",
    109        "purchaseTime" : 1290114783411,
    110        "purchaseState" : 0,
    111        "purchaseToken" : "rojeslcdyyiapnqcynkjyyjh" }]
    112 }
    113 </pre>
    114 
    115 <p>After receiving a purchase token, your apps can store the token locally or
    116 pass it to your backend servers, which can then use it to query the billing
    117 status or cancel the subscription remotely. If your app will store the token
    118 locally, please read the <a
    119 href="{@docRoot}google/play/billing/billing_best_practices.html">Security and
    120 Design</a> document for best practices for maintaining the security of your
    121 data.</p>
    122 
    123 <h2 id="version">Checking the In-app Billing API Version</h2>
    124 
    125 <p>Subscriptions support is available only in versions of Google Play that
    126 support the In-app Billing v2 API (Google Play 3.5 and higher). For your app,
    127 an essential first step at launch is to check whether the Version of Google Play
    128 installed on the device supports the In-app Billing v2 API and
    129 subscriptions.</p>
    130 
    131 <p>To do this, create a CHECK_BILLING_SUPPORTED request Bundle that includes the
    132 required key-value pairs, together with</p>
    133 
    134 <ul>
    135   <li>The <code>API_VERSION</code> key, assigning a value of 2.</li>
    136   <li>The <code>BILLING_REQUEST_ITEM_TYPE</code> key, assigning a value of subs</li>
    137 </ul>
    138 
    139 <p>Send the request using <code>sendBillingRequest(Bundle)</code> and receive
    140 the response Bundle. You can extract the response from the
    141 <code>BILLING_RESPONSE_RESPONSE_CODE</code> key of the response. RESULT_OK
    142 indicates that subscriptions are supported.</p>
    143 
    144 <p>The sample app declares constants for the accepted
    145 <code>BILLING_REQUEST_ITEM_TYPE</code> values (from Consts.java):</p>
    146 
    147 <pre class="pretty-print">   // These are the types supported in the IAB v2
    148    public static final String ITEM_TYPE_INAPP = "inapp";
    149    public static final String ITEM_TYPE_SUBSCRIPTION = "subs";
    150 </pre>
    151 
    152 <p>It sets up a convenience method for building the request bundle (from BillingService.java):</p>
    153 
    154 <pre class="pretty-print">       protected Bundle makeRequestBundle(String method) {
    155            Bundle request = new Bundle();
    156            request.putString(Consts.BILLING_REQUEST_METHOD, method);
    157            request.putInt(Consts.BILLING_REQUEST_<code>API_VERSION</code>, 2);
    158            request.putString(Consts.BILLING_REQUEST_PACKAGE_NAME, getPackageName());
    159            return request;
    160        }
    161 </pre>
    162 
    163 <p>Heres an example of how to test support for In-App Billing v2 and subscriptions
    164 (from BillingService.java):</p>
    165 
    166 <pre class="pretty-print">   /**
    167     * Wrapper class that checks if in-app billing is supported.
    168     */
    169    class CheckBillingSupported extends BillingRequest {
    170        public String mProductType = null;
    171        public CheckBillingSupported() {
    172            // This object is never created as a side effect of starting this
    173            // service so we pass -1 as the startId to indicate that we should
    174            // not stop this service after executing this request.
    175            super(-1);
    176        }
    177 
    178        public CheckBillingSupported(String type) {
    179            super(-1);
    180            mProductType = type;
    181        }
    182 
    183        &#64;Override
    184        protected long run() throws RemoteException {
    185            Bundle request = makeRequestBundle("CHECK_BILLING_SUPPORTED");
    186            if (mProductType != null) {
    187                request.putString(Consts.<code>BILLING_REQUEST_ITEM_TYPE</code>, mProductType);
    188            }
    189            Bundle response = mService.sendBillingRequest(request);
    190            int responseCode = response.getInt(Consts.<code>BILLING_RESPONSE_RESPONSE_CODE</code>);
    191            if (Consts.DEBUG) {
    192                Log.i(TAG, "CheckBillingSupported response code: " +
    193                        ResponseCode.valueOf(responseCode));
    194            }
    195            boolean billingSupported = (responseCode == ResponseCode.RESULT_OK.ordinal());
    196            ResponseHandler.checkBillingSupportedResponse(billingSupported, mProductType);
    197            return Consts.BILLING_RESPONSE_INVALID_REQUEST_ID;
    198        }
    199    }
    200 </pre>
    201 
    202 <h2 id="purchase">Requesting a Subscription Purchase</h2>
    203 
    204 <p>Once youve checked the API Version as described above and determined that
    205 subscriptions are supported, you can present subscription products to the user
    206 for purchase. When the user has selected a subscription product and initiated a
    207 purchase, your app handles the purchase just as it would for other in-app
    208 products &mdash; by sending a REQUEST_PURCHASE request. You can then launch
    209 Google Play to display the checkout user interface and handle the financial
    210 transaction..  
    211 
    212 <p>The REQUEST_PURCHASE includes a Bundle containing the item details, as
    213 described in the <a
    214 href="{@docRoot}google/play/billing/v2/api.html">In-app Billing
    215 Overview</a>. For a subscription, the Bundle must also specify:</p>
    216 
    217 <ul>
    218   <li>The <code>ITEM_ID</code> key, with a value that specifies a valid, published
    219   subscription product.</li>
    220   <li>The <code>ITEM_TYPE</code> key, with a value of subs
    221   (<code>ITEM_TYPE_SUBSCRIPTION</code> in the sample app). If the request does not
    222   specify the subscription's <code>ITEM_TYPE</code>, Google Play attempts to
    223   handle the request as a standard in-app purchase (one-time purchase).</li>
    224 </ul>
    225 
    226 <p>Google Play synchronously returns a response bundle that includes
    227 <code>RESPONSE_CODE</code>, <code>PURCHASE_INTENT</code>, and
    228 <code>REQUEST_ID</code>. Your app uses the <code>PURCHASE_INTENT</code> to
    229 launch the checkout UI and the message flow proceeds exactly as described in <a
    230 href="{@docRoot}google/play/billing/v2/api.html#billing-message-
    231 sequence">Messaging sequence</a>.</p>
    232 
    233 <p>Heres how the sample app initiates a purchase for a subscription, where
    234 <code>mProductType</code> is <code>ITEM_TYPE_SUBSCRIPTION</code> (from
    235 BillingService.java).</p>
    236 
    237 <pre class="pretty-print">   /**
    238     * Wrapper class that requests a purchase.
    239     */
    240    class RequestPurchase extends BillingRequest {
    241        public final String mProductId;
    242        public final String mDeveloperPayload;
    243        public final String mProductType;
    244 
    245 . . .
    246 
    247        &#64;Override
    248        protected long run() throws RemoteException {
    249            Bundle request = makeRequestBundle("REQUEST_PURCHASE");
    250            request.putString(Consts.BILLING_REQUEST_ITEM_ID, mProductId);
    251            request.putString(Consts.<code>BILLING_REQUEST_ITEM_TYPE</code>, mProductType);
    252            // Note that the developer payload is optional.
    253            if (mDeveloperPayload != null) {
    254                request.putString(Consts.BILLING_REQUEST_DEVELOPER_PAYLOAD, mDeveloperPayload);
    255            }
    256            Bundle response = mService.sendBillingRequest(request);
    257            PendingIntent pendingIntent
    258                    = response.getParcelable(Consts.BILLING_RESPONSE_PURCHASE_INTENT);
    259            if (pendingIntent == null) {
    260                Log.e(TAG, "Error with requestPurchase");
    261                return Consts.BILLING_RESPONSE_INVALID_REQUEST_ID;
    262            }
    263 
    264            Intent intent = new Intent();
    265            ResponseHandler.buyPageIntentResponse(pendingIntent, intent);
    266            return response.getLong(Consts.BILLING_RESPONSE_REQUEST_ID,
    267                    Consts.BILLING_RESPONSE_INVALID_REQUEST_ID);
    268        }
    269 
    270        &#64;Override
    271        protected void responseCodeReceived(ResponseCode responseCode) {
    272            ResponseHandler.responseCodeReceived(BillingService.this, this, responseCode);
    273        }
    274    }
    275 </pre>
    276 
    277 <h2 id="restoring">Restoring Transactions</h2>
    278 
    279 <p>Subscriptions always use  the <em>managed by user account</em> purchase type,
    280 so that you can restore a record of subscription transactions on the device when
    281 needed. When a user installs your app onto a new device, or when the user
    282 uninstalls/reinstalls the app on the original device, your app should restore
    283 the subscriptions that the user has purchased.</p>
    284 
    285 <p>The process for restoring subscriptions transactions is the same as described
    286 in <a
    287 href="{@docRoot}google/play/billing/v2/api.html#billing-message-
    288 sequence">Messaging sequence</a>. Your app sends a
    289 <code>RESTORE_TRANSACTIONS</code> request to Google Play. Google Play sends two
    290 broadcast intents as asynchronous responses &mdash; a <code>RESPONSE_CODE</code>
    291 intent and a <code>PURCHASE_STATE_CHANGED</code> intent.</p>
    292 
    293 <p>The <code>PURCHASE_STATE_CHANGED</code> intent contains a notification ID
    294 that your app can use to retrieve the purchase details, including the purchase
    295 token, by sending a standard <code>GET_PURCHASE_INFORMATION</code> request. The
    296 <code>Bundle</code> returned in the call includes an JSON array of order objects
    297 corresponding to subscription (and in-app product) purchases that you can
    298 restore locally.</p>
    299 
    300 <p>Your app can store the restored purchase state and other transaction details
    301 in the way that best meets your needs. Your app can use it later to check the
    302 subscription validity, although please read the <a
    303 href="{@docRoot}google/play/billing/billing_best_practices.html">Security and
    304 Design</a> document for best practices for maintaining the security of your
    305 data.</p>
    306 
    307 <h2 id="validity">Checking Subscription Validity</h2>
    308 
    309 <p>Subscriptions are time-bound purchases that require successful billing
    310 recurrences over time to remain valid. Your app should check the validity of
    311 purchased subscriptions at launch or prior to granting access to subscriber
    312 content.</p>
    313 
    314 <p>With In-app Billing, you validate a subscription by keeping track of its
    315 purchase state and then checking the state whenever needed. Google Play 
    316 provides two ways to let you know when the purchase
    317 state of a subscription changes:</p>
    318 
    319 <ul>
    320   <li><em>In-app Billing Notifications</em>. Google Play pushes a notification
    321   to your app to indicate a change in the purchase state of a subscription. Your app can
    322   store the most recent purchase state for a given purchase token and then check
    323   that state at run time, as needed.</li>
    324   <li><em>Google Play Android Developer API</em>. You can use this HTTP-based
    325   API to poll Google Play for the current purchase state of a subscription. You
    326   can store the purchased state for each <code>purchaseToken</code> on your
    327   backend servers. For more information, see <a href="#play-dev-api">Google Play
    328   Android Developer API</a>, below.</li>
    329 </ul>
    330 
    331 <p>For most use-cases, especially those where backend servers are already keeping
    332 track of subscribed users, implementing a combination of both methods is the
    333 recommended approach. A typical implementation might work like this:</p>
    334 
    335 <ul>
    336   <li>When the user successfully purchases a new subscription, your app notifies a
    337   backend server, which stores the purchase token, user name, and other
    338   information in a secure location.</li>
    339   <li>Since your app cannot know the expiration date, your server can poll Google
    340   Play to get the expiration and store it with the purchase token and other
    341   data.</li>
    342   <li>Because your server now knows the expiration date, it does not need to poll
    343   Google Play again until after the expiration date, at which time it can confirm
    344   that the subscription was not cancelled.</li>
    345   <li>On the client side, your app can continue to update the server whenever the
    346   purchase state changes, storing the state locally.</li>
    347 </ul>
    348 
    349 <p>If you are using both notifications and the Google Play Android Developer API to validate subscriptions, we recommend the following:</p>
    350 
    351 <ul>
    352   <li>If your app wants to check validity but you cant reach your server (or
    353 you dont have a server), use the latest purchase state received by
    354 notification.</li>
    355   <li>If you have a server and its reachable, always give preference to the
    356 purchase state obtained from your server over the state received in
    357 notifications.</li>
    358 </ul>
    359 
    360 <p>If necessary, you can also use a <code>RESTORE_TRANSACTIONS</code> request to retrieve a record of all managed and in-app products purchased by the user, which you can then store locally. However, using <code>RESTORE_TRANSACTIONS</code> on a regular basis is not recommended because of performance impacts.</p>
    361 
    362 <p>Regardless of the approach you choose, your app should check subscriptions
    363 and validity at launch, such as prior to accessing subscriber content, game
    364 levels, and so on.</p>
    365 
    366 <p class="table-caption"><strong>Table 1.</strong> Summary of purchaseState
    367 values for subscription purchases, as received with a
    368 <code>PURCHASE_STATE_CHANGED</code> intent.</p>
    369 
    370 <table>
    371 <tr>
    372 <th>State</th><th>purchaseState Value</th><th>Comments</th>
    373 </tr>
    374 <tr>
    375 <td>Purchased successfully</td><td><code>0</code></td><td>Sent at original purchase only (not at recurring billing cycles).</td></tr>
    376 <td>Cancelled</td><td><code>1</code></td><td>Sent at original purchase only if the purchase has failed for some reason. </td></tr>
    377 <td>Refunded</td><td><code>2</code></td><td>The purchase was refunded.</code></td></tr>
    378 <td>Subscription expired</td><td><code>3</code></td><td>Sent at the end of a billing cycle to indicate that the subscription expired without renewal because of non-payment or user-cancellation. Your app does not need to grant continued access to the subscription content. 
    379 </td></tr>
    380 </table>
    381 
    382 
    383 <h2 id="viewstatus">Letting the User Cancel or View Subscriptions</h2>
    384 
    385 <p>In-app Billing does not currently provide an API to let users directly view or cancel
    386 subscriptions from within the purchasing app. Instead, users can launch the Play
    387 Store app on their devices and go to the My Apps screen to manage subscriptions. In My Apps,
    388 users can see a list of their subscriptions organized by application. Tapping one of the
    389 subscriptions loads the app's product page, from which users can see active subscriptions
    390 and billing status and cancel subscriptions as needed.</p>
    391 
    392 <p>To make it easier for users to find and manage their subscriptions from inside your app, 
    393 we recommend that you offer a "View My Subscriptions" or "Manage Subscriptions" option in
    394 your UI that directly loads your app's product page in the Play Store app.</p>
    395 
    396 <p>To do this, create an intent with the <a 
    397 href="{@docRoot}reference/android/content/Intent.html#ACTION_VIEW">ACTION_VIEW</a>
    398 action and include the <code>market://</code> URI (rather than the <code>http://</code>
    399 URI) of your app's details page. Heres an example:</p>
    400 
    401 <pre style="pretty-print">Intent intent = new Intent(Intent.ACTION_VIEW);
    402 intent.setData(Uri.parse("market://details?id=com.example.app"));
    403 startActivity(intent);</pre>
    404 
    405 <p>For more information, see 
    406   <a href="{@docRoot}distribute/tools/promote/linking.html">Linking to Your Products</a>.</p>
    407 
    408 <h2 id="purchase-state-changes">Recurring Billing, Cancellation, and Changes In Purchase State</h2>
    409 
    410 <p>Google Play notifies your app when the user completes the purchase of a
    411 subscription, but the purchase state does not change over time, provided that
    412 recurring billing takes place successfully. Google Play does not notify your app
    413 of a purchase state change <em>until the subscription expires because of
    414 non-payment or user cancellation</em>. </p>
    415 
    416 <p>Over the life of a subscription, your app does not need to initiate any
    417 recurring billing events &mdash; those are all handled by Google Play and they
    418 are transparent to your application if billing is successful.</p>
    419 
    420 <p>When the user cancels a subscription during an active billing cycle, Google
    421 Play <em>does not</em> notify your app immediately of the change in purchase
    422 state. Instead, it waits until the end of the active billing cycle and then
    423 notifies your app that the purchase state has changed to "Expired". </p>
    424 
    425 <p>Similarly, if payment for the next billing cycle fails, Google Play waits
    426 until the end of the active billing cycle and then notifies your app at that time that the
    427 purchase state has changed to "Expired".</p>
    428 
    429 <p>Your app can handle user cancellation and non-payment in the same way, since both cause
    430 a change to the same "Expired" purchase state. Once the purchase state has become "Expired",
    431 your app does not need to grant further access to the subscription content.</p>
    432 
    433 <h2 id="modifying">Modifying Your App for Subscriptions</h2>
    434 
    435 <p>For subscriptions, you make the same types of modifications to your app as
    436 are described in <a
    437 href="{@docRoot}google/play/billing/v2/billing_integrate.html#billing-implement">
    438 Modifying your Application Code</a>.</p>
    439 
    440 <p>Note that, in your UI that lets users view and select subscriptions for
    441 purchase, you should add logic to check for purchased subscriptions and validate
    442 them. Your UI should not present subscriptions if the user has already purchased
    443 them.</p>
    444 
    445 
    446 
    447 
    448 
    449