Billing API v3 IabHelper NullPointerException

Edit 4/15: Łapanie nullpointera w IabHelper zdaje się zatrzymywać ten problem. Nie widzę już WYJĄTKÓW, przyjmę to jako odpowiedź.


Edit 4/04: [[12]} trochę głębiej. Istnieją bloki try catch obsługujące RemoteExceptions i JSONExceptions dla metody queryPurchases, ale bez obsługi NullPointerException. Co mam zamiar spróbować jest zawierać obsługę wyjątków NullPointer więc IabHelper wygląda to tak podczas próby querySkuDetails:

    catch (NullPointerException e) {
        throw new IabException(IABHELPER_UNKNOWN_ERROR, "NullPointer while refreshing inventory.", e);
    }

Właśnie zgłosiłem błąd:

Https://code.google.com/p/marketbilling/issues/detail?id=114


Edit 3/25: cóż, wciąż odbieram ten wypadek... teraz dzieje się to podczas próby uzyskania kontekstu w linii 3 następującego fragmentu z IabHelper: {]}

int queryPurchases(Inventory inv, String itemType) throws JSONException, RemoteException {
    logDebug("Querying owned items, item type: " + itemType);
    logDebug("Package name: " + mContext.getPackageName());

Jest to frustrujące, ponieważ w moim manifeście zawsze używam pełnej nazwy ścieżki mojej aplikacji dla "nazwa".

Przykład "kom.myappname.bla.ClassName "

Ja też próbowałem to przekazać, MyClass.to getApplicationContext () do mHelper. Jednak wszystkie one wytwarzają te same wyniki NullPointer losowo z urządzeń na wolności. Próbowałem też name=".MyClass " w manifeście. Tak to wygląda obecnie:
mHelper = new IabHelper(MyClass.this, myKey);

Edit 3/18/13: nadal otrzymuję wyjątki, nawet z nową wersją IabHelper wdrożoną na 3/17.

Zaczynam dostrzegać wzór tutaj, że awarie są wszystkie podczas próby uzyskania kontekstu Podczas wykonywania mContext.getPackageName (). Jestem ciekaw, dlaczego to działa na wszystkich moich urządzeniach testowych, a ja nie mogę odtworzyć tej awarii, a tylko wydaje się być na niewielkiej liczbie urządzeń.

Oto nowy crash:

java.lang.NullPointerException
    at com.myapp.util.IabHelper.queryPurchases(SourceFile:836)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:558)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:522)
    at com.myapp.util.IabHelper$2.run(SourceFile:617)
    at java.lang.Thread.run(Thread.java:1019)

Spowodowane przez IabHelper...

line 836: logDebug("Package name: " + mContext.getPackageName());

Edit 3/17/13: widzę, że w ciągu ostatnich kilku miesięcy opublikowano wiele poprawek, spróbuję najnowszego kodu dostępnego tutaj i zobaczę, czy to rozwiązuje problem:

Https://code.google.com/p/marketbilling/source/browse/v3/src/com/example/android/trivialdrivesample/util


W jednej z moich aplikacji używam interfejsu API billing i dołączonego do niego kodu boilerplate.

Używam najnowszej wersji billing API dostępnej za pośrednictwem Menedżera SDK od 16.03.2013.

W mojej aktywności odpytywam inwentarz używając:

final List<String> skuList = new ArrayList<String>();
skuList.add("sku1");
skuList.add("sku2");
skuList.add("sku3");
if (skuList != null) {
    if (skuList.size() > 0) {
        try {
            mHelper.queryInventoryAsync(true, skuList, mGotInventoryListener);
        } catch (Exception e) { 
            ACRA.getErrorReporter().handleException(e);
        }
    }
}

Otrzymuję wiele Raporty NullPointerException z klasy IabHelper dla następujących urządzeń. Nie mogę odtworzyć problemu i nie mogę znaleźć żadnych informacji dotyczących tych awarii, i jest powodem, dla którego zamieszczam to pytanie.

Mam niezliczoną ilość innych sprawdzeń null i bloków try/catch w części "developer facing" API billing, w tym w onqueryinventoryfounded, więc wiem, że ten wyjątek nie jest wyrzucany z "mojego kodu" (ponieważ nie przechwytuję crashów z każda z klas mojej aplikacji), ale zamiast tego jest wyrzucany z samego IabHelper. Nie zmodyfikowałem IabHelper inaczej niż ta zalecana poprawka: https://stackoverflow.com/a/14737699

Crash # 1 Galaxy Nexus

java.lang.NullPointerException
    at com.myapp.util.IabHelper.querySkuDetails(SourceFile:802)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:471)
    at com.myapp.util.IabHelper$2.run(SourceFile:521)
    at java.lang.Thread.run(Thread.java:856)

Spowodowane przez IabHelper...

line 802: Bundle skuDetails = mService.getSkuDetails(3, mContext.getPackageName(), ITEM_TYPE_INAPP, querySkus);    

Crash # 2 Samsung GT-S5570L

java.lang.NullPointerException
    at com.myapp.util.IabHelper.queryPurchases(SourceFile:735)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:465)
    at com.myapp.util.IabHelper$2.run(SourceFile:521)
    at java.lang.Thread.run(Thread.java:1019)

Spowodowane przez IabHelper...

line 735: Bundle ownedItems = mService.getPurchases(3, mContext.getPackageName(), ITEM_TYPE_INAPP, continueToken);
Author: logray, 2013-03-17

6 answers

Edit 4/15: Łapanie nullpointera w IabHelper zdaje się powstrzymywać ten problem. Nie widzę już WYJĄTKÓW, przyjmę to jako odpowiedź.


