Search code examples
javaandroidjsongsonauto-value

Unable to convert JSON to ArrayList when using AutoValue


I am using AutoValue to for things like getter/setter and Parcelable.

I am storing the JSON value of objects in SharedPreferences and then retrieving them when needed. However, this throws an exception when I try to convert the JSON array stored as String in SharedPreferences to ArrayList<T> since I am using AutoValue which does not allow to use the Constructor directly.

Here is the exception that I am getting -

java.lang.RuntimeException: Failed to invoke public org.yadavvi.tutorials.governor.data.Governor() with no args
at com.google.gson.internal.ConstructorConstructor$3.construct(ConstructorConstructor.java:111)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:206)
at com.google.gson.internal.bind.TypeAdapterRuntimeTypeWrapper.read(TypeAdapterRuntimeTypeWrapper.java:40)
at com.google.gson.internal.bind.CollectionTypeAdapterFactory$Adapter.read(CollectionTypeAdapterFactory.java:82)
at com.google.gson.internal.bind.CollectionTypeAdapterFactory$Adapter.read(CollectionTypeAdapterFactory.java:61)
at com.google.gson.Gson.fromJson(Gson.java:879)
at com.google.gson.Gson.fromJson(Gson.java:844)
at com.google.gson.Gson.fromJson(Gson.java:793)
at org.yadavvi.tutorials.governor.data.source.local.GovernorLocalDataSource.getGovernors(GovernorLocalDataSource.java:60)
at org.yadavvi.tutorials.governor.data.source.local.GovernorLocalDataSourceTest.getGovernors_callsOnSuccessOfGetGovernorsCallback(GovernorLocalDataSourceTest.java:69)

....

Caused by: java.lang.InstantiationException: Can't instantiate abstract class org.yadavvi.tutorials.governor.data.Governor
at java.lang.reflect.Constructor.newInstance(Native Method)
at com.google.gson.internal.ConstructorConstructor$3.construct(ConstructorConstructor.java:108)
... 41 more

Here is how the Model/AutoValue class looks like -

package org.yadavvi.tutorials.governor.data;


import com.google.auto.value.AutoValue;
import com.google.gson.Gson;
import com.google.gson.TypeAdapter;
import com.google.gson.annotations.SerializedName;

/**
 * Immutable Governor created using AutoValue.
 */
@AutoValue
public abstract class Governor {

    public static Governor create(String name) {
        return new AutoValue_Governor(name);
    }

    @SerializedName("name")
    abstract String name();
}

and here is how the class where I try to convert the JSON array string to ArrayList<T> looks like -

package org.yadavvi.tutorials.governor.data.source.local;


import android.content.Context;
import android.content.SharedPreferences;
import android.support.annotation.NonNull;
import android.util.Log;

import org.yadavvi.tutorials.governor.data.Governor;
import org.yadavvi.tutorials.governor.data.source.GovernorDataSource;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.reflect.TypeToken;

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;


public class GovernorLocalDataSource implements GovernorDataSource {

    ....

    @Override
    public void getGovernors(@NonNull GetGovernorsCallback callback) {
        Gson gson = new GsonBuilder().create();
        List<Governor> governors;

        String governorsString = mSharedPreference.getString(GOVERNORS, "");
        Log.i(TAG, "governorsString: " + governorsString);
        if (governorsString.equalsIgnoreCase("")) {
            governors = populateGovernors();
            governorsString = gson.toJson(governors);
            mSharedPreference.edit().putString(GOVERNORS, governorsString).commit();
        } else {
            governors = gson.fromJson(governorsString, new TypeToken<ArrayList<Governor>>(){}.getType());
        }

        if (governors != null && governors.size() > 0) {
            callback.onSuccess(governors);
        } else {
            callback.onFailure();
        }
    }

    ....

}

The exception is thrown at this line -

governors = gson.fromJson(governorsString, new TypeToken<ArrayList<Governor>>(){}.getType());

EDIT: I have some AutoValue-Extensions in my Android project.

The app build.gradle looks like this -

apply plugin: 'com.android.application'

android {
    compileSdkVersion 24
    buildToolsVersion "24.0.2"
    defaultConfig {
        applicationId "org.yadavvi.tutorials.governor"
        minSdkVersion 21
        targetSdkVersion 24
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs')
    testCompile 'junit:junit:4.12'
    testCompile 'org.mockito:mockito-core:2.2.5'
    testCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
        exclude group: 'com.android.support', module: 'support-annotations'
    })

    androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
        exclude group: 'com.android.support', module: 'support-annotations'
    })
    androidTestCompile 'org.mockito:mockito-core:2.2.5'

    compile 'com.android.support:appcompat-v7:24.2.1'
    compile 'com.google.auto.value:auto-value:1.2'
    compile 'com.ryanharter.auto.value:auto-value-gson:0.4.3'
}

Solution

  • I have added a custom TypeAdapterFactory to the GsonBuilder and it seems to work fine.

    The TypeAdapterFactory looks something like this -

    package org.yadavvi.tutorials.governor.data;
    
    import com.google.gson.typeadapterfactory;
    import com.ryanharter.auto.value.gson.gsontypeadapterfactory;
    
    @GsonTypeAdapterFactory
    public abstract class GovernorAdapterFactory implements TypeAdapterFactory {
    
        public static GovernorAdapterFactory create() {
            return new AutoValueGson_GovernorAdapterFactory();
        }
    
    }
    

    The GovernorLocalDataSource is changed like so -

    public class GovernorLocalDataSource implements GovernorDataSource {
    
        @Override
        public void getGovernors(@NonNull GetGovernorsCallback callback) {
           Gson gson = new GsonBuilder()
                   .registerTypeAdapterFactory(GovernorAdapterFactory.create())
                   .create();
           List<Governor> governors;
    
           String governorsString = mSharedPreference.getString(GOVERNORS, EMPTY_STRING);
           if (governorsString.isEmpty()) {
               governors = populateGovernors();
               governorsString = gson.toJson(governors);
               mSharedPreference.edit().putString(GOVERNORS, governorsString).commit();
           } else {
               Type type = new TypeToken<ArrayList<Governor>>() {}.getType();
               governors = gson.fromJson(governorsString, type);
           }
    
           if (governors != null && governors.size() > 0) {
               callback.onSuccess(governors);
           } else {
               callback.onFailure();
           }
    
        }
    
    }
    

    PS: It would be great if someone could explain WHY this works.