Edit 4/04: Istnieją bloki try catch obsługujące RemoteExceptions i JSONExceptions dla metody queryPurchases, ale bez obsługi NullPointerException. Co mam zamiar spróbować jest zawierać obsługę wyjątków NullPointer więc iabhelper wygląda jak to podczas próby querySkuDetails:

    catch (NullPointerException e) {
        throw new IabException(IABHELPER_UNKNOWN_ERROR, "NullPointer while refreshing inventory.", e);
    }

Właśnie zgłosiłem błąd na ten temat:

Https://code.google.com/p/marketbilling/issues/detail?id=114


Zmień

        if (querySkuDetails) {
            r = querySkuDetails(ITEM_TYPE_INAPP, inv, moreItemSkus);
            if (r != BILLING_RESPONSE_RESULT_OK) {
                throw new IabException(r, "Error refreshing inventory (querying prices of items).");
            }
        }

Do

        if (querySkuDetails) {
            try {
                r = querySkuDetails(ITEM_TYPE_INAPP, inv, moreItemSkus);
                if (r != BILLING_RESPONSE_RESULT_OK) {
                    throw new IabException(r, "Error refreshing inventory (querying prices of items).");
                }
            } catch (NullPointerException e) {
                throw new IabException(IABHELPER_UNKNOWN_ERROR, "NPE while refreshing inventory.", e);
            }
        }

Zmień

            if (querySkuDetails) {
                r = querySkuDetails(ITEM_TYPE_SUBS, inv, moreSubsSkus);
                if (r != BILLING_RESPONSE_RESULT_OK) {
                    throw new IabException(r, "Error refreshing inventory (querying prices of subscriptions).");
                }
            }

Do

            if (querySkuDetails) {
                try {
                    r = querySkuDetails(ITEM_TYPE_SUBS, inv, moreSubsSkus);
                    if (r != BILLING_RESPONSE_RESULT_OK) {
                        throw new IabException(r, "Error refreshing inventory (querying prices of subscriptions).");
                    }
                } catch (NullPointerException e) {
                    throw new IabException(IABHELPER_UNKNOWN_ERROR, "NPE while refreshing inventory.", e);
                }
            }
 23
Author: logray,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-09-10 16:49:06

Prawdopodobnie używasz operacji asynchronicznych. Obecny IabHelper nie jest bezpieczny w przypadku korzystania z...metody asynchroniczne. Problem polega na tym, że w każdej chwili uruchomiona jest operacja asynchroniczna, którą można wywołać w głównym wątku. W tym przypadku otrzymasz NullPointerExceptions i IllegalStateExceptions.

Oto łatka naprawiająca to:

Index: src/com/evotegra/aCoDriver/iabUtil/IabHelper.java
===================================================================
--- src/com/evotegra/aCoDriver/iabUtil/IabHelper.java   (revision 1162)
+++ src/com/evotegra/aCoDriver/iabUtil/IabHelper.java   (working copy)
@@ -86,7 +86,10 @@

     // Is an asynchronous operation in progress?
     // (only one at a time can be in progress)
-    boolean mAsyncInProgress = false;
+    volatile boolean mAsyncInProgress = false;
+    
+    // is set to true if dispose is called while a thread is running. Allows graceful shutdown
+    volatile boolean mDisposeRequested = false;

     // (for logging/debugging)
     // if mAsyncInProgress == true, what asynchronous operation is in progress?
@@ -285,6 +288,12 @@
      * disposed of, it can't be used again.
      */
     public void dispose() {
+       // do not dispose while an async Thread is running. Will cause all kinds of exceptions.
+       // In this case dispose must be called from thread after setting mAsyncInProgress to true
+       if (mAsyncInProgress) {
+           mDisposeRequested = true;
+           return;
+       }
         logDebug("Disposing.");
         mSetupDone = false;
         if (mServiceConn != null) {
@@ -827,6 +836,7 @@
         logDebug("Ending async operation: " + mAsyncOperation);
         mAsyncOperation = "";
         mAsyncInProgress = false;
+        if (mDisposeRequested) IabHelper.this.dispose();
     }

Lub pobrać łatkę tutaj. http://code.google.com/p/marketbilling/issues/detail?id=139&thanks=139&ts=1375614409

 20
Author: tmanthey,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-08-04 11:32:24

Lekko zmodyfikuj początek metody queryPurchases, aby wyglądał tak:

int queryPurchases(Inventory inv, String itemType) throws JSONException, RemoteException {
        // Query purchases
        //logDebug("Querying owned items, item type: " + itemType);
       //logDebug("Package name: " + mContext.getPackageName());
       boolean verificationFailed = false;
       String continueToken = null;

        do {
//            logDebug("Calling getPurchases with continuation token: " + continueToken);
            if(mDisposed || mService==null) return IABHELPER_UNKNOWN_ERROR;
            Bundle ownedItems = mService.getPurchases(3, mContext.getPackageName(),
                    itemType, continueToken);

Podziękowania dla sebastie za wskazanie przyczyny tego.

 0
Author: Thunder,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-01-02 11:34:16

Tmanthey patch wymaga również

mDisposeRequested = false;

Po usunięciu następuje

 0
Author: Markos Evlogimenos,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-05-23 12:24:56

Jeśli dostajesz ten błąd na emulatorze, może to być bardzo prosta rzecz, która zdarza się w ponad połowie przypadków.

Sprawdź, czy używasz Google API SDK, a nie zwykłego SDK!!!

 -1
Author: sandalone,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-03-18 19:05:52

IabHelper jest przestarzały i został zastąpiony przez BillingClient. Zobacz https://developer.android.com/google/play/billing/billing_library.html

 -1
Author: DSchmidt,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-10-11 19:09:48