pax_global_header00006660000000000000000000000064137574733310014527gustar00rootroot0000000000000052 comment=0ffdfabd373756dc11a2c85185cd0adebe5ae10e auto-value-gson-1.3.1/000077500000000000000000000000001375747333100145575ustar00rootroot00000000000000auto-value-gson-1.3.1/.github/000077500000000000000000000000001375747333100161175ustar00rootroot00000000000000auto-value-gson-1.3.1/.github/workflows/000077500000000000000000000000001375747333100201545ustar00rootroot00000000000000auto-value-gson-1.3.1/.github/workflows/ci.yml000066400000000000000000000015731375747333100213000ustar00rootroot00000000000000name: CI on: [push, pull_request] jobs: build: name: JDK ${{ matrix.java_version }} runs-on: ubuntu-latest strategy: matrix: java_version: [1.8] steps: - name: Checkout uses: actions/checkout@v1 - name: Gradle Wrapper Validation uses: gradle/wrapper-validation-action@v1 - name: Install JDK ${{ matrix.java_version }} uses: actions/setup-java@v1 with: java-version: ${{ matrix.java_version }} - name: Build with Gradle run: ./gradlew clean check --stacktrace - name: Upload snapshot (main only) run: ./gradlew uploadArchives -PSONATYPE_NEXUS_USERNAME=${{ secrets.SonatypeUsername }} -PSONATYPE_NEXUS_PASSWORD=${{ secrets.SonatypePassword }} if: success() && github.ref == 'refs/heads/main' && github.event_name != 'pull_request' && matrix.java_version == '1.8' auto-value-gson-1.3.1/.gitignore000066400000000000000000000001471375747333100165510ustar00rootroot00000000000000# IntelliJ IDEA .idea *.iml annotations # Gradle .gradle gradlew.bat build local.properties .DS_Storeauto-value-gson-1.3.1/CHANGELOG.md000066400000000000000000000123771375747333100164020ustar00rootroot00000000000000# Change Log ## Version 1.3.1 (2020-11-25) * Fix generated proguard names for inner classes (#246) * Use reflection name in proguardNameOf() (#247) * Recognize @Nullable as a TYPE_USE annotation. (#254) * fix read method for builder pattern. (#256) ## Verson 1.3.0 (2020-01-19) * Resolve materialized types in subclasses of generic types (#241) * Extract factory to its own artifact (`auto-value-gson-factory`) (#238) * Rename extension artifact to `auto-value-gson-extension` and add legacy artifact with the original name (`auto-value-gson`) for existing users. The legacy artifact is empty and just depends on both the extension and factory artifacts. See README ["Download"](README.md#download) section for why you should move away from the legacy artifact! (#239) * Generate proguard rules on-demand (#236) ## Version 1.2.0 (2020-01-10) * Make generated class + constructor package private (#228) * Make field name policy support opt-in (#230) * Generate human readable helper toString() methods (#231) * Optimize hierarchy lookups for GenerateTypeAdapter (#232) * Make generated factories package private (#229) ## Version 1.1.1 (2019-10-18) * Fix factory method with generics #226 (#227) ## Version 1.1.0 (2019-10-15) * Adds support for property defaults, normalization, and validation via Builders (#224) * Migrate off deprecated generatedAnnotation method (#215) ## Version 1.0.0 (2019-02-08) * Adds support for wildcard types. * Respect FieldNamingStrategy when set on the Gson builder (#124) * Use correct "Generated"-annotation based on JDK version. (#186) * Add support for incremental annotaiton processing. (#188, #189) * Allow no-arg static typeAdapter() methods (#195) * Remove support for @GsonTypeAdapter, empty collection defaults, and mutable adapters (#201) * Refactor annotations artifact to runtime artifact for consistency with other libraries (#202) * Embed Proguard rules (#203) * Switch transient annotation to shared AutoTransient library (#207) * Add support for non-public typeAdapter() methods (#210) ## Version 0.8.0 (2018-05-31) * Fixes an issue causing TypeAdapters to be generated non-deterministically. * Updates TypeAdapters to be lazily initialized. * Suppress unchecked for generated adapter read & write methods. * Fixes usage of Java 8 API in GenerateTypeAdapter (#166) * Switch from ConcurrentHashMap to Collections.synchronizedMap() ## Version 0.7.0 (2017-12-20) * Implement `@GeneratedTypeAdapter` support (#160) * Generated code optimizations (#156) * Make annotations java 7 compatible (#157) ## Version 0.6.0 (2017-10-03) * Standalone annotations artifact (#118) (#126) * Generate Nullable annotation at constructor arguments. (#144) * Remove unused arguments to CodeBlock.of() (#146) * Support inner Factory class generation (#117) ## Version 0.5.0 (2017-07-07) * Allow ignoring fields for serialization/deserialization (#91) * Support non-public TypeAdapter factory methods (#121) (#122) * Make defaults off by default (#133) * Add @Generated annotation when available (#135) ## Version 0.4.6 (2017-02-03) * Serializes collections to null by default, adding compiler option to serialize to empty variants. (#103) * Skip null values in JSON (#100) * Use TypeToken.getParameterized for all parameterized fields. (#99) ## Version 0.4.5 (2016-12-08) * Adds support for custom field adapters (#83) * Implements support for default values. (#84) * Handle parameterized generic fields. (#66) ## Version 0.4.4 (2016-10-25) #### Supports: AutoValue 1.3 * Adds better null support. (fixes #23, #50, #67) ## Version 0.4.3 (2016-10-17) #### Supports: AutoValue 1.3 * Updates AutoValue dependency to 1.3 * Initializes known non-nullable collection types to empty collections. (#71) * Use `getDeclaredClass()` instead of `getClass()` on enums. (#72) ## Version 0.4.2 (2016-08-05) #### Supports: AutoValue: 1.3-rc1 * Bumps version number to fix release. ## Version 0.4.1 (2016-08-01) #### Supports: AutoValue: 1.3-rc1 * Adds support for generic AutoValue types!! ## Version 0.4.0 (2016-07-28) #### Supports: AutoValue: 1.3-rc1 * Adds alternate serialization names * Changes to AutoValue-esque factory generation. ## Version 0.3.2-rc1 (2016-06-13) #### Supports: AutoValue: 1.3-rc1 * Updates extension to support AutoValue 1.3-rc1 * Removes final modifier from AutoValueGsonAdapterFactory. ## Version 0.3.1 (2016-05-17) #### Supports: AutoValue 1.2 * Replaces equals check with `isAssignableFrom` in generated type adapter factory. ## Version 0.3.0 (2016-05-11) #### Supports: AutoValue 1.2 * Adds support for single AutoValueGsonTypeAdapterFactory for all AutoValue gson types. * Fixes generation of TypeAdapter.read() for 'char' values * Adds sample project. ## Version 0.2.5 (2016-04-13) #### Supports: AutoValue 1.2 * Fixes issue causing failure for non-standard class and package names. ## Version 0.2.3 (2016-04-13) #### Supports: AutoValue 1.2 * Adds selective generation based on public static method returning TypeAdapter ## Version 0.2.2 (2016-04-05) #### Supports: AutoValue 1.2-rc1 * Fixes issue causing method prefixes (`get`, `is`) to be ignored. ## Version 0.2.1 (2016-03-22) Fixes snapshot issues with 0.2.0 release. Only guaranteed to support AutoValue 1.2-rc1 ## Version 0.2.0 (2016-03-21) Initial release. Only guaranteed to support AutoValue 1.2-rc1 auto-value-gson-1.3.1/LICENSE.txt000066400000000000000000000261351375747333100164110ustar00rootroot00000000000000 Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright [yyyy] [name of copyright owner] Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.auto-value-gson-1.3.1/README.md000066400000000000000000000222671375747333100160470ustar00rootroot00000000000000# AutoValue: Gson Extension An extension for Google's [AutoValue](https://github.com/google/auto) that creates a simple [Gson](https://github.com/google/gson) TypeAdapterFactory for each AutoValue annotated object. ## Usage Simply include auto-value-gson in your project and add a public static method to your `@AutoValue` annotated class returning a TypeAdapter. You can also annotate your properties using `@SerializedName` to define an alternate name for de/serialization. ```java @AutoValue public abstract class Foo { abstract String bar(); @SerializedName("Baz") abstract String baz(); abstract int quux(); abstract String with_underscores(); // The public static method returning a TypeAdapter is what // tells auto-value-gson to create a TypeAdapter for Foo. public static TypeAdapter typeAdapter(Gson gson) { return new AutoValue_Foo.GsonTypeAdapter(gson); } } ``` Now build your project and de/serialize your Foo. ## The TypeAdapter To trigger TypeAdapter generation, you need include a non-private static factory method that accepts a `Gson` parameter and returns a `TypeAdapter` for your AutoValue type. From within this method you can instantiate a new `GsonTypeAdapter` which will have been generated as an inner class of your AutoValue generated implementation. ```java @AutoValue public abstract class Foo { // properties... public static TypeAdapter typeAdapter(Gson gson) { return new AutoValue_Foo.GsonTypeAdapter(gson); } } ``` ## Generics support If your annotated class uses generics, you'll have to modify your static method a little so AutoValue will know how to generate an appropriate adapter. Simply add a `TypeToken` parameter and pass it to the generated `GsonTypeAdapter` class. To have support for fields with generic parameters (eg. `List`) you need to upgrade your Gson dependency to at least **2.8.0**, which introduces the helper `TypeToken.getParameterized()` see [Gson Changelog](https://github.com/google/gson/blob/master/CHANGELOG.md#version-28). ```java @AutoValue public abstract class Foo { abstract A data(); abstract List dataList(); abstract Map> dataMap(); public static TypeAdapter> typeAdapter(Gson gson, Type[] types) { return new AutoValue_Foo.GsonTypeAdapter<>(gson, types); } } ``` Note that the `types` is an array of the `Type` representations of the given type's generics. If `Foo` is parameterized as `Foo`, then the `Type` array passed in should be an array of `{String.class, Integer.class, Boolean.class}`. ## Transient types To ignore certain properties from serialization, you can use the `@AutoTransient` annotation. This comes from a shared transience annotations library and is an `api` dependency of the runtime artifact. You can annotate a property and it will be treated as `transient` for both serialization and deserialization. Note that this should only be applied to nullable properties. ## Builder Support If your `@AutoValue` class has a builder, auto-value-gson will use the builder to instantiate the class. If the `@AutoValue` class has a static no-argument factory method for its builder, it will be used. If there are multiple factory methods, the one annotated `@AutoValueGsonBuilder` will be used. This can be useful for setting default values. ```java @AutoValue public abstract class Foo { abstract int bar(); abstract String quux(); public static Builder builder() { return new AutoValue_Foo.Builder(); } @AutoValueGsonBuilder public static Builder builderWithDefaults() { return new builder().quux("QUUX"); } } ``` ## Field name policy If you want the generated adapter classes to use the input `Gson` instance's field name policy, you can enable this via `autovaluegson.useFieldNamePolicy` processor option. This acts as a flag (any value is ignored) and can be set like any other annotation processor option. In Gradle, this could look like this: ```gradle tasks.withType(JavaCompile) { options.compilerArgs += "-Aautovaluegson.useFieldNamePolicy" } ``` ## Factory Optionally, auto-value-gson can create a single [TypeAdapterFactory](https://google.github.io/gson/apidocs/com/google/gson/TypeAdapterFactory.html) so that you don't have to add each generated TypeAdapter to your Gson instance manually. To generate a `TypeAdapterFactory` for all of your auto-value-gson classes, simply create an abstract class that implements `TypeAdapterFactory` and annotate it with `@GsonTypeAdapterFactory`, and auto-value-gson will create an implementation for you. You simply need to provide a static factory method, just like your AutoValue classes, and you can use the generated `TypeAdapterFactory` to help Gson de/serialize your types. ```java @GsonTypeAdapterFactory public abstract class MyAdapterFactory implements TypeAdapterFactory { // Static factory method to access the package // private generated implementation public static TypeAdapterFactory create() { return new AutoValueGson_MyAdapterFactory(); } } ``` Then you simply need to register the Factory with Gson. ```java Gson gson = new GsonBuilder() .registerTypeAdapterFactory(MyAdapterFactory.create()) .create(); ``` ## @GenerateTypeAdapter There is an annotation in the `auto-value-gson-runtime` artifact called `@GenerateTypeAdapter`. This annotation can be set on types to indicate to the extension that you want the generated adapter to be a top level class in the same package. The name of this class will be the AutoValue class's name plus `_GsonTypeAdapter` suffix. Types annotated with this can also be (de)serialized dynamically at runtime with a provided runtime `TypeAdapterFactory` implementation in the annotation called `FACTORY`. The type name and generated typeadapter class's name *must not be obfuscated* for this to work. The extension that runs during annotation processing will automatically generate custom .pro rules for Proguard/R8 for this, so it should require no extra configuration. When this annotation is used, there will be no intermediate AutoValue class generated (as opposed to the default logic, which generates an intermediate class and generates the `TypeAdapter` as a static inner class of it). There is no need to declare a static `TypeAdapter<...> typeAdapter()` method anymore for this case, though you can optionally define one if you still want to use the `@GsonTypeAdapterFactory` generator for them. `@GenerateTypeAdapter` is compatible with the factory approach above, just make your static method's implementation point to it. It can also be an alternative to it if you use the runtime factory, particularly if you have a multimodule project and are willing to accept a small amount of (heavily cached) reflection. The generated class will have the same parameters as if it were the inner class. If it's generic, its constructor accepts a `Gson` instance and `TypeToken` of the generics. If it's not generic, it's just a `Gson` instance. Example usage: ```java @GenerateTypeAdapter @AutoValue public class Foo { // ... } // Generates public final class Foo_GsonTypeAdapter extends TypeAdapter { public Foo_GsonTypeAdapter(Gson gson) { //... } } // Or with generics @GenerateTypeAdapter @AutoValue public class Foo { // ... } // Generates public final class Foo_GsonTypeAdapter extends TypeAdapter { public Foo_GsonTypeAdapter(Gson gson, TypeToken> typeToken) { //... } } // Using the runtime FACTORY new GsonBuilder() .registerTypeAdapterFactory(GenerateTypeAdapter.FACTORY) .create() .toJson(myFooInstance); ``` ## Download Add a Gradle dependency to the `annotationProcessor`/`kapt` and `implementation`/`api` configuration. ```kotlin annotationProcessor("com.ryanharter.auto.value:auto-value-gson-extension:1.3.1") implementation("com.ryanharter.auto.value:auto-value-gson-runtime:1.3.1") // Optional @GsonTypeAdapterFactory support annotationProcessor("com.ryanharter.auto.value:auto-value-gson-factory:1.3.1") // Legacy generic artifact that includes both -extension and -factory above. This exists to not // break existing users, but shouldn't be used because it includes both the -factory artifact as // well as the -extension artifact. This can have a negative impact on build times if you don't // actually use the factory support, as it is an aggregating incremental processor thus slower // compared to just using the isolating incremental behavior of the extension. annotationProcessor("com.ryanharter.auto.value:auto-value-gson:1.3.1") ``` Snapshots of the latest development version are available in [Sonatype's `snapshots` repository](https://oss.sonatype.org/content/repositories/snapshots/). ## License ``` Copyright 2015 Ryan Harter. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ``` auto-value-gson-1.3.1/RELEASING.md000066400000000000000000000013321375747333100164110ustar00rootroot00000000000000Releasing ======== 1. Change the version in `gradle.properties` to a non-SNAPSHOT version. 2. Update the `CHANGELOG.md` for the impending release. 3. Update the `README.md` with the new version. 4. `git commit -am "Prepare for release X.Y.Z."` (where X.Y.Z is the new version) 5. `./gradlew clean uploadArchives`. 6. Visit [Sonatype Nexus](https://oss.sonatype.org/) and promote the artifact. 7. `git tag -a X.Y.X -m "Version X.Y.Z"` (where X.Y.Z is the new version) 8. Update the `gradle.properties` to the next SNAPSHOT version. 9. `git commit -am "Prepare next development version."` 10. `git push && git push --tags` If step 5 or 6 fails, drop the Sonatype repo, fix the problem, commit, and start again at step 5.auto-value-gson-1.3.1/auto-value-gson-extension/000077500000000000000000000000001375747333100216175ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/build.gradle000066400000000000000000000042521375747333100241010ustar00rootroot00000000000000import org.gradle.internal.jvm.Jvm plugins { id 'com.github.johnrengelman.shadow' version '4.0.1' id 'java-library' id 'maven-publish' } // // disable default jar tasks configurations.runtimeOnly.artifacts.removeAll { it.archiveTask.is jar } tasks.getByName('jar').enabled = false // create extra configuration for shaded dependencies, so they're not included in the pom def shadedConfig = configurations.create('compileShaded') configurations.compileOnly.extendsFrom(shadedConfig) shadowJar { minimize() classifier = '' configurations = [shadedConfig] relocate 'com.google.auto.common', 'autovaluegson.shaded.com.google.auto.common' relocate 'com.google.common', 'autovaluegson.shaded.com.google.common' relocate 'com.google.thirdparty', 'autovaluegson.shaded.com.google.thirdparty' exclude 'afu/**' exclude 'org/**' exclude 'com/google/errorprone/annotations/**' exclude 'com/google/j2objc/annotations/**' exclude 'javax/**' exclude 'META-INF/maven/com.google.auto/auto-common/**' } artifacts { runtime shadowJar archives shadowJar } // sourceCompatibility = JavaVersion.VERSION_1_8 targetCompatibility = JavaVersion.VERSION_1_8 dependencies { annotationProcessor "net.ltgt.gradle.incap:incap-processor:0.2" annotationProcessor 'com.google.auto.service:auto-service:1.0-rc5' annotationProcessor 'com.google.auto.value:auto-value:1.7' compileOnly "net.ltgt.gradle.incap:incap:0.2" compileOnly 'com.google.auto.service:auto-service:1.0-rc5' api 'com.google.auto.value:auto-value:1.7' api 'com.google.auto.value:auto-value-annotations:1.7' api project(':auto-value-gson-runtime') compileShaded 'com.google.auto:auto-common:0.10' compileShaded 'com.google.guava:guava:27.0-jre' implementation 'com.squareup:javapoet:1.12.0' implementation deps.gson testImplementation 'junit:junit:4.12' testImplementation 'com.google.truth:truth:0.42' testImplementation 'com.google.testing.compile:compile-testing:0.16' testImplementation files(Jvm.current().getToolsJar()) } apply from: rootProject.file('gradle/gradle-mvn-push.gradle') auto-value-gson-1.3.1/auto-value-gson-extension/gradle.properties000066400000000000000000000001371375747333100251740ustar00rootroot00000000000000POM_ARTIFACT_ID=auto-value-gson-extension POM_NAME=AutoValue: Gson Extension POM_PACKAGING=jar auto-value-gson-1.3.1/auto-value-gson-extension/src/000077500000000000000000000000001375747333100224065ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/000077500000000000000000000000001375747333100233325ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/000077500000000000000000000000001375747333100242535ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/000077500000000000000000000000001375747333100250315ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/ryanharter/000077500000000000000000000000001375747333100272105ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/ryanharter/auto/000077500000000000000000000000001375747333100301605ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100312745ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100322425ustar00rootroot00000000000000AutoValueGsonExtension.java000077500000000000000000001175121375747333100374710ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/ryanharter/auto/value/gsonpackage com.ryanharter.auto.value.gson; import com.google.auto.common.GeneratedAnnotations; import com.google.auto.common.MoreElements; import com.google.auto.common.MoreTypes; import com.google.auto.service.AutoService; import com.google.auto.value.extension.AutoValueExtension; import com.google.common.base.Defaults; import com.google.common.base.Joiner; import com.google.common.base.Strings; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Lists; import com.google.common.collect.Sets; import com.google.common.primitives.Primitives; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.annotations.SerializedName; import com.google.gson.reflect.TypeToken; import com.google.gson.stream.JsonReader; import com.google.gson.stream.JsonToken; import com.google.gson.stream.JsonWriter; import com.ryanharter.auto.value.gson.internal.WildcardUtil; import com.squareup.javapoet.AnnotationSpec; import com.squareup.javapoet.ArrayTypeName; import com.squareup.javapoet.ClassName; import com.squareup.javapoet.CodeBlock; import com.squareup.javapoet.CodeBlock.Builder; import com.squareup.javapoet.FieldSpec; import com.squareup.javapoet.JavaFile; import com.squareup.javapoet.MethodSpec; import com.squareup.javapoet.NameAllocator; import com.squareup.javapoet.ParameterSpec; import com.squareup.javapoet.ParameterizedTypeName; import com.squareup.javapoet.TypeName; import com.squareup.javapoet.TypeSpec; import com.squareup.javapoet.TypeVariableName; import com.squareup.javapoet.WildcardTypeName; import io.sweers.autotransient.AutoTransient; import java.io.IOException; import java.lang.reflect.Type; import java.util.ArrayList; import java.util.Iterator; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Optional; import java.util.Set; import java.util.stream.Collectors; import java.util.stream.Stream; import javax.annotation.Nullable; import javax.annotation.processing.Filer; import javax.annotation.processing.Messager; import javax.annotation.processing.ProcessingEnvironment; import javax.annotation.processing.SupportedOptions; import javax.lang.model.element.AnnotationMirror; import javax.lang.model.element.ExecutableElement; import javax.lang.model.element.TypeElement; import javax.lang.model.element.TypeParameterElement; import javax.lang.model.type.TypeKind; import javax.lang.model.type.TypeMirror; import javax.lang.model.util.ElementFilter; import javax.tools.Diagnostic; import static com.google.common.base.CaseFormat.LOWER_CAMEL; import static com.google.common.base.CaseFormat.UPPER_CAMEL; import static com.ryanharter.auto.value.gson.AutoValueGsonExtension.USE_FIELD_NAME_POLICY; import static javax.lang.model.element.Modifier.ABSTRACT; import static javax.lang.model.element.Modifier.FINAL; import static javax.lang.model.element.Modifier.PRIVATE; import static javax.lang.model.element.Modifier.PUBLIC; import static javax.lang.model.element.Modifier.STATIC; import static javax.lang.model.element.Modifier.VOLATILE; @SupportedOptions(USE_FIELD_NAME_POLICY) @AutoService(AutoValueExtension.class) public class AutoValueGsonExtension extends AutoValueExtension { public static final String GENERATED_COMMENTS = "https://github.com/rharter/auto-value-gson"; static final String USE_FIELD_NAME_POLICY = "autovaluegson.useFieldNamePolicy"; static class Property { @Nullable static Property create( Messager messager, String humanName, ExecutableElement element, TypeMirror actualType ) { Property property = new Property(humanName, element, actualType); if (property.isTransient() && !property.nullable()) { messager.printMessage(Diagnostic.Kind.ERROR, "Required property cannot be transient!", element); return null; } else { return property; } } final String methodName; final String humanName; final ExecutableElement element; final TypeName type; final TypeName builderType; final ImmutableSet annotations; final boolean nullable; final boolean isTransient; private Property(String humanName, ExecutableElement element, TypeMirror actualType) { this.methodName = element.getSimpleName().toString(); this.humanName = humanName; this.element = element; type = TypeName.get(actualType); builderType = TypeName.get(element.getReturnType()); annotations = buildAnnotations(element); nullable = nullableAnnotation() != null; isTransient = element.getAnnotation(AutoTransient.class) != null; } String serializedName() { SerializedName serializedName = element.getAnnotation(SerializedName.class); if (serializedName != null) { return serializedName.value(); } else { return humanName; } } String[] serializedNameAlternate() { SerializedName serializedName = element.getAnnotation(SerializedName.class); if (serializedName != null) { return serializedName.alternate(); } else { return new String[0]; } } boolean hasSerializedNameAnnotation() { SerializedName serializedName = element.getAnnotation(SerializedName.class); return serializedName != null; } boolean isTransient() { return isTransient; } boolean nullable() { return nullable; } String nullableAnnotation() { for (String annotationString : annotations) { if (annotationString.equals("@Nullable") || annotationString.endsWith(".Nullable")) { return annotationString; } } return null; } private ImmutableSet buildAnnotations(ExecutableElement element) { ImmutableSet.Builder builder = ImmutableSet.builder(); for (AnnotationMirror annotation : element.getAnnotationMirrors()) { builder.add(annotation.getAnnotationType().asElement().toString()); } for (AnnotationMirror annotation : element.getReturnType().getAnnotationMirrors()) { builder.add(annotation.getAnnotationType().asElement().toString()); } return builder.build(); } } private boolean useFieldNamePolicy = false; @Override public IncrementalExtensionType incrementalType(ProcessingEnvironment processingEnvironment) { return IncrementalExtensionType.ISOLATING; } @Override public boolean applicable(Context context) { useFieldNamePolicy = context.processingEnvironment() .getOptions() .containsKey(USE_FIELD_NAME_POLICY); return isApplicable(context.autoValueClass(), context.processingEnvironment().getMessager()); } public static boolean isApplicable(TypeElement type, Messager messager) { //noinspection UnstableApiUsage if (MoreElements.isAnnotationPresent(type, GenerateTypeAdapter.class)) { return true; } // check that the class contains a non-private static method returning a TypeAdapter TypeName typeName = TypeName.get(type.asType()); ParameterizedTypeName typeAdapterType = ParameterizedTypeName.get( ClassName.get(TypeAdapter.class), typeName); TypeName returnedTypeAdapter = null; for (ExecutableElement method : ElementFilter.methodsIn(type.getEnclosedElements())) { if (method.getModifiers().contains(STATIC) && !method.getModifiers().contains(PRIVATE)) { TypeMirror rType = method.getReturnType(); TypeName returnType = TypeName.get(rType); if (returnType.equals(typeAdapterType)) { return true; } if (returnType.equals(typeAdapterType.rawType) || (returnType instanceof ParameterizedTypeName && ((ParameterizedTypeName) returnType).rawType.equals(typeAdapterType.rawType))) { returnedTypeAdapter = returnType; } } } if (returnedTypeAdapter == null) { return false; } // emit a warning if the user added a method returning a TypeAdapter, but not of the right type if (returnedTypeAdapter instanceof ParameterizedTypeName) { ParameterizedTypeName paramReturnType = (ParameterizedTypeName) returnedTypeAdapter; TypeName argument = paramReturnType.typeArguments.get(0); // If the original type uses generics, user's don't have to nest the generic type args if (typeName instanceof ParameterizedTypeName) { ParameterizedTypeName pTypeName = (ParameterizedTypeName) typeName; return pTypeName.rawType.equals(argument); } else { messager.printMessage(Diagnostic.Kind.WARNING, String.format("Found static method returning TypeAdapter<%s> on %s class. " + "Skipping GsonTypeAdapter generation.", argument, type)); } } else { messager.printMessage(Diagnostic.Kind.WARNING, "Found static method returning " + "TypeAdapter with no type arguments, skipping GsonTypeAdapter generation."); } return false; } @Override public String generateClass(Context context, String className, String classToExtend, boolean isFinal) { ProcessingEnvironment env = context.processingEnvironment(); Optional generatedAnnotationSpec = GeneratedAnnotations.generatedAnnotation(env.getElementUtils(), env.getSourceVersion()) .map(AutoValueGsonExtension::createGeneratedAnnotationSpec); TypeElement type = context.autoValueClass(); boolean generateExternalAdapter = type.getAnnotation(GenerateTypeAdapter.class) != null; List properties = Lists.newArrayList(); for (Map.Entry entry : context.properties().entrySet()) { Property property = Property.create( context.processingEnvironment().getMessager(), entry.getKey(), entry.getValue(), context.propertyTypes().get(entry.getKey()) ); if (property == null) { return null; } properties.add(property); } Map types = convertPropertiesToTypes(context.properties()); ClassName classNameClass = ClassName.get(context.packageName(), className); ClassName autoValueClass = ClassName.get(context.autoValueClass()); List typeParams = context.autoValueClass().getTypeParameters(); List params = new ArrayList<>(typeParams.size()); ClassName superclassRawType = ClassName.get(context.packageName(), classToExtend); TypeName superClassType = superclassRawType; if (!typeParams.isEmpty()) { for (TypeParameterElement typeParam : typeParams) { params.add(TypeVariableName.get(typeParam)); } superClassType = ParameterizedTypeName.get(ClassName.get(context.packageName(), classToExtend), params.toArray(new TypeName[params.size()])); } ClassName adapterClassName = generateExternalAdapter ? ClassName.get(context.packageName(), Joiner.on("_").join(autoValueClass.simpleNames()) + "_GsonTypeAdapter") : classNameClass.nestedClass("GsonTypeAdapter"); ClassName finalSuperClass = generateExternalAdapter ? classNameClass : superclassRawType; TypeSpec typeAdapter = createTypeAdapter(type, classNameClass, autoValueClass, adapterClassName, finalSuperClass, properties, params, context.builder().orElse(null), context.processingEnvironment()); Filer filer = context.processingEnvironment().getFiler(); if (generateExternalAdapter) { try { TypeSpec.Builder builder = typeAdapter.toBuilder(); generatedAnnotationSpec.ifPresent(builder::addAnnotation); JavaFile.builder(context.packageName(), builder.build()) .skipJavaLangImports(true) .build() .writeTo(filer); } catch (IOException e) { context.processingEnvironment().getMessager() .printMessage(Diagnostic.Kind.ERROR, String.format( "Failed to write external TypeAdapter for element \"%s\" with reason \"%s\"", type, e.getMessage())); } ClassName proguardTarget = ClassName.get(context.autoValueClass()); List adapterConstructorParams = Lists.newArrayList(); typeAdapter.methodSpecs.stream().filter(MethodSpec::isConstructor).findFirst() .ifPresent(c -> { for (ParameterSpec p : c.parameters) { adapterConstructorParams.add(proguardNameOf(p.type)); } }); ProguardConfig proguardConfig = ProguardConfig.create( proguardTarget, adapterClassName, adapterConstructorParams ); try { proguardConfig.writeTo(filer, context.autoValueClass()); } catch (IOException e) { context.processingEnvironment().getMessager() .printMessage(Diagnostic.Kind.ERROR, String.format( "Failed to write proguard file for element \"%s\" with reason \"%s\"", context.autoValueClass(), e.getMessage())); } return null; } else { TypeSpec.Builder subclass = TypeSpec.classBuilder(classNameClass) .superclass(superClassType) .addType(typeAdapter.toBuilder() .addModifiers(STATIC) .build()) .addMethod(generateConstructor(properties, types)); generatedAnnotationSpec.ifPresent(subclass::addAnnotation); if (!typeParams.isEmpty()) { subclass.addTypeVariables(params); } if (isFinal) { subclass.addModifiers(FINAL); } else { subclass.addModifiers(ABSTRACT); } return JavaFile.builder(context.packageName(), subclass.build()).build().toString(); } } private static AnnotationSpec createGeneratedAnnotationSpec(TypeElement generatedAnnotationTypeElement) { return AnnotationSpec.builder(ClassName.get(generatedAnnotationTypeElement)) .addMember("value", "$S", AutoValueGsonExtension.class.getName()) .addMember("comments", "$S", GENERATED_COMMENTS) .build(); } private ImmutableMap createFields(List properties) { ImmutableMap.Builder fields = ImmutableMap.builder(); ClassName jsonAdapter = ClassName.get(TypeAdapter.class); Set seenTypes = Sets.newHashSet(); NameAllocator nameAllocator = new NameAllocator(); for (Property property : properties) { if (property.isTransient()) { continue; } TypeName type = property.type.isPrimitive() ? property.type.box() : property.type; ParameterizedTypeName adp = ParameterizedTypeName.get(jsonAdapter, type); if (!seenTypes.contains(property.type)) { fields.put(property.type, FieldSpec.builder(adp, nameAllocator.newName(simpleName(property.type)) + "_adapter", PRIVATE, VOLATILE) .build()); seenTypes.add(property.type); } } return fields.build(); } private static String simpleName(TypeName typeName) { if (typeName instanceof ClassName) { return UPPER_CAMEL.to(LOWER_CAMEL, ((ClassName) typeName).simpleName()); } else if (typeName instanceof ParameterizedTypeName) { ParameterizedTypeName parameterizedTypeName = (ParameterizedTypeName) typeName; return UPPER_CAMEL.to(LOWER_CAMEL, parameterizedTypeName.rawType.simpleName()) + (parameterizedTypeName.typeArguments.isEmpty() ? "" : "__") + simpleName(parameterizedTypeName.typeArguments); } else if (typeName instanceof ArrayTypeName) { return "array__" + simpleName(((ArrayTypeName) typeName).componentType); } else if (typeName instanceof WildcardTypeName) { WildcardTypeName wildcardTypeName = (WildcardTypeName) typeName; return "wildcard__" + simpleName(ImmutableList.builder().addAll(wildcardTypeName.lowerBounds) .addAll(wildcardTypeName.upperBounds) .build()); } else if (typeName instanceof TypeVariableName) { TypeVariableName variable = (TypeVariableName) typeName; return variable.name + (variable.bounds.isEmpty() ? "" : "__") + simpleName(variable.bounds); } else { return typeName.toString(); } } private static String simpleName(List typeNames) { return Joiner.on("_").join(typeNames.stream() .map(AutoValueGsonExtension::simpleName) .collect(Collectors.toList())); } private MethodSpec generateConstructor(List properties, Map types) { List params = Lists.newArrayList(); for (Property property : properties) { ParameterSpec.Builder builder = ParameterSpec.builder(property.type, property.humanName); if (property.nullable()) { builder.addAnnotation(ClassName.bestGuess(property.nullableAnnotation())); } params.add(builder.build()); } MethodSpec.Builder builder = MethodSpec.constructorBuilder() .addParameters(params); StringBuilder superFormat = new StringBuilder("super("); for (int i = properties.size(); i > 0; i--) { superFormat.append("$N"); if (i > 1) superFormat.append(", "); } superFormat.append(")"); builder.addStatement(superFormat.toString(), types.keySet().toArray()); return builder.build(); } /** * Converts the ExecutableElement properties to TypeName properties */ private Map convertPropertiesToTypes(Map properties) { Map types = new LinkedHashMap<>(); for (Map.Entry entry : properties.entrySet()) { ExecutableElement el = entry.getValue(); types.put(entry.getKey(), TypeName.get(el.getReturnType())); } return types; } private TypeSpec createTypeAdapter( TypeElement autoValueType, ClassName className, ClassName autoValueClassName, ClassName gsonTypeAdapterName, ClassName superClassType, List properties, List typeParams, @Nullable BuilderContext builderContext, ProcessingEnvironment processingEnvironment) { ClassName typeAdapterClass = ClassName.get(TypeAdapter.class); final TypeName autoValueTypeName = !typeParams.isEmpty() ? ParameterizedTypeName.get(autoValueClassName, typeParams.toArray(new TypeName[typeParams.size()])) : autoValueClassName; ParameterizedTypeName superClass = ParameterizedTypeName.get(typeAdapterClass, autoValueTypeName); ParameterSpec gsonParam = ParameterSpec.builder(Gson.class, "gson").build(); MethodSpec.Builder constructor = MethodSpec.constructorBuilder() .addParameter(gsonParam); if (!typeParams.isEmpty()) { ParameterSpec typeAdapter = ParameterSpec .builder(ArrayTypeName.of(Type.class), "types") .build(); constructor.addParameter(typeAdapter); constructor.addStatement("typeArgs = $N", typeAdapter); } ImmutableMap adapters = createFields(properties); if (useFieldNamePolicy) { constructor.addStatement("$1T fields = new $1T()", ParameterizedTypeName.get(ArrayList.class, String.class)); for (Property prop : properties) { constructor.addStatement("fields.add($S)", prop.humanName); } } constructor.addStatement("this.gson = gson"); if (useFieldNamePolicy) { constructor.addStatement("this.realFieldNames = $T.renameFields($T.class, fields, gson.fieldNamingStrategy())", ClassName.get("com.ryanharter.auto.value.gson.internal", "Util"), superClassType); } ClassName jsonAdapter = ClassName.get(TypeAdapter.class); TypeSpec.Builder classBuilder = TypeSpec.classBuilder(gsonTypeAdapterName) .addOriginatingElement(autoValueType) .addTypeVariables(typeParams) .addModifiers(FINAL) .superclass(superClass) .addFields(adapters.values()); if (useFieldNamePolicy) { classBuilder.addField(FieldSpec.builder(ParameterizedTypeName.get(Map.class, String.class, String.class), "realFieldNames", PRIVATE, FINAL).build()); } classBuilder.addField(FieldSpec.builder(Gson.class, "gson", PRIVATE, FINAL).build()) .addMethod(constructor.build()) .addMethod(createWriteMethod(autoValueTypeName, properties, adapters, jsonAdapter, typeParams)) .addMethod(createReadMethod(className, autoValueClassName, autoValueTypeName, properties, adapters, jsonAdapter, typeParams, builderContext, processingEnvironment)) .addMethod(MethodSpec.methodBuilder("toString") .addAnnotation(Override.class) .addModifiers(PUBLIC) .returns(String.class) .addStatement("return new $T().append($S).append($S).append($S).toString()", StringBuilder.class, "TypeAdapter(", Joiner.on(".").join(autoValueClassName.simpleNames()), ")" ) .build()); if (!typeParams.isEmpty()) { classBuilder.addField(FieldSpec.builder(Type[].class, "typeArgs", PRIVATE, FINAL).build()); } return classBuilder.build(); } private static void addConditionalAdapterAssignment(CodeBlock.Builder block, FieldSpec adapterField, Property prop, ClassName jsonAdapter, List typeParams) { TypeName type = prop.type.isPrimitive() ? prop.type.box() : prop.type; ParameterizedTypeName adp = ParameterizedTypeName.get(jsonAdapter, type); block.addStatement("$T $N = this.$N", adapterField.type, adapterField, adapterField); block.beginControlFlow("if ($N == null)", adapterField); if (prop.type instanceof ParameterizedTypeName || prop.type instanceof TypeVariableName) { block.addStatement("this.$N = $N = ($T) gson.getAdapter($L)", adapterField, adapterField, adp, makeParameterizedType(prop.type, typeParams)); } else { block.addStatement("this.$N = $N = gson.getAdapter($T.class)", adapterField, adapterField, type); } block.endControlFlow(); } private static void addBuilderFieldSetting(Builder block, Property prop, FieldSpec adapter, ParameterSpec jsonReader, FieldSpec builder, BuilderContext builderContext) { Stream setterMethodSpecs = builderContext.setters().get(prop.humanName).stream() .map(setterMethod -> MethodSpec.overriding(setterMethod).build()); // If setter param type matches field type Optional setter = setterMethodSpecs // Find setter with param type equal to field type. .filter(methodSpec -> methodSpec.parameters.get(0).type.equals(prop.builderType)) .findFirst(); if (setter.isPresent()) { block.addStatement("$N.$N($N.read($N))", builder, setter.get(), adapter, jsonReader); } else { // Optional fields are not supported. // See https://github.com/rharter/auto-value-gson/issues/161 String errorMsg = "Setter not found for " + prop.element; throw new IllegalArgumentException(errorMsg); } } private static void addFieldSetting(CodeBlock.Builder block, Property prop, Map fields, FieldSpec adapter, ParameterSpec jsonReader) { block.addStatement("$N = $N.read($N)", fields.get(prop), adapter, jsonReader); } private MethodSpec createWriteMethod(TypeName autoValueClassName, List properties, ImmutableMap adapters, ClassName jsonAdapter, List typeParams) { ParameterSpec jsonWriter = ParameterSpec.builder(JsonWriter.class, "jsonWriter").build(); ParameterSpec annotatedParam = ParameterSpec.builder(autoValueClassName, "object").build(); MethodSpec.Builder writeMethod = MethodSpec.methodBuilder("write") .addAnnotation(Override.class) .addAnnotation(AnnotationSpec.builder(SuppressWarnings.class) .addMember("value", "\"unchecked\"") .build()) .addModifiers(PUBLIC) .addParameter(jsonWriter) .addParameter(annotatedParam) .addException(IOException.class); writeMethod.beginControlFlow("if ($N == null)", annotatedParam); writeMethod.addStatement("$N.nullValue()", jsonWriter); writeMethod.addStatement("return"); writeMethod.endControlFlow(); writeMethod.addStatement("$N.beginObject()", jsonWriter); for (Property prop : properties) { if (prop.isTransient()) { continue; } if (prop.hasSerializedNameAnnotation()) { writeMethod.addStatement("$N.name($S)", jsonWriter, prop.serializedName()); } else if (useFieldNamePolicy) { writeMethod.addStatement("$N.name(realFieldNames.get($S))", jsonWriter, prop.humanName); } else { writeMethod.addStatement("$N.name($S)", jsonWriter, prop.humanName); } // for adapters handling non-primitive values, initialize the // adapter only when the value is actually present (non-null), // otherwise use a generic method of writing the null value FieldSpec adapterField = adapters.get(prop.type); CodeBlock.Builder block = CodeBlock.builder(); if (!prop.type.isPrimitive()) { writeMethod.beginControlFlow("if ($N.$N() == null)", annotatedParam, prop.methodName); writeMethod.addStatement("$N.nullValue()", jsonWriter); writeMethod.nextControlFlow("else"); addConditionalAdapterAssignment(block, adapterField, prop, jsonAdapter, typeParams); writeMethod.addCode(block.build()); writeMethod.addStatement("$N.write($N, $N.$N())", adapterField, jsonWriter, annotatedParam, prop.methodName); writeMethod.endControlFlow(); } else { block.add("{\n"); block.indent(); addConditionalAdapterAssignment(block, adapterField, prop, jsonAdapter, typeParams); block.addStatement("$N.write($N, $N.$N())", adapterField, jsonWriter, annotatedParam, prop.methodName); block.unindent(); block.add("}\n"); writeMethod.addCode(block.build()); } } writeMethod.addStatement("$N.endObject()", jsonWriter); return writeMethod.build(); } private MethodSpec createReadMethod(ClassName className, ClassName autoValueClassName, TypeName autoValueTypeName, List properties, ImmutableMap adapters, ClassName jsonAdapter, List typeParams, @Nullable BuilderContext builderContext, ProcessingEnvironment processingEnvironment) { ParameterSpec jsonReader = ParameterSpec.builder(JsonReader.class, "jsonReader").build(); MethodSpec.Builder readMethod = MethodSpec.methodBuilder("read") .addAnnotation(Override.class) .addAnnotation(AnnotationSpec.builder(SuppressWarnings.class) .addMember("value", "\"unchecked\"") .build()) .addModifiers(PUBLIC) .returns(autoValueTypeName) .addParameter(jsonReader) .addException(IOException.class); // Validate the builderContext if there is one. if (builderContext != null) { if (!builderContext.buildMethod().isPresent()) { processingEnvironment.getMessager() .printMessage( Diagnostic.Kind.ERROR, "Could not determine the build method. Make sure it is named \"build\".", builderContext.builderType()); return readMethod.build(); } Set builderMethods = builderContext.builderMethods(); if (builderMethods.size() > 1) { Set annotatedMethods = builderMethods.stream() .filter(e -> MoreElements.isAnnotationPresent(e, AutoValueGsonBuilder.class)) .collect(Collectors.toSet()); if (annotatedMethods.size() > 1) { processingEnvironment.getMessager() .printMessage( Diagnostic.Kind.ERROR, "Too many @AutoValueGsonBuilder annotated builder methods.", annotatedMethods.stream().findAny().get() ); return readMethod.build(); } if (annotatedMethods.isEmpty()) { processingEnvironment.getMessager().printMessage( Diagnostic.Kind.ERROR, "Too many builder methods. Annotate builder method with @AutoValueGsonBuilder.", builderMethods.stream().findAny().get() ); return readMethod.build(); } } } ClassName token = ClassName.get(JsonToken.NULL.getDeclaringClass()); readMethod.beginControlFlow("if ($N.peek() == $T.NULL)", jsonReader, token); readMethod.addStatement("$N.nextNull()", jsonReader); readMethod.addStatement("return null"); readMethod.endControlFlow(); readMethod.addStatement("$N.beginObject()", jsonReader); // Will be empty if using a AutoValue builder Map fields = new LinkedHashMap<>(properties.size()); // Will be absent if not using AutoValue builder Optional builderField = Optional.ofNullable(builderContext) .map(ctx -> FieldSpec .builder(TypeName.get(ctx.builderType().asType()), "builder") .build()); if (builderField.isPresent()) { Set builderMethods = builderContext.builderMethods(); if (builderMethods.size() == 0) { // If no builder method defined, instantiate directly. readMethod.addStatement("$T $N = new $T.$L()", builderField.get().type, builderField.get(), className, builderContext.builderType().getSimpleName()); } else { ExecutableElement builderMethod; if (builderMethods.size() == 1) { // If there is only 1, use it. builderMethod = builderMethods.stream().findFirst().get(); } else { // Otherwise, find the only builder method that is annotated. Set annotatedMethods = builderMethods.stream() .filter(e -> MoreElements.isAnnotationPresent(e, AutoValueGsonBuilder.class)) .collect(Collectors.toSet()); if (annotatedMethods.size() == 1) { builderMethod = annotatedMethods.stream().findFirst().get(); } else { throw new IllegalStateException(); } } readMethod.addStatement("$T $N = $T.$N()", builderField.get().type, builderField.get(), autoValueClassName, builderMethod.getSimpleName()); } } else { // add the properties for (Property prop : properties) { TypeName fieldType = prop.type; FieldSpec field = FieldSpec.builder(fieldType, prop.humanName).build(); fields.put(prop, field); CodeBlock defaultValue = getDefaultValue(prop, field); readMethod.addCode("$[$T $N = ", field.type, field); if (defaultValue != null) { readMethod.addCode(defaultValue); } else { readMethod.addCode("$L", "null"); } readMethod.addCode(";\n$]"); } } readMethod.beginControlFlow("while ($N.hasNext())", jsonReader); FieldSpec name = FieldSpec.builder(String.class, "_name").build(); readMethod.addStatement("$T $N = $N.nextName()", name.type, name, jsonReader); readMethod.beginControlFlow("if ($N.peek() == $T.NULL)", jsonReader, token); readMethod.addStatement("$N.nextNull()", jsonReader); readMethod.addStatement("continue"); readMethod.endControlFlow(); readMethod.beginControlFlow("switch ($N)", name); for (Property prop : properties) { if (prop.isTransient()) { continue; } if (prop.hasSerializedNameAnnotation()) { for (String alternate : prop.serializedNameAlternate()) { readMethod.addCode("case $S:\n", alternate); } readMethod.beginControlFlow("case $S:", prop.serializedName()); FieldSpec adapterField = adapters.get(prop.type); CodeBlock.Builder block = CodeBlock.builder(); addConditionalAdapterAssignment(block, adapterField, prop, jsonAdapter, typeParams); if (builderField.isPresent()) { addBuilderFieldSetting(block, prop, adapterField, jsonReader, builderField.get(), builderContext); } else { addFieldSetting(block, prop, fields, adapterField, jsonReader); } readMethod.addCode(block.build()); readMethod.addStatement("break"); readMethod.endControlFlow(); } } // skip value if field is not serialized... readMethod.beginControlFlow("default:"); for (Property prop : properties) { if (prop.isTransient()) { continue; } if (!prop.hasSerializedNameAnnotation()) { if (useFieldNamePolicy) { readMethod.beginControlFlow("if (realFieldNames.get($S).equals(_name))", prop.humanName); } else { readMethod.beginControlFlow("if ($S.equals(_name))", prop.humanName); } FieldSpec adapterField = adapters.get(prop.type); CodeBlock.Builder block = CodeBlock.builder(); addConditionalAdapterAssignment(block, adapterField, prop, jsonAdapter, typeParams); if (builderField.isPresent()) { addBuilderFieldSetting(block, prop, adapterField, jsonReader, builderField.get(), builderContext); } else { addFieldSetting(block, prop, fields, adapterField, jsonReader); } readMethod.addCode(block.build()); readMethod.addStatement("continue"); readMethod.endControlFlow(); } } readMethod.addStatement("$N.skipValue()", jsonReader); readMethod.endControlFlow(); // default case readMethod.endControlFlow(); // switch readMethod.endControlFlow(); // while readMethod.addStatement("$N.endObject()", jsonReader); if (builderField.isPresent()) { readMethod.addStatement("return $N.$L()", builderField.get(), builderContext.buildMethod().get().getSimpleName()); } else { StringBuilder format = new StringBuilder("return new "); format.append(className.simpleName().replaceAll("\\$", "")); if (autoValueTypeName instanceof ParameterizedTypeName) { format.append("<>"); } format.append("("); Iterator iterator = fields.values().iterator(); while (iterator.hasNext()) { iterator.next(); format.append("$N"); if (iterator.hasNext()) format.append(", "); } format.append(")"); readMethod.addStatement(format.toString(), fields.values().toArray()); } return readMethod.build(); } /** * Returns a default value for initializing well-known types, or else {@code null}. */ private CodeBlock getDefaultValue(Property prop, FieldSpec field) { if (field.type.isPrimitive()) { String defaultValue = getDefaultPrimitiveValue(field.type); if (defaultValue != null) { return CodeBlock.of("$L", defaultValue); } else { return CodeBlock.of("$T.valueOf(null)", field.type); } } if (prop.nullable()) { return null; } TypeMirror type = prop.element.getReturnType(); if (type.getKind() != TypeKind.DECLARED) { return null; } TypeElement typeElement = MoreTypes.asTypeElement(type); if (typeElement == null) { return null; } return null; } /** * * @param type * @return the default primitive value as a String. Returns null if unable to determine default value */ private String getDefaultPrimitiveValue(TypeName type) { String valueString = null; try { Class primitiveClass = Primitives.unwrap(Class.forName(type.box().toString())); if (primitiveClass != null) { Object defaultValue = Defaults.defaultValue(primitiveClass); if (defaultValue != null) { valueString = defaultValue.toString(); if (!Strings.isNullOrEmpty(valueString)) { switch (type.toString()) { case "double": valueString = valueString + "d"; break; case "float": valueString = valueString + "f"; break; case "long": valueString = valueString + "L"; break; case "char": valueString = "'" + valueString + "'"; break; } } } } } catch (ClassNotFoundException ignored) { //Swallow and return null } return valueString; } private static CodeBlock makeParameterizedType(TypeName typeName, List typeParams) { CodeBlock.Builder block = CodeBlock.builder(); if (typeName instanceof TypeVariableName) { block.add("$T.get(typeArgs[$L])", TypeToken.class, typeParams.indexOf(typeName)); } else{ ParameterizedTypeName paramType = (ParameterizedTypeName) typeName; block.add("$T.getParameterized($T.class", TypeToken.class, paramType.rawType); for (TypeName type : paramType.typeArguments) { buildParameterizedTypeArguments(block, type, typeParams); } block.add(")"); } return block.build(); } private static void buildParameterizedTypeArguments(CodeBlock.Builder block, TypeName typeArg, List typeParams) { block.add(", "); if (typeArg instanceof ParameterizedTypeName) { // type argument itself can be parameterized ParameterizedTypeName paramTypeArg = (ParameterizedTypeName) typeArg; block.add("$T.getParameterized($T.class", TypeToken.class, paramTypeArg.rawType); for (TypeName type : paramTypeArg.typeArguments) { buildParameterizedTypeArguments(block, type, typeParams); } block.add(").getType()"); } else if (typeArg instanceof TypeVariableName) { block.add("typeArgs[$L]", typeParams.indexOf(typeArg)); } else if (typeArg instanceof WildcardTypeName) { WildcardTypeName wildcard = (WildcardTypeName) typeArg; TypeName target; String method; if (wildcard.lowerBounds.size() == 1) { target = wildcard.lowerBounds.get(0); method = "supertypeOf"; } else if (wildcard.upperBounds.size() == 1) { target = wildcard.upperBounds.get(0); method = "subtypeOf"; } else { throw new IllegalArgumentException( "Unrepresentable wildcard type. Cannot have more than one bound: " + wildcard); } block.add("$T.$L($T.class)", WildcardUtil.class, method, target); } else { block.add("$T.class", typeArg); } } @Nullable private static ClassName rawType(TypeName typeName) { if (typeName instanceof ClassName) { return (ClassName) typeName; } else if (typeName instanceof ArrayTypeName) { return rawType(((ArrayTypeName) typeName).componentType); } else if (typeName instanceof ParameterizedTypeName) { return ((ParameterizedTypeName) typeName).rawType; } else if (typeName instanceof WildcardTypeName) { return rawType(((WildcardTypeName) typeName).upperBounds.get(0)); } else { return null; } } private static String proguardNameOf(TypeName typeName) { if (typeName instanceof ClassName) { return ((ClassName) typeName).reflectionName(); } else if (typeName instanceof ArrayTypeName) { return proguardNameOf(((ArrayTypeName) typeName).componentType) + "[]"; } else if (typeName instanceof ParameterizedTypeName) { return ((ParameterizedTypeName) typeName).rawType.reflectionName(); } else if (typeName instanceof TypeVariableName) { return "java.lang.Object"; } else { throw new UnsupportedOperationException("Unrecognized TypeName type: " + typeName); } } } ProguardConfig.java000066400000000000000000000056271375747333100357510ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/main/java/com/ryanharter/auto/value/gsonpackage com.ryanharter.auto.value.gson; import com.google.auto.value.AutoValue; import com.squareup.javapoet.ClassName; import java.io.IOException; import java.io.Writer; import java.util.List; import javax.annotation.processing.Filer; import javax.lang.model.element.Element; import static javax.tools.StandardLocation.CLASS_OUTPUT; /** * Represents a proguard configuration for a given spec. This covers two main areas: *
    *
  • Keeping the target class name to GenerateTypeAdapter.FACTORY's reflective lookup of the adapter.
  • *
  • Keeping the generated adapter class name + public constructor for reflective lookup.
  • *
*

* Each rule is intended to be as specific and targeted as possible to reduce footprint, and each is * conditioned on usage of the original target type. *

* To keep this processor as an {@code ISOLATING} incremental processor, we generate one file per * target class with a deterministic name (see {@link #outputFile}) with an appropriate originating * element. */ @AutoValue abstract class ProguardConfig { abstract ClassName targetClass(); abstract ClassName adapterName(); abstract List adapterConstructorParams(); abstract String outputFile(); static ProguardConfig create( ClassName targetClass, ClassName adapterName, List adapterConstructorParams) { String outputFile = "META-INF/proguard/avg-" + targetClass.canonicalName() + ".pro"; return new AutoValue_ProguardConfig(targetClass, adapterName, adapterConstructorParams, outputFile); } /** Writes this to {@code filer}. */ final void writeTo(Filer filer, Element... originatingElements) throws IOException { try (Writer writer = filer.createResource(CLASS_OUTPUT, "", outputFile(), originatingElements) .openWriter()) { writeTo(writer); } } private void writeTo(Appendable out) throws IOException { // // -if class {the target class} // -keepnames class {the target class} // -if class {the target class} // -keep class {the generated adapter} { // (...); // } // String targetName = targetClass().reflectionName(); String adapterCanonicalName = adapterName().canonicalName(); // Keep the class name for GenerateTypeAdapter.FACTORY's reflective lookup based on it out.append("-if class ") .append(targetName) .append("\n"); out.append("-keepnames class ") .append(targetName) .append("\n"); out.append("-if class ") .append(targetName) .append("\n"); out.append("-keep class ") .append(adapterCanonicalName) .append(" {\n"); // Keep the constructor for GenerateTypeAdapter.FACTORY's reflective lookup String constructorArgs = String.join(",", adapterConstructorParams()); out.append(" (") .append(constructorArgs) .append(");\n"); out.append("}\n"); } } auto-value-gson-1.3.1/auto-value-gson-extension/src/test/000077500000000000000000000000001375747333100233655ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/000077500000000000000000000000001375747333100243065ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/com/000077500000000000000000000000001375747333100250645ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/com/ryanharter/000077500000000000000000000000001375747333100272435ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/com/ryanharter/auto/000077500000000000000000000000001375747333100302135ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100313275ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100322755ustar00rootroot00000000000000AutoValueGsonExtensionTest.java000077500000000000000000004036051375747333100403650ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-extension/src/test/java/com/ryanharter/auto/value/gsonpackage com.ryanharter.auto.value.gson; import com.google.auto.value.processor.AutoValueProcessor; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Lists; import com.google.testing.compile.Compilation; import com.google.testing.compile.JavaFileObjects; import java.io.ByteArrayInputStream; import java.io.InputStream; import java.io.OutputStream; import java.io.Reader; import java.io.StringReader; import java.io.Writer; import java.net.URI; import java.nio.charset.Charset; import java.util.Arrays; import java.util.Map; import java.util.TreeMap; import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; import static com.google.common.truth.Truth.assertAbout; import static com.google.common.truth.Truth.assertThat; import static com.google.testing.compile.CompilationSubject.compilations; import static com.google.testing.compile.Compiler.javac; import static com.google.testing.compile.JavaSourceSubjectFactory.javaSource; import static com.google.testing.compile.JavaSourcesSubjectFactory.javaSources; import static javax.tools.JavaFileObject.Kind.OTHER; @RunWith(JUnit4.class) public class AutoValueGsonExtensionTest { private JavaFileObject nullable; private JavaFileObject typeTargetNullable; @Before public void setup() { nullable = JavaFileObjects.forSourceString("com.ryanharter.auto.value.gson.Nullable", "" + "package com.ryanharter.auto.value.gson;\n" + "import java.lang.annotation.Retention;\n" + "import java.lang.annotation.Target;\n" + "import static java.lang.annotation.ElementType.METHOD;\n" + "import static java.lang.annotation.ElementType.PARAMETER;\n" + "import static java.lang.annotation.ElementType.FIELD;\n" + "import static java.lang.annotation.RetentionPolicy.SOURCE;\n" + "@Retention(SOURCE)\n" + "@Target({METHOD, PARAMETER, FIELD})\n" + "public @interface Nullable {\n" + "}"); typeTargetNullable = JavaFileObjects.forSourceString("com.ryanharter.auto.value.gson.Nullable", "" + "package com.ryanharter.auto.value.gson;\n" + "import java.lang.annotation.Retention;\n" + "import java.lang.annotation.Target;\n" + "import static java.lang.annotation.ElementType.TYPE_USE;\n" + "import static java.lang.annotation.RetentionPolicy.SOURCE;\n" + "@Retention(SOURCE)\n" + "@Target(TYPE_USE)\n" + "public @interface Nullable {\n" + "}"); } @Test public void simple() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.gson.annotations.SerializedName;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.common.collect.ImmutableMap;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import java.io.IOException;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "@AutoValue abstract class Test {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" // Reference type + "abstract String a();\n" // Array type + "abstract int[] b();\n" // Primitive type + "abstract int c();\n" // SerializedName + "@SerializedName(\"_D\") abstract String d();\n" // Nullable type + "@Nullable abstract String e();\n" // Parametrized type, multiple parameters + "abstract ImmutableMap f();\n" // Parametrized type, single parameter + "abstract Set g();\n" // Nested parameterized type + "abstract Map> h();\n" // SerializedName with alternate + "@SerializedName(value = \"_I\", alternate = {\"_I_1\", \"_I_2\"}) abstract String i();\n" // Nullable collection type + "@Nullable abstract List j();\n" // Deeply nested parameterized type + "abstract Map>>>> o();\n" + " static class TestTypeAdapter extends TypeAdapter {\n" + " @Override public void write(JsonWriter out, String value) throws IOException {}\n" + " @Override public String read(JsonReader in) throws IOException { return null; }\n" + " }\n" + " static class TestListTypeAdapter extends TypeAdapter> {\n" + " @Override public void write(JsonWriter out, List value) throws IOException {}\n" + " @Override public List read(JsonReader in) throws IOException { return null; }\n" + " }\n" + " static class TestTypeAdapterFactory implements TypeAdapterFactory {\n" + " @Override public TypeAdapter create(Gson gson, TypeToken type) { return null; }\n" + " }\n" + "}\n" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package test;\n" + "\n" + "import com.google.common.collect.ImmutableMap;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import com.ryanharter.auto.value.gson.internal.WildcardUtil;\n" + "import java.io.IOException;\n" + "import java.lang.Integer;\n" + "import java.lang.Number;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.ArrayList;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(String a, int[] b, int c, String d, @Nullable String e,\n" + " ImmutableMap f, Set g, Map> h, " + "String i,\n" + " @Nullable List j,\n" + " Map>>>> o) {\n" + " super(a, b, c, d, e, f, g, h, i, j, o);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + " private volatile TypeAdapter array__int_adapter;\n" + " private volatile TypeAdapter int__adapter;\n" + " private volatile TypeAdapter> " + "immutableMap__string_number_adapter;\n" + " private volatile TypeAdapter> set__string_adapter;\n" + " private volatile TypeAdapter>> " + "map__string_set__string_adapter;\n" + " private volatile TypeAdapter> " + "list__wildcard__string_adapter;\n" + " private volatile TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"a\");\n" + " fields.add(\"b\");\n" + " fields.add(\"c\");\n" + " fields.add(\"d\");\n" + " fields.add(\"e\");\n" + " fields.add(\"f\");\n" + " fields.add(\"g\");\n" + " fields.add(\"h\");\n" + " fields.add(\"i\");\n" + " fields.add(\"j\");\n" + " fields.add(\"o\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Test.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"a\"));\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"b\"));\n" + " if (object.b() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter array__int_adapter = this.array__int_adapter;\n" + " if (array__int_adapter == null) {\n" + " this.array__int_adapter = array__int_adapter = gson.getAdapter(int[].class);\n" + " }\n" + " array__int_adapter.write(jsonWriter, object.b());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"c\"));\n" + " {\n" + " TypeAdapter int__adapter = this.int__adapter;\n" + " if (int__adapter == null) {\n" + " this.int__adapter = int__adapter = gson.getAdapter(Integer.class);\n" + " }\n" + " int__adapter.write(jsonWriter, object.c());\n" + " }\n" + " jsonWriter.name(\"_D\");\n" + " if (object.d() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.d());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"e\"));\n" + " if (object.e() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.e());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"f\"));\n" + " if (object.f() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> immutableMap__string_number_adapter " + "= this.immutableMap__string_number_adapter;\n" + " if (immutableMap__string_number_adapter == null) {\n" + " this.immutableMap__string_number_adapter = " + "immutableMap__string_number_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(ImmutableMap.class, String.class, Number.class)" + ");\n" + " }\n" + " immutableMap__string_number_adapter.write(jsonWriter, object.f());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"g\"));\n" + " if (object.g() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> set__string_adapter = this.set__string_adapter;\n" + " if (set__string_adapter == null) {\n" + " this.set__string_adapter = set__string_adapter = (TypeAdapter>) " + "gson.getAdapter(TypeToken.getParameterized(Set.class, String.class));\n" + " }\n" + " set__string_adapter.write(jsonWriter, object.g());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"h\"));\n" + " if (object.h() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>> map__string_set__string_adapter = this" + ".map__string_set__string_adapter;\n" + " if (map__string_set__string_adapter == null) {\n" + " this.map__string_set__string_adapter = map__string_set__string_adapter = " + "(TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Set.class, String.class).getType()));\n" + " }\n" + " map__string_set__string_adapter.write(jsonWriter, object.h());\n" + " }\n" + " jsonWriter.name(\"_I\");\n" + " if (object.i() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.i());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"j\"));\n" + " if (object.j() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> list__wildcard__string_adapter = this" + ".list__wildcard__string_adapter;\n" + " if (list__wildcard__string_adapter == null) {\n" + " this.list__wildcard__string_adapter = list__wildcard__string_adapter = " + "(TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(List" + ".class, WildcardUtil.subtypeOf(String.class)));\n" + " }\n" + " list__wildcard__string_adapter.write(jsonWriter, object.j());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"o\"));\n" + " if (object.o() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter =" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " if " + "(map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "== null) {\n" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "= map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + " = (TypeAdapter>>>>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, " + "TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken" + ".getParameterized(Map.class, String.class, WildcardUtil.subtypeOf(String.class))" + ".getType()).getType()).getType()).getType()));\n" + " }\n" + " " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + ".write(jsonWriter, object.o());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " String a = null;\n" + " int[] b = null;\n" + " int c = 0;\n" + " String d = null;\n" + " String e = null;\n" + " ImmutableMap f = null;\n" + " Set g = null;\n" + " Map> h = null;\n" + " String i = null;\n" + " List j = null;\n" + " Map>>>> o = null;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " case \"_D\": {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " d = string_adapter.read(jsonReader);\n" + " break;\n" + " }\n" + " case \"_I_1\":\n" + " case \"_I_2\":\n" + " case \"_I\": {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " i = string_adapter.read(jsonReader);\n" + " break;\n" + " }\n" + " default: {\n" + " if (realFieldNames.get(\"a\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " a = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"b\").equals(_name)) {\n" + " TypeAdapter array__int_adapter = this.array__int_adapter;\n" + " if (array__int_adapter == null) {\n" + " this.array__int_adapter = array__int_adapter = gson.getAdapter(int[]" + ".class);\n" + " }\n" + " b = array__int_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"c\").equals(_name)) {\n" + " TypeAdapter int__adapter = this.int__adapter;\n" + " if (int__adapter == null) {\n" + " this.int__adapter = int__adapter = gson.getAdapter(Integer.class);\n" + " }\n" + " c = int__adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"e\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " e = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"f\").equals(_name)) {\n" + " TypeAdapter> " + "immutableMap__string_number_adapter = this.immutableMap__string_number_adapter;\n" + " if (immutableMap__string_number_adapter == null) {\n" + " this.immutableMap__string_number_adapter = " + "immutableMap__string_number_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(ImmutableMap.class, String.class, Number.class)" + ");\n" + " }\n" + " f = immutableMap__string_number_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"g\").equals(_name)) {\n" + " TypeAdapter> set__string_adapter = this.set__string_adapter;\n" + " if (set__string_adapter == null) {\n" + " this.set__string_adapter = set__string_adapter = " + "(TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(Set.class, " + "String.class));\n" + " }\n" + " g = set__string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"h\").equals(_name)) {\n" + " TypeAdapter>> map__string_set__string_adapter = " + "this.map__string_set__string_adapter;\n" + " if (map__string_set__string_adapter == null) {\n" + " this.map__string_set__string_adapter = map__string_set__string_adapter" + " = (TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized" + "(Map.class, String.class, TypeToken.getParameterized(Set.class, String.class).getType" + "()));\n" + " }\n" + " h = map__string_set__string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"j\").equals(_name)) {\n" + " TypeAdapter> list__wildcard__string_adapter = " + "this.list__wildcard__string_adapter;\n" + " if (list__wildcard__string_adapter == null) {\n" + " this.list__wildcard__string_adapter = list__wildcard__string_adapter =" + " (TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(List" + ".class, WildcardUtil.subtypeOf(String.class)));\n" + " }\n" + " j = list__wildcard__string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"o\").equals(_name)) {\n" + " TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter =" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " if " + "(map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "== null) {\n" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "= map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + " = (TypeAdapter>>>>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, " + "TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken" + ".getParameterized(Map.class, String.class, WildcardUtil.subtypeOf(String.class))" + ".getType()).getType()).getType()).getType()));\n" + " }\n" + " o = " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + ".read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(a, b, c, d, e, f, g, h, i, j, o);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}" ); assertAbout(javaSources()) .that(Arrays.asList(nullable, source)) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void simpleExternal() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.ryanharter.auto.value.gson.GenerateTypeAdapter;\n" + "@GenerateTypeAdapter\n" + "@AutoValue\n" + "abstract class Test {\n" + " abstract String a();\n" + "}\n" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/Test_GsonTypeAdapter", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import java.io.IOException;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class Test_GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + "\n" + " private final Gson gson;\n" + "\n" + " Test_GsonTypeAdapter(Gson gson) {\n" + " this.gson = gson;\n" + " }\n" + "\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(\"a\");\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + "\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " String a = null;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (\"a\".equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " a = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(a);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")\").toString();\n" + " }\n" + "}\n" ); JavaFileObject expectedProguard = proguardResource( "META-INF/proguard/avg-test.Test.pro", "-if class test.Test\n" + "-keepnames class test.Test\n" + "-if class test.Test\n" + "-keep class test.Test_GsonTypeAdapter {\n" + " (com.google.gson.Gson);\n" + "}\n"); assertAbout(javaSources()) .that(ImmutableList.of(source)) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected) .and() .generatesFiles(expectedProguard); } @Test public void simpleExternalGeneric() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.ryanharter.auto.value.gson.GenerateTypeAdapter;\n" + "@GenerateTypeAdapter\n" + "@AutoValue\n" + "abstract class Test {\n" + " abstract T a();\n" + "}\n" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/Test_GsonTypeAdapter", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import java.io.IOException;\n" + "import java.lang.reflect.Type;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class Test_GsonTypeAdapter extends TypeAdapter> {\n" + " private volatile TypeAdapter T_adapter;\n" + "\n" + " private final Gson gson;\n" + "\n" + " private final Type[] typeArgs;\n" + "\n" + " Test_GsonTypeAdapter(Gson gson, Type[] types) {\n" + " typeArgs = types;\n" + " this.gson = gson;\n" + " }\n" + "\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(\"a\");\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter T_adapter = this.T_adapter;\n" + " if (T_adapter == null) {\n" + " this.T_adapter = T_adapter = (TypeAdapter) gson.getAdapter(TypeToken" + ".get(typeArgs[0]));\n" + " }\n" + " T_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + "\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " T a = null;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (\"a\".equals(_name)) {\n" + " TypeAdapter T_adapter = this.T_adapter;\n" + " if (T_adapter == null) {\n" + " this.T_adapter = T_adapter = (TypeAdapter) gson.getAdapter(TypeToken.get(typeArgs[0]));\n" + " }\n" + " a = T_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test<>(a);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")\").toString();\n" + " }\n" + "}" ); JavaFileObject expectedProguard = proguardResource( "META-INF/proguard/avg-test.Test.pro", "-if class test.Test\n" + "-keepnames class test.Test\n" + "-if class test.Test\n" + "-keep class test.Test_GsonTypeAdapter {\n" + " (com.google.gson.Gson,java.lang.reflect.Type[]);\n" + "}\n"); assertAbout(javaSources()) .that(ImmutableList.of(source)) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected) .and() .generatesFiles(expectedProguard); } @Test public void simpleNoFieldNamePolicy() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.gson.annotations.SerializedName;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.common.collect.ImmutableMap;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import java.io.IOException;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "@AutoValue abstract class Test {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" // Reference type + "abstract String a();\n" // Array type + "abstract int[] b();\n" // Primitive type + "abstract int c();\n" // SerializedName + "@SerializedName(\"_D\") abstract String d();\n" // Nullable type + "@Nullable abstract String e();\n" // Parametrized type, multiple parameters + "abstract ImmutableMap f();\n" // Parametrized type, single parameter + "abstract Set g();\n" // Nested parameterized type + "abstract Map> h();\n" // SerializedName with alternate + "@SerializedName(value = \"_I\", alternate = {\"_I_1\", \"_I_2\"}) abstract String i();\n" // Nullable collection type + "@Nullable abstract List j();\n" // Deeply nested parameterized type + "abstract Map>>>> o();\n" + " static class TestTypeAdapter extends TypeAdapter {\n" + " @Override public void write(JsonWriter out, String value) throws IOException {}\n" + " @Override public String read(JsonReader in) throws IOException { return null; }\n" + " }\n" + " static class TestListTypeAdapter extends TypeAdapter> {\n" + " @Override public void write(JsonWriter out, List value) throws IOException {}\n" + " @Override public List read(JsonReader in) throws IOException { return null; }\n" + " }\n" + " static class TestTypeAdapterFactory implements TypeAdapterFactory {\n" + " @Override public TypeAdapter create(Gson gson, TypeToken type) { return null; }\n" + " }\n" + "}\n" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package " + "test;\n" + "\n" + "import com.google.common.collect.ImmutableMap;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.ryanharter.auto.value.gson.internal.WildcardUtil;\n" + "import java.io.IOException;\n" + "import java.lang.Integer;\n" + "import java.lang.Number;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(String a, int[] b, int c, String d, @Nullable String e,\n" + " ImmutableMap f, Set g, Map> h, " + "String i,\n" + " @Nullable List j,\n" + " Map>>>> o) {\n" + " super(a, b, c, d, e, f, g, h, i, j, o);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + " private volatile TypeAdapter array__int_adapter;\n" + " private volatile TypeAdapter int__adapter;\n" + " private volatile TypeAdapter> " + "immutableMap__string_number_adapter;\n" + " private volatile TypeAdapter> set__string_adapter;\n" + " private volatile TypeAdapter>> " + "map__string_set__string_adapter;\n" + " private volatile TypeAdapter> " + "list__wildcard__string_adapter;\n" + " private volatile TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " this.gson = gson;\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(\"a\");\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.name(\"b\");\n" + " if (object.b() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter array__int_adapter = this.array__int_adapter;\n" + " if (array__int_adapter == null) {\n" + " this.array__int_adapter = array__int_adapter = gson.getAdapter(int[].class);\n" + " }\n" + " array__int_adapter.write(jsonWriter, object.b());\n" + " }\n" + " jsonWriter.name(\"c\");\n" + " {\n" + " TypeAdapter int__adapter = this.int__adapter;\n" + " if (int__adapter == null) {\n" + " this.int__adapter = int__adapter = gson.getAdapter(Integer.class);\n" + " }\n" + " int__adapter.write(jsonWriter, object.c());\n" + " }\n" + " jsonWriter.name(\"_D\");\n" + " if (object.d() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.d());\n" + " }\n" + " jsonWriter.name(\"e\");\n" + " if (object.e() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.e());\n" + " }\n" + " jsonWriter.name(\"f\");\n" + " if (object.f() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> immutableMap__string_number_adapter " + "= this.immutableMap__string_number_adapter;\n" + " if (immutableMap__string_number_adapter == null) {\n" + " this.immutableMap__string_number_adapter = " + "immutableMap__string_number_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(ImmutableMap.class, String.class, Number.class)" + ");\n" + " }\n" + " immutableMap__string_number_adapter.write(jsonWriter, object.f());\n" + " }\n" + " jsonWriter.name(\"g\");\n" + " if (object.g() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> set__string_adapter = this.set__string_adapter;\n" + " if (set__string_adapter == null) {\n" + " this.set__string_adapter = set__string_adapter = (TypeAdapter>) " + "gson.getAdapter(TypeToken.getParameterized(Set.class, String.class));\n" + " }\n" + " set__string_adapter.write(jsonWriter, object.g());\n" + " }\n" + " jsonWriter.name(\"h\");\n" + " if (object.h() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>> map__string_set__string_adapter = this" + ".map__string_set__string_adapter;\n" + " if (map__string_set__string_adapter == null) {\n" + " this.map__string_set__string_adapter = map__string_set__string_adapter = " + "(TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Set.class, String.class).getType()));\n" + " }\n" + " map__string_set__string_adapter.write(jsonWriter, object.h());\n" + " }\n" + " jsonWriter.name(\"_I\");\n" + " if (object.i() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.i());\n" + " }\n" + " jsonWriter.name(\"j\");\n" + " if (object.j() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> list__wildcard__string_adapter = this" + ".list__wildcard__string_adapter;\n" + " if (list__wildcard__string_adapter == null) {\n" + " this.list__wildcard__string_adapter = list__wildcard__string_adapter = " + "(TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(List" + ".class, WildcardUtil.subtypeOf(String.class)));\n" + " }\n" + " list__wildcard__string_adapter.write(jsonWriter, object.j());\n" + " }\n" + " jsonWriter.name(\"o\");\n" + " if (object.o() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter =" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " if " + "(map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "== null) {\n" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "= map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + " = (TypeAdapter>>>>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, " + "TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken" + ".getParameterized(Map.class, String.class, WildcardUtil.subtypeOf(String.class))" + ".getType()).getType()).getType()).getType()));\n" + " }\n" + " " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + ".write(jsonWriter, object.o());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " String a = null;\n" + " int[] b = null;\n" + " int c = 0;\n" + " String d = null;\n" + " String e = null;\n" + " ImmutableMap f = null;\n" + " Set g = null;\n" + " Map> h = null;\n" + " String i = null;\n" + " List j = null;\n" + " Map>>>> o = null;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " case \"_D\": {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " d = string_adapter.read(jsonReader);\n" + " break;\n" + " }\n" + " case \"_I_1\":\n" + " case \"_I_2\":\n" + " case \"_I\": {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " i = string_adapter.read(jsonReader);\n" + " break;\n" + " }\n" + " default: {\n" + " if (\"a\".equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " a = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"b\".equals(_name)) {\n" + " TypeAdapter array__int_adapter = this.array__int_adapter;\n" + " if (array__int_adapter == null) {\n" + " this.array__int_adapter = array__int_adapter = gson.getAdapter(int[]" + ".class);\n" + " }\n" + " b = array__int_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"c\".equals(_name)) {\n" + " TypeAdapter int__adapter = this.int__adapter;\n" + " if (int__adapter == null) {\n" + " this.int__adapter = int__adapter = gson.getAdapter(Integer.class);\n" + " }\n" + " c = int__adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"e\".equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " e = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"f\".equals(_name)) {\n" + " TypeAdapter> " + "immutableMap__string_number_adapter = this.immutableMap__string_number_adapter;\n" + " if (immutableMap__string_number_adapter == null) {\n" + " this.immutableMap__string_number_adapter = " + "immutableMap__string_number_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(ImmutableMap.class, String.class, Number.class)" + ");\n" + " }\n" + " f = immutableMap__string_number_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"g\".equals(_name)) {\n" + " TypeAdapter> set__string_adapter = this.set__string_adapter;\n" + " if (set__string_adapter == null) {\n" + " this.set__string_adapter = set__string_adapter = " + "(TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(Set.class, " + "String.class));\n" + " }\n" + " g = set__string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"h\".equals(_name)) {\n" + " TypeAdapter>> map__string_set__string_adapter = " + "this.map__string_set__string_adapter;\n" + " if (map__string_set__string_adapter == null) {\n" + " this.map__string_set__string_adapter = map__string_set__string_adapter" + " = (TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized" + "(Map.class, String.class, TypeToken.getParameterized(Set.class, String.class).getType" + "()));\n" + " }\n" + " h = map__string_set__string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"j\".equals(_name)) {\n" + " TypeAdapter> list__wildcard__string_adapter = " + "this.list__wildcard__string_adapter;\n" + " if (list__wildcard__string_adapter == null) {\n" + " this.list__wildcard__string_adapter = list__wildcard__string_adapter =" + " (TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(List" + ".class, WildcardUtil.subtypeOf(String.class)));\n" + " }\n" + " j = list__wildcard__string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (\"o\".equals(_name)) {\n" + " TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter =" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " if " + "(map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "== null) {\n" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "= map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + " = (TypeAdapter>>>>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, " + "TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken" + ".getParameterized(Map.class, String.class, WildcardUtil.subtypeOf(String.class))" + ".getType()).getType()).getType()).getType()));\n" + " }\n" + " o = " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + ".read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(a, b, c, d, e, f, g, h, i, j, o);\n" + " }\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}" ); assertAbout(javaSources()) .that(Arrays.asList(nullable, source)) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void privateMethod() { // Private methods exclude them from AVGson consideration JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "@AutoValue abstract class Test {\n" + " private static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" // Reference type + " public abstract String a();\n" // Array type + " public abstract int[] b();\n" // Primitive type + " public abstract int c();\n" // Parametrized type, multiple parameters + " public abstract Map e();\n" // Parametrized type, single parameter + " public abstract Set f();\n" // Nested parameterized type + " public abstract Map> g();\n" // Nullable type + " @Nullable abstract String i();\n" + "}\n" ); Compilation compilation = javac() .withProcessors(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compile(nullable, source); assertAbout(compilations()) .that(compilation) .succeeded(); assertThat(compilation.generatedSourceFiles()).hasSize(1); assertThat(compilation.generatedSourceFiles().get(0).getName()).endsWith("AutoValue_Test.java"); } @Test public void simpleWithBuilder() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.gson.annotations.SerializedName;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.common.collect.ImmutableMap;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import java.io.IOException;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "@AutoValue abstract class Test {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" // Reference type + "abstract String a();\n" // Array type + "abstract int[] b();\n" // Primitive type + "abstract int c();\n" // SerializedName + "@SerializedName(\"_D\") abstract String d();\n" // Nullable type + "@Nullable abstract String e();\n" // Parametrized type, multiple parameters + "abstract ImmutableMap f();\n" // Parametrized type, single parameter + "abstract Set g();\n" // Nested parameterized type + "abstract Map> h();\n" // SerializedName with alternate + "@SerializedName(value = \"_I\", alternate = {\"_I_1\", \"_I_2\"}) abstract String i();\n" // Nullable collection type + "@Nullable abstract List j();\n" // Deeply nested parameterized type + "abstract Map>>>> o();\n" + " @AutoValue.Builder static abstract class Builder {\n" + " abstract Builder a(String a);\n" + " abstract Builder b(int[] b);\n" + " abstract Builder c(int c);\n" + " abstract Builder d(String d);\n" + " abstract Builder e(String e);\n" + " abstract Builder f(ImmutableMap f);\n" + " abstract Builder g(Set g);\n" + " abstract Builder h(Map> h);\n" + " abstract Builder i(String i);\n" + " abstract Builder j(List j);\n" + " abstract Builder o(Map>>>> o);\n" + " abstract Test build();\n" + " }\n" + " static class TestTypeAdapter extends TypeAdapter {\n" + " @Override public void write(JsonWriter out, String value) throws IOException {}\n" + " @Override public String read(JsonReader in) throws IOException { return null; }\n" + " }\n" + " static class TestListTypeAdapter extends TypeAdapter> {\n" + " @Override public void write(JsonWriter out, List value) throws IOException {}\n" + " @Override public List read(JsonReader in) throws IOException { return null; }\n" + " }\n" + " static class TestTypeAdapterFactory implements TypeAdapterFactory {\n" + " @Override public TypeAdapter create(Gson gson, TypeToken type) { return null; }\n" + " }\n" + "}\n" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package test;\n" + "\n" + "import com.google.common.collect.ImmutableMap;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import com.ryanharter.auto.value.gson.internal.WildcardUtil;\n" + "import java.io.IOException;\n" + "import java.lang.Integer;\n" + "import java.lang.Number;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.ArrayList;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import java.util.Set;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(String a, int[] b, int c, String d, @Nullable String e,\n" + " ImmutableMap f, Set g, Map> h, " + "String i,\n" + " @Nullable List j,\n" + " Map>>>> o) {\n" + " super(a, b, c, d, e, f, g, h, i, j, o);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + " private volatile TypeAdapter array__int_adapter;\n" + " private volatile TypeAdapter int__adapter;\n" + " private volatile TypeAdapter> " + "immutableMap__string_number_adapter;\n" + " private volatile TypeAdapter> set__string_adapter;\n" + " private volatile TypeAdapter>> " + "map__string_set__string_adapter;\n" + " private volatile TypeAdapter> " + "list__wildcard__string_adapter;\n" + " private volatile TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"a\");\n" + " fields.add(\"b\");\n" + " fields.add(\"c\");\n" + " fields.add(\"d\");\n" + " fields.add(\"e\");\n" + " fields.add(\"f\");\n" + " fields.add(\"g\");\n" + " fields.add(\"h\");\n" + " fields.add(\"i\");\n" + " fields.add(\"j\");\n" + " fields.add(\"o\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Test.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"a\"));\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"b\"));\n" + " if (object.b() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter array__int_adapter = this.array__int_adapter;\n" + " if (array__int_adapter == null) {\n" + " this.array__int_adapter = array__int_adapter = gson.getAdapter(int[].class);\n" + " }\n" + " array__int_adapter.write(jsonWriter, object.b());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"c\"));\n" + " {\n" + " TypeAdapter int__adapter = this.int__adapter;\n" + " if (int__adapter == null) {\n" + " this.int__adapter = int__adapter = gson.getAdapter(Integer.class);\n" + " }\n" + " int__adapter.write(jsonWriter, object.c());\n" + " }\n" + " jsonWriter.name(\"_D\");\n" + " if (object.d() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.d());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"e\"));\n" + " if (object.e() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.e());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"f\"));\n" + " if (object.f() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> immutableMap__string_number_adapter " + "= this.immutableMap__string_number_adapter;\n" + " if (immutableMap__string_number_adapter == null) {\n" + " this.immutableMap__string_number_adapter = " + "immutableMap__string_number_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(ImmutableMap.class, String.class, Number.class)" + ");\n" + " }\n" + " immutableMap__string_number_adapter.write(jsonWriter, object.f());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"g\"));\n" + " if (object.g() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> set__string_adapter = this.set__string_adapter;\n" + " if (set__string_adapter == null) {\n" + " this.set__string_adapter = set__string_adapter = (TypeAdapter>) " + "gson.getAdapter(TypeToken.getParameterized(Set.class, String.class));\n" + " }\n" + " set__string_adapter.write(jsonWriter, object.g());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"h\"));\n" + " if (object.h() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>> map__string_set__string_adapter = this" + ".map__string_set__string_adapter;\n" + " if (map__string_set__string_adapter == null) {\n" + " this.map__string_set__string_adapter = map__string_set__string_adapter = " + "(TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Set.class, String.class).getType()));\n" + " }\n" + " map__string_set__string_adapter.write(jsonWriter, object.h());\n" + " }\n" + " jsonWriter.name(\"_I\");\n" + " if (object.i() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.i());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"j\"));\n" + " if (object.j() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> list__wildcard__string_adapter = this" + ".list__wildcard__string_adapter;\n" + " if (list__wildcard__string_adapter == null) {\n" + " this.list__wildcard__string_adapter = list__wildcard__string_adapter = " + "(TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(List" + ".class, WildcardUtil.subtypeOf(String.class)));\n" + " }\n" + " list__wildcard__string_adapter.write(jsonWriter, object.j());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"o\"));\n" + " if (object.o() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>>>>> " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter =" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " if " + "(map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "== null) {\n" + " this" + ".map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter " + "= map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + " = (TypeAdapter>>>>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, " + "TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken" + ".getParameterized(Map.class, String.class, WildcardUtil.subtypeOf(String.class))" + ".getType()).getType()).getType()).getType()));\n" + " }\n" + " " + "map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter" + ".write(jsonWriter, object.o());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " Test.Builder builder = new AutoValue_Test.Builder();\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " case \"_D\": {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " builder.d(string_adapter.read(jsonReader));\n" + " break;\n" + " }\n" + " case \"_I_1\":\n" + " case \"_I_2\":\n" + " case \"_I\": {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " builder.i(string_adapter.read(jsonReader));\n" + " break;\n" + " }\n" + " default: {\n" + " if (realFieldNames.get(\"a\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " builder.a(string_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"b\").equals(_name)) {\n" + " TypeAdapter array__int_adapter = this.array__int_adapter;\n" + " if (array__int_adapter == null) {\n" + " this.array__int_adapter = array__int_adapter = gson.getAdapter(int[].class);\n" + " }\n" + " builder.b(array__int_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"c\").equals(_name)) {\n" + " TypeAdapter int__adapter = this.int__adapter;\n" + " if (int__adapter == null) {\n" + " this.int__adapter = int__adapter = gson.getAdapter(Integer.class);\n" + " }\n" + " builder.c(int__adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"e\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " builder.e(string_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"f\").equals(_name)) {\n" + " TypeAdapter> immutableMap__string_number_adapter = this.immutableMap__string_number_adapter;\n" + " if (immutableMap__string_number_adapter == null) {\n" + " this.immutableMap__string_number_adapter = immutableMap__string_number_adapter = (TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(ImmutableMap.class, String.class, Number.class));\n" + " }\n" + " builder.f(immutableMap__string_number_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"g\").equals(_name)) {\n" + " TypeAdapter> set__string_adapter = this.set__string_adapter;\n" + " if (set__string_adapter == null) {\n" + " this.set__string_adapter = set__string_adapter = (TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(Set.class, String.class));\n" + " }\n" + " builder.g(set__string_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"h\").equals(_name)) {\n" + " TypeAdapter>> map__string_set__string_adapter = this.map__string_set__string_adapter;\n" + " if (map__string_set__string_adapter == null) {\n" + " this.map__string_set__string_adapter = map__string_set__string_adapter = (TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Set.class, String.class).getType()));\n" + " }\n" + " builder.h(map__string_set__string_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"j\").equals(_name)) {\n" + " TypeAdapter> list__wildcard__string_adapter = this.list__wildcard__string_adapter;\n" + " if (list__wildcard__string_adapter == null) {\n" + " this.list__wildcard__string_adapter = list__wildcard__string_adapter = (TypeAdapter>) gson.getAdapter(TypeToken.getParameterized(List.class, WildcardUtil.subtypeOf(String.class)));\n" + " }\n" + " builder.j(list__wildcard__string_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"o\").equals(_name)) {\n" + " TypeAdapter>>>>> map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter = this.map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter;\n" + " if (map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter == null) {\n" + " this.map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter = map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter = (TypeAdapter>>>>>) gson.getAdapter(TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map.class, String.class, TypeToken.getParameterized(Map.class, String.class, WildcardUtil.subtypeOf(String.class)).getType()).getType()).getType()).getType()));\n" + " }\n" + " builder.o(map__string_map__string_map__string_map__string_map__string_wildcard__string_adapter.read(jsonReader));\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return builder.build();\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}" ); assertAbout(javaSources()) .that(Arrays.asList(nullable, source)) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void propertyMethodReferencedWithPrefix() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue public abstract class Test {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import java.io.IOException;\n" + "import java.lang.Boolean;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.ArrayList;\n" + "import java.util.Map;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(String name, boolean awesome) {\n" + " super(name, awesome);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + " private volatile TypeAdapter boolean__adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"name\");\n" + " fields.add(\"awesome\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Test.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"name\"));\n" + " if (object.getName() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.getName());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"awesome\"));\n" + " {\n" + " TypeAdapter boolean__adapter = this.boolean__adapter;\n" + " if (boolean__adapter == null) {\n" + " this.boolean__adapter = boolean__adapter = gson.getAdapter(Boolean.class);\n" + " }\n" + " boolean__adapter.write(jsonWriter, object.isAwesome());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " String name = null;\n" + " boolean awesome = false;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (realFieldNames.get(\"name\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " name = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"awesome\").equals(_name)) {\n" + " TypeAdapter boolean__adapter = this.boolean__adapter;\n" + " if (boolean__adapter == null) {\n" + " this.boolean__adapter = boolean__adapter = gson.getAdapter(Boolean" + ".class);\n" + " }\n" + " awesome = boolean__adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(name, awesome);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}"); assertAbout(javaSource()) .that(source) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void handlesDefaultAccessTypeAdapterMethod() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue abstract class Test {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" + " abstract String getName();\n" + " abstract boolean isAwesome();\n" + "}" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import java.io.IOException;\n" + "import java.lang.Boolean;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.ArrayList;\n" + "import java.util.Map;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(String name, boolean awesome) {\n" + " super(name, awesome);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + " private volatile TypeAdapter boolean__adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"name\");\n" + " fields.add(\"awesome\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Test.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"name\"));\n" + " if (object.getName() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.getName());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"awesome\"));\n" + " {\n" + " TypeAdapter boolean__adapter = this.boolean__adapter;\n" + " if (boolean__adapter == null) {\n" + " this.boolean__adapter = boolean__adapter = gson.getAdapter(Boolean.class);\n" + " }\n" + " boolean__adapter.write(jsonWriter, object.isAwesome());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " String name = null;\n" + " boolean awesome = false;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (realFieldNames.get(\"name\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " name = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"awesome\").equals(_name)) {\n" + " TypeAdapter boolean__adapter = this.boolean__adapter;\n" + " if (boolean__adapter == null) {\n" + " this.boolean__adapter = boolean__adapter = gson.getAdapter(Boolean" + ".class);\n" + " }\n" + " awesome = boolean__adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(name, awesome);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}"); assertAbout(javaSource()) .that(source) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void generatesNothingWithoutTypeAdapterMethod() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "@AutoValue public abstract class Test {\n" + " public abstract String a();\n" + " public abstract boolean b();\n" + "}" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "" + "package test;\n" + "\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\"com.google.auto.value.processor.AutoValueProcessor\")\n" + " final class AutoValue_Test extends Test {\n" + "\n" + " private final String a;\n" + " private final boolean b;\n" + "\n" + " AutoValue_Test(\n" + " String a,\n" + " boolean b) {\n" + " if (a == null) {\n" + " throw new NullPointerException(\"Null a\");\n" + " }\n" + " this.a = a;\n" + " this.b = b;\n" + " }\n" + "\n" + " @Override\n" + " public String a() {\n" + " return a;\n" + " }\n" + "\n" + " @Override\n" + " public boolean b() {\n" + " return b;\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return \"Test{\"\n" + " + \"a=\" + a + \", \"\n" + " + \"b=\" + b\n" + " + \"}\";\n" + " }\n" + "\n" + " @Override\n" + " public boolean equals(Object o) {\n" + " if (o == this) {\n" + " return true;\n" + " }\n" + " if (o instanceof Test) {\n" + " Test that = (Test) o;\n" + " return this.a.equals(that.a())\n" + " && this.b == that.b();\n" + " }\n" + " return false;\n" + " }\n" + "\n" + " @Override\n" + " public int hashCode() {\n" + " int h$ = 1;\n" + " h$ *= 1000003;\n" + " h$ ^= a.hashCode();\n" + " h$ *= 1000003;\n" + " h$ ^= b ? 1231 : 1237;\n" + " return h$;\n" + " }\n" + "\n" + "}"); assertAbout(javaSource()) .that(source) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .withWarningCount(2) .and() .generatesSources(expected); } @Test public void emitsWarningForWrongTypeAdapterTypeArgument() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;" + " }\n" + " public abstract String a();\n" + " public abstract boolean b();\n" + "}" ); JavaFileObject source2 = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "public class Bar {\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, source2)) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .withWarningContaining("Found static method returning TypeAdapter on " + "test.Foo class. Skipping GsonTypeAdapter generation."); } @Test public void emitsWarningForNoTypeAdapterTypeArgument() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;" + " }\n" + " public abstract String a();\n" + " public abstract boolean b();\n" + "}" ); assertAbout(javaSource()) .that(source1) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .withWarningContaining("Found static method returning TypeAdapter with no type " + "arguments, skipping GsonTypeAdapter generation."); } @Test public void compilesWithCapitalPackageName() { JavaFileObject source1 = JavaFileObjects.forSourceString("MyPackage.Foo", "" + "package MyPackage;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Foo.GsonTypeAdapter(gson);" + " }\n" + " public abstract String a();\n" + " public abstract boolean b();\n" + "}" ); assertAbout(javaSource()) .that(source1) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .withWarningCount(2); } @Test public void generatesCorrectDefaultCharPrimitiveValue() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue public abstract class Test {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" + "public abstract char c();\n" + "}\n" ); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import java.io.IOException;\n" + "import java.lang.Character;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.ArrayList;\n" + "import java.util.Map;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(char c) {\n" + " super(c);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter char__adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"c\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Test.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"c\"));\n" + " {\n" + " TypeAdapter char__adapter = this.char__adapter;\n" + " if (char__adapter == null) {\n" + " this.char__adapter = char__adapter = gson.getAdapter(Character.class);\n" + " }\n" + " char__adapter.write(jsonWriter, object.c());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " char c = '\u0000';\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (realFieldNames.get(\"c\").equals(_name)) {\n" + " TypeAdapter char__adapter = this.char__adapter;\n" + " if (char__adapter == null) {\n" + " this.char__adapter = char__adapter = gson.getAdapter(Character.class);\n" + " }\n" + " c = char__adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(c);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}" ); assertAbout(javaSources()) .that(Arrays.asList(nullable, source)) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void handlesGenericTypes() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import java.lang.reflect.Type;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter> typeAdapter(Gson gson, Type[] " + "types) {\n" + " return new AutoValue_Foo.GsonTypeAdapter<>(gson, types); }\n" + " public abstract C c();\n" + " public abstract A a();\n" + " public abstract B b();\n" + " public abstract List list();\n" + " public abstract Map> map();\n" + " public abstract String d();\n" + "}" ); JavaFileObject expected = JavaFileObjects.forSourceString("test.AutoValue_Test", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import java.io.IOException;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.lang.reflect.Type;\n" + "import java.util.ArrayList;\n" + "import java.util.List;\n" + "import java.util.Map;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Foo extends $AutoValue_Foo {\n" + " AutoValue_Foo(C c, A a, B b, List list, Map> map, String d) {\n" + " super(c, a, b, list, map, d);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter>" + " {\n" + " private volatile TypeAdapter C_adapter;\n" + " private volatile TypeAdapter A_adapter;\n" + " private volatile TypeAdapter B_adapter;\n" + " private volatile TypeAdapter> list__A_adapter;\n" + " private volatile TypeAdapter>> map__string_list__C_adapter;\n" + " private volatile TypeAdapter string_adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " private final Type[] typeArgs;\n" + " GsonTypeAdapter(Gson gson, Type[] types) {\n" + " typeArgs = types;\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"c\");\n" + " fields.add(\"a\");\n" + " fields.add(\"b\");\n" + " fields.add(\"list\");\n" + " fields.add(\"map\");\n" + " fields.add(\"d\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Foo.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Foo object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"c\"));\n" + " if (object.c() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter C_adapter = this.C_adapter;\n" + " if (C_adapter == null) {\n" + " this.C_adapter = C_adapter = (TypeAdapter) gson.getAdapter(TypeToken.get" + "(typeArgs[2]));\n" + " }\n" + " C_adapter.write(jsonWriter, object.c());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"a\"));\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter A_adapter = this.A_adapter;\n" + " if (A_adapter == null) {\n" + " this.A_adapter = A_adapter = (TypeAdapter) gson.getAdapter(TypeToken.get" + "(typeArgs[0]));\n" + " }\n" + " A_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"b\"));\n" + " if (object.b() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter B_adapter = this.B_adapter;\n" + " if (B_adapter == null) {\n" + " this.B_adapter = B_adapter = (TypeAdapter) gson.getAdapter(TypeToken.get" + "(typeArgs[1]));\n" + " }\n" + " B_adapter.write(jsonWriter, object.b());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"list\"));\n" + " if (object.list() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter> list__A_adapter = this.list__A_adapter;\n" + " if (list__A_adapter == null) {\n" + " this.list__A_adapter = list__A_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(List.class, typeArgs[0]));\n" + " }\n" + " list__A_adapter.write(jsonWriter, object.list());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"map\"));\n" + " if (object.map() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter>> map__string_list__C_adapter = this" + ".map__string_list__C_adapter;\n" + " if (map__string_list__C_adapter == null) {\n" + " this.map__string_list__C_adapter = map__string_list__C_adapter = " + "(TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(List.class, typeArgs[2]).getType()));\n" + " }\n" + " map__string_list__C_adapter.write(jsonWriter, object.map());\n" + " }\n" + " jsonWriter.name(realFieldNames.get(\"d\"));\n" + " if (object.d() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.d());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Foo read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " C c = null;\n" + " A a = null;\n" + " B b = null;\n" + " List list = null;\n" + " Map> map = null;\n" + " String d = null;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (realFieldNames.get(\"c\").equals(_name)) {\n" + " TypeAdapter C_adapter = this.C_adapter;\n" + " if (C_adapter == null) {\n" + " this.C_adapter = C_adapter = (TypeAdapter) gson.getAdapter" + "(TypeToken.get(typeArgs[2]));\n" + " }\n" + " c = C_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"a\").equals(_name)) {\n" + " TypeAdapter A_adapter = this.A_adapter;\n" + " if (A_adapter == null) {\n" + " this.A_adapter = A_adapter = (TypeAdapter) gson.getAdapter" + "(TypeToken.get(typeArgs[0]));\n" + " }\n" + " a = A_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"b\").equals(_name)) {\n" + " TypeAdapter B_adapter = this.B_adapter;\n" + " if (B_adapter == null) {\n" + " this.B_adapter = B_adapter = (TypeAdapter) gson.getAdapter" + "(TypeToken.get(typeArgs[1]));\n" + " }\n" + " b = B_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"list\").equals(_name)) {\n" + " TypeAdapter> list__A_adapter = this.list__A_adapter;\n" + " if (list__A_adapter == null) {\n" + " this.list__A_adapter = list__A_adapter = (TypeAdapter>) gson" + ".getAdapter(TypeToken.getParameterized(List.class, typeArgs[0]));\n" + " }\n" + " list = list__A_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"map\").equals(_name)) {\n" + " TypeAdapter>> map__string_list__C_adapter = this" + ".map__string_list__C_adapter;\n" + " if (map__string_list__C_adapter == null) {\n" + " this.map__string_list__C_adapter = map__string_list__C_adapter = " + "(TypeAdapter>>) gson.getAdapter(TypeToken.getParameterized(Map" + ".class, String.class, TypeToken.getParameterized(List.class, typeArgs[2]).getType()));\n" + " }\n" + " map = map__string_list__C_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " if (realFieldNames.get(\"d\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " d = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Foo<>(c, a, b, list, map, d);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Foo\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}"); assertAbout(javaSource()) .that(source1) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void transientProperties() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import io.sweers.autotransient.AutoTransient;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "@AutoValue public abstract class Test {\n" // a is not ignored, must be present in the JSON + " public abstract String a();\n" // transientProperty is nullable and ignored, so a default value is not required + " @Nullable @AutoTransient public abstract String transientProperty();\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" + "}"); JavaFileObject expected = JavaFileObjects.forSourceString("test/AutoValue_Test", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.stream.JsonReader;\n" + "import com.google.gson.stream.JsonToken;\n" + "import com.google.gson.stream.JsonWriter;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "import com.ryanharter.auto.value.gson.internal.Util;\n" + "import java.io.IOException;\n" + "import java.lang.Override;\n" + "import java.lang.String;\n" + "import java.lang.StringBuilder;\n" + "import java.lang.SuppressWarnings;\n" + "import java.util.ArrayList;\n" + "import java.util.Map;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.AutoValueGsonExtension\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValue_Test extends $AutoValue_Test {\n" + " AutoValue_Test(String a, @Nullable String transientProperty) {\n" + " super(a, transientProperty);\n" + " }\n" + "\n" + " static final class GsonTypeAdapter extends TypeAdapter {\n" + " private volatile TypeAdapter string_adapter;\n" + " private final Map realFieldNames;\n" + " private final Gson gson;\n" + " GsonTypeAdapter(Gson gson) {\n" + " ArrayList fields = new ArrayList();\n" + " fields.add(\"a\");\n" + " fields.add(\"transientProperty\");\n" + " this.gson = gson;\n" + " this.realFieldNames = Util.renameFields($AutoValue_Test.class, fields, gson" + ".fieldNamingStrategy());\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public void write(JsonWriter jsonWriter, Test object) throws IOException {\n" + " if (object == null) {\n" + " jsonWriter.nullValue();\n" + " return;\n" + " }\n" + " jsonWriter.beginObject();\n" + " jsonWriter.name(realFieldNames.get(\"a\"));\n" + " if (object.a() == null) {\n" + " jsonWriter.nullValue();\n" + " } else {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " string_adapter.write(jsonWriter, object.a());\n" + " }\n" + " jsonWriter.endObject();\n" + " }\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public Test read(JsonReader jsonReader) throws IOException {\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " return null;\n" + " }\n" + " jsonReader.beginObject();\n" + " String a = null;\n" + " String transientProperty = null;\n" + " while (jsonReader.hasNext()) {\n" + " String _name = jsonReader.nextName();\n" + " if (jsonReader.peek() == JsonToken.NULL) {\n" + " jsonReader.nextNull();\n" + " continue;\n" + " }\n" + " switch (_name) {\n" + " default: {\n" + " if (realFieldNames.get(\"a\").equals(_name)) {\n" + " TypeAdapter string_adapter = this.string_adapter;\n" + " if (string_adapter == null) {\n" + " this.string_adapter = string_adapter = gson.getAdapter(String.class);\n" + " }\n" + " a = string_adapter.read(jsonReader);\n" + " continue;\n" + " }\n" + " jsonReader.skipValue();\n" + " }\n" + " }\n" + " }\n" + " jsonReader.endObject();\n" + " return new AutoValue_Test(a, transientProperty);\n" + " }\n" + "\n" + " @Override\n" + " public String toString() {\n" + " return new StringBuilder().append(\"TypeAdapter(\").append(\"Test\").append(\")" + "\").toString();\n" + " }\n" + " }\n" + "}"); Map nullables = new TreeMap<>(); nullables.put("using method-target @Nullable", nullable); nullables.put("using type-target @Nullable", typeTargetNullable); nullables.forEach( (nullableName, nullableObject) -> { assertAbout(javaSources()) .that(Arrays.asList(nullableObject, source)) .named(nullableName) .withCompilerOptions("-A" + AutoValueGsonExtension.USE_FIELD_NAME_POLICY) .processedWith( new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesSources(expected); }); } @Test public void transientRequiredProperty_shouldFail() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import io.sweers.autotransient.AutoTransient;\n" + "import com.ryanharter.auto.value.gson.Nullable;\n" + "@AutoValue public abstract class Test {\n" + " @AutoTransient public abstract String transientProperty();\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return new AutoValue_Test.GsonTypeAdapter(gson);\n" + " }\n" + "}"); assertAbout(javaSources()) .that(Arrays.asList(nullable, source)) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .failsToCompile() .withErrorContaining("Required property cannot be transient!"); } @Test public void proguardRulesShouldUseReflectionName() { JavaFileObject source = JavaFileObjects.forSourceString("test.Test", "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.ryanharter.auto.value.gson.GenerateTypeAdapter;\n" + "\n" + "public abstract class Test {\n" + " @GenerateTypeAdapter\n" + " @AutoValue\n" + " public static abstract class InnerClass {\n" + " abstract String property();\n" + " }\n" + "}"); JavaFileObject expectedProguard = proguardResource( "META-INF/proguard/avg-test.Test.InnerClass.pro", "-if class test.Test$InnerClass\n" + "-keepnames class test.Test$InnerClass\n" + "-if class test.Test$InnerClass\n" + "-keep class test.Test_InnerClass_GsonTypeAdapter {\n" + " (com.google.gson.Gson);\n" + "}\n"); assertAbout(javaSources()) .that(ImmutableList.of(source)) .processedWith(new AutoValueProcessor(Lists.newArrayList(new AutoValueGsonExtension()))) .compilesWithoutError() .and() .generatesFiles(expectedProguard); } private static JavaFileObject proguardResource(String path, String source) { return new ResourceFile(path, source); } private static class ResourceFile extends SimpleJavaFileObject { final String source; final long lastModified; ResourceFile(String path, String source) { super(URI.create(path), OTHER); this.source = source; this.lastModified = System.currentTimeMillis(); } @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) { return source; } @Override public OutputStream openOutputStream() { throw new IllegalStateException(); } @Override public InputStream openInputStream() { return new ByteArrayInputStream(source.getBytes(Charset.defaultCharset())); } @Override public Writer openWriter() { throw new IllegalStateException(); } @Override public Reader openReader(boolean ignoreEncodingErrors) { return new StringReader(source); } @Override public long getLastModified() { return lastModified; } } } auto-value-gson-1.3.1/auto-value-gson-factory/000077500000000000000000000000001375747333100212525ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/build.gradle000066400000000000000000000041261375747333100235340ustar00rootroot00000000000000import org.gradle.internal.jvm.Jvm plugins { id 'com.github.johnrengelman.shadow' version '4.0.1' id 'java-library' id 'maven-publish' } // // disable default jar tasks configurations.runtimeOnly.artifacts.removeAll { it.archiveTask.is jar } tasks.getByName('jar').enabled = false // create extra configuration for shaded dependencies, so they're not included in the pom def shadedConfig = configurations.create('compileShaded') configurations.compileOnly.extendsFrom(shadedConfig) shadowJar { minimize() classifier = '' configurations = [shadedConfig] relocate 'com.google.auto.common', 'autovaluegson.factory.shaded.com.google.auto.common' relocate 'com.google.common', 'autovaluegson.factory.shaded.com.google.common' relocate 'com.google.thirdparty', 'autovaluegson.factory.shaded.com.google.thirdparty' exclude 'afu/**' exclude 'org/**' exclude 'com/google/errorprone/annotations/**' exclude 'com/google/j2objc/annotations/**' exclude 'javax/**' exclude 'META-INF/maven/com.google.auto/auto-common/**' } artifacts { runtime shadowJar archives shadowJar } // sourceCompatibility = JavaVersion.VERSION_1_8 targetCompatibility = JavaVersion.VERSION_1_8 dependencies { annotationProcessor "net.ltgt.gradle.incap:incap-processor:0.2" annotationProcessor 'com.google.auto.service:auto-service:1.0-rc5' compileOnly "net.ltgt.gradle.incap:incap:0.2" compileOnly 'com.google.auto.service:auto-service:1.0-rc5' api 'com.google.auto.value:auto-value-annotations:1.7' api project(':auto-value-gson-extension') compileShaded 'com.google.auto:auto-common:0.10' compileShaded 'com.google.guava:guava:27.0-jre' implementation 'com.squareup:javapoet:1.12.0' implementation deps.gson testImplementation 'junit:junit:4.12' testImplementation 'com.google.truth:truth:0.42' testImplementation 'com.google.testing.compile:compile-testing:0.16' testImplementation files(Jvm.current().getToolsJar()) } apply from: rootProject.file('gradle/gradle-mvn-push.gradle') auto-value-gson-1.3.1/auto-value-gson-factory/gradle.properties000066400000000000000000000001451375747333100246260ustar00rootroot00000000000000POM_ARTIFACT_ID=auto-value-gson-factory POM_NAME=AutoValue: Gson Extension Factory POM_PACKAGING=jar auto-value-gson-1.3.1/auto-value-gson-factory/src/000077500000000000000000000000001375747333100220415ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/000077500000000000000000000000001375747333100227655ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/000077500000000000000000000000001375747333100237065ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/000077500000000000000000000000001375747333100244645ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/ryanharter/000077500000000000000000000000001375747333100266435ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/ryanharter/auto/000077500000000000000000000000001375747333100276135ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100307275ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100316755ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/ryanharter/auto/value/gson/factory/000077500000000000000000000000001375747333100333445ustar00rootroot00000000000000AutoValueGsonAdapterFactoryProcessor.java000066400000000000000000000361311375747333100434210ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/main/java/com/ryanharter/auto/value/gson/factorypackage com.ryanharter.auto.value.gson.factory; import com.google.auto.common.GeneratedAnnotations; import com.google.auto.common.Visibility; import com.google.auto.service.AutoService; import com.google.auto.value.AutoValue; import com.google.common.collect.ImmutableSet; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.TypeAdapterFactory; import com.google.gson.reflect.TypeToken; import com.ryanharter.auto.value.gson.AutoValueGsonExtension; import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory; import com.squareup.javapoet.AnnotationSpec; import com.squareup.javapoet.ClassName; import com.squareup.javapoet.JavaFile; import com.squareup.javapoet.MethodSpec; import com.squareup.javapoet.ParameterSpec; import com.squareup.javapoet.ParameterizedTypeName; import com.squareup.javapoet.TypeName; import com.squareup.javapoet.TypeSpec; import com.squareup.javapoet.TypeVariableName; import java.io.IOException; import java.lang.reflect.ParameterizedType; import java.util.List; import java.util.Optional; import java.util.Set; import java.util.stream.Collectors; import javax.annotation.processing.AbstractProcessor; import javax.annotation.processing.ProcessingEnvironment; import javax.annotation.processing.Processor; import javax.annotation.processing.RoundEnvironment; import javax.lang.model.SourceVersion; import javax.lang.model.element.Element; import javax.lang.model.element.ExecutableElement; import javax.lang.model.element.PackageElement; import javax.lang.model.element.TypeElement; import javax.lang.model.element.VariableElement; import javax.lang.model.type.TypeKind; import javax.lang.model.type.TypeMirror; import javax.lang.model.util.ElementFilter; import javax.lang.model.util.Elements; import javax.lang.model.util.Types; import net.ltgt.gradle.incap.IncrementalAnnotationProcessor; import static com.google.auto.common.MoreElements.getPackage; import static com.ryanharter.auto.value.gson.AutoValueGsonExtension.GENERATED_COMMENTS; import static java.util.stream.Collectors.toList; import static javax.lang.model.element.Modifier.ABSTRACT; import static javax.lang.model.element.Modifier.FINAL; import static javax.lang.model.element.Modifier.PRIVATE; import static javax.lang.model.element.Modifier.PUBLIC; import static javax.lang.model.element.Modifier.STATIC; import static javax.tools.Diagnostic.Kind.ERROR; import static net.ltgt.gradle.incap.IncrementalAnnotationProcessorType.AGGREGATING; /** * Generates a Gson {@link TypeAdapterFactory} that adapts all {@link AutoValue} annotated * Gson serializable classes. */ @IncrementalAnnotationProcessor(AGGREGATING) @AutoService(Processor.class) public class AutoValueGsonAdapterFactoryProcessor extends AbstractProcessor { private Types typeUtils; private Elements elementUtils; @Override public Set getSupportedAnnotationTypes() { return ImmutableSet.of(AutoValue.class.getName(), GsonTypeAdapterFactory.class.getName()); } @Override public SourceVersion getSupportedSourceVersion() { return SourceVersion.latestSupported(); } @Override public synchronized void init(ProcessingEnvironment processingEnv) { super.init(processingEnv); typeUtils = processingEnv.getTypeUtils(); elementUtils = processingEnv.getElementUtils(); } @Override public boolean process(Set annotations, RoundEnvironment roundEnv) { Set adapterFactories = roundEnv.getElementsAnnotatedWith(GsonTypeAdapterFactory.class); if (adapterFactories.isEmpty()) { return false; } Set autoValueElements = roundEnv.getElementsAnnotatedWith(AutoValue.class); List elements = autoValueElements.stream() .map(e -> (TypeElement) e) .filter(e -> AutoValueGsonExtension.isApplicable(e, processingEnv.getMessager())) .sorted((o1, o2) -> { final String o1Name = classNameOf(o1, "."); final String o2Name = classNameOf(o2, "."); return o1Name.compareTo(o2Name); }) .collect(Collectors.toList()); if (elements.isEmpty()) { Element reportableElement = adapterFactories.iterator().next(); if (!autoValueElements.isEmpty()) { processingEnv.getMessager().printMessage(ERROR, "Failed to write TypeAdapterFactory: Cannot generate class for this " + "@GsonTypeAdapterFactory-annotated element because while @AutoValue-annotated " + "elements were found on the compilation classpath, none of them contain a " + "requisite public static TypeAdapter-returning signature method to opt in to " + "being included in @GsonTypeAdapterFactory-generated factories. See the " + "auto-value-gson README for more information on declaring these.", reportableElement); } else { processingEnv.getMessager().printMessage(ERROR, "Failed to write TypeAdapterFactory: Cannot generate class for this " + "@GsonTypeAdapterFactory-annotated element because no @AutoValue-annotated " + "elements were found on the compilation classpath.", reportableElement); } return false; } for (Element element : adapterFactories) { if (!element.getModifiers().contains(ABSTRACT)) { error(element, "Must be abstract!"); } TypeElement type = (TypeElement) element; // Safe to cast because this is only applicable on types anyway if (!implementsTypeAdapterFactory(type)) { error(element, "Must implement TypeAdapterFactory!"); } String adapterName = classNameOf(type, "_"); String qualifiedName = classNameOf(type, "."); PackageElement packageElement = packageElementOf(type); String packageName = packageElement.getQualifiedName().toString(); List applicableElements = elements.stream() .filter(e -> { Visibility typeVisibility = Visibility.ofElement(e); switch (typeVisibility) { case PRIVATE: return false; case DEFAULT: case PROTECTED: //noinspection UnstableApiUsage if (!getPackage(e).equals(packageElement)) { return false; } break; } // If we got here, the class is visible. Now check the typeAdapter method ExecutableElement adapterMethod = getTypeAdapterMethod(e); if (adapterMethod == null) { return false; } Visibility methodVisibility = Visibility.ofElement(adapterMethod); switch (methodVisibility) { case PRIVATE: return false; case DEFAULT: case PROTECTED: //noinspection UnstableApiUsage if (!getPackage(adapterMethod).equals(packageElement)) { return false; } break; } return true; }) .collect(toList()); TypeSpec typeAdapterFactory = createTypeAdapterFactory(type, applicableElements, packageName, adapterName, qualifiedName); JavaFile file = JavaFile.builder(packageName, typeAdapterFactory).build(); try { file.writeTo(processingEnv.getFiler()); } catch (IOException e) { processingEnv.getMessager().printMessage(ERROR, "Failed to write TypeAdapterFactory: " + e.getLocalizedMessage(), element); } } // return false so other processors can consume the @AutoValue annotation return false; } private static AnnotationSpec createGeneratedAnnotationSpec(TypeElement generatedAnnotationTypeElement) { return AnnotationSpec.builder(ClassName.get(generatedAnnotationTypeElement)) .addMember("value", "$S", AutoValueGsonAdapterFactoryProcessor.class.getName()) .addMember("comments", "$S", GENERATED_COMMENTS) .build(); } private TypeSpec createTypeAdapterFactory( TypeElement sourceElement, List elements, String packageName, String adapterName, String qualifiedName) { TypeSpec.Builder factory = TypeSpec.classBuilder( ClassName.get(packageName, "AutoValueGson_" + adapterName)); Optional generatedAnnotationSpec = GeneratedAnnotations.generatedAnnotation(processingEnv.getElementUtils(), processingEnv.getSourceVersion()) .map(AutoValueGsonAdapterFactoryProcessor::createGeneratedAnnotationSpec); generatedAnnotationSpec.ifPresent(factory::addAnnotation); factory.addOriginatingElement(sourceElement); factory.addModifiers(FINAL); factory.superclass(ClassName.get(packageName, qualifiedName)); ParameterSpec gson = ParameterSpec.builder(Gson.class, "gson").build(); TypeVariableName t = TypeVariableName.get("T"); ParameterSpec type = ParameterSpec .builder(ParameterizedTypeName.get(ClassName.get(TypeToken.class), t), "type") .build(); ParameterizedTypeName result = ParameterizedTypeName.get(ClassName.get(TypeAdapter.class), t); MethodSpec.Builder create = MethodSpec.methodBuilder("create") .addModifiers(PUBLIC) .addTypeVariable(t) .addAnnotation(Override.class) .addAnnotation(AnnotationSpec.builder(SuppressWarnings.class) .addMember("value", "\"unchecked\"") .build()) .addParameters(ImmutableSet.of(gson, type)) .returns(result) .addStatement("Class rawType = $N.getRawType()", type); List> properties = elements.stream() .peek(factory::addOriginatingElement) .map(e -> Pair.create(e, getTypeAdapterMethod(e))) .filter(entry -> entry.second != null) .collect(Collectors.toList()); for (int i = 0, elementsSize = properties.size(); i < elementsSize; i++) { Pair pair = properties.get(i); Element element = pair.first; TypeName elementType = rawType(element); if (i == 0) { create.beginControlFlow("if ($T.class.isAssignableFrom(rawType))", elementType); } else { create.nextControlFlow("else if ($T.class.isAssignableFrom(rawType))", elementType); } ExecutableElement typeAdapterMethod = pair.second; List params = typeAdapterMethod.getParameters(); if (params == null || params.size() == 0) { create.addStatement("return (TypeAdapter<$T>) $T." + typeAdapterMethod.getSimpleName() + "()", t, elementType); } else if (params.size() == 1) { create.addStatement("return (TypeAdapter<$T>) $T." + typeAdapterMethod.getSimpleName() + "($N)", t, elementType, gson); } else { create.addStatement("return (TypeAdapter<$T>) $T." + typeAdapterMethod.getSimpleName() + "($N, (($T) $N.getType()).getActualTypeArguments())", t, elementType, gson, ParameterizedType.class, type); } } create.nextControlFlow("else"); create.addStatement("return null"); create.endControlFlow(); factory.addMethod(create.build()); return factory.build(); } private TypeName rawType(Element element) { TypeName type = TypeName.get(element.asType()); if (type instanceof ParameterizedTypeName) { type = ((ParameterizedTypeName) type).rawType; } return type; } private ExecutableElement getTypeAdapterMethod(TypeElement element) { TypeName type = TypeName.get(element.asType()); ParameterizedTypeName typeAdapterType = ParameterizedTypeName .get(ClassName.get(TypeAdapter.class), type); for (ExecutableElement method : ElementFilter.methodsIn(element.getEnclosedElements())) { if (method.getModifiers().contains(STATIC) && !method.getModifiers().contains(PRIVATE)) { TypeName returnType = TypeName.get(method.getReturnType()); if (returnType.equals(typeAdapterType)) { return method; } else if (returnType instanceof ParameterizedTypeName) { ParameterizedTypeName paramReturnType = (ParameterizedTypeName) returnType; TypeName argument = paramReturnType.typeArguments.get(0); // If the original type uses generics, user's don't have to nest the generic type args if (type instanceof ParameterizedTypeName) { ParameterizedTypeName pTypeName = (ParameterizedTypeName) type; if (pTypeName.rawType.equals(argument)) { return method; } } } } } return null; } private void error(Element element, String message, Object... args) { if (args.length > 0) { message = String.format(message, args); } processingEnv.getMessager().printMessage(ERROR, message, element); } private boolean implementsTypeAdapterFactory(TypeElement type) { TypeMirror typeAdapterFactoryType = elementUtils.getTypeElement(TypeAdapterFactory.class.getCanonicalName()).asType(); TypeMirror typeMirror = type.asType(); if (!type.getInterfaces().isEmpty() || typeMirror.getKind() != TypeKind.NONE) { while (typeMirror.getKind() != TypeKind.NONE) { if (searchInterfacesAncestry(typeMirror, typeAdapterFactoryType)) { return true; } type = (TypeElement) typeUtils.asElement(typeMirror); typeMirror = type.getSuperclass(); } } return false; } private boolean searchInterfacesAncestry(TypeMirror rootIface, TypeMirror target) { TypeElement rootIfaceElement = (TypeElement) typeUtils.asElement(rootIface); // check if it implements valid interfaces for (TypeMirror iface : rootIfaceElement.getInterfaces()) { TypeElement ifaceElement = (TypeElement) typeUtils.asElement(rootIface); while (iface.getKind() != TypeKind.NONE) { if (typeUtils.isSameType(iface, target)) { return true; } // go up if (searchInterfacesAncestry(iface, target)) { return true; } // then move on iface = ifaceElement.getSuperclass(); } } return false; } /** * Returns the name of the given type, including any enclosing types but not the package, separated * by a delimiter. */ private static String classNameOf(TypeElement type, String delimiter) { StringBuilder name = new StringBuilder(type.getSimpleName().toString()); while (type.getEnclosingElement() instanceof TypeElement) { type = (TypeElement) type.getEnclosingElement(); name.insert(0, type.getSimpleName() + delimiter); } return name.toString(); } /** * Returns the package element that the given type is in. If the type is in the default * (unnamed) package then the name is the empty string. */ private static PackageElement packageElementOf(TypeElement type) { //noinspection UnstableApiUsage return getPackage(type); } private static class Pair { private final F first; private final S second; private Pair(F first, S second) { this.first = first; this.second = second; } static Pair create(F first, S second) { return new Pair<>(first, second); } } } auto-value-gson-1.3.1/auto-value-gson-factory/src/test/000077500000000000000000000000001375747333100230205ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/000077500000000000000000000000001375747333100237415ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/000077500000000000000000000000001375747333100245175ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/ryanharter/000077500000000000000000000000001375747333100266765ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/ryanharter/auto/000077500000000000000000000000001375747333100276465ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100307625ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100317305ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/ryanharter/auto/value/gson/factory/000077500000000000000000000000001375747333100333775ustar00rootroot00000000000000AutoValueGsonAdapterFactoryProcessorTest.java000066400000000000000000000643161375747333100443220ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-factory/src/test/java/com/ryanharter/auto/value/gson/factorypackage com.ryanharter.auto.value.gson.factory; import com.google.common.collect.ImmutableSet; import com.google.testing.compile.JavaFileObjects; import javax.tools.JavaFileObject; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; import static com.google.common.truth.Truth.assertAbout; import static com.google.testing.compile.JavaSourcesSubjectFactory.javaSources; @RunWith(JUnit4.class) public class AutoValueGsonAdapterFactoryProcessorTest { @Test public void generatesTypeAdapterFactory() { JavaFileObject fooSource = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); // BarSource's typeAdapter method accepts no parameter, which is valid for the processor factory JavaFileObject barSource = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "@AutoValue public abstract class Bar {\n" + " public static TypeAdapter typeAdapter() {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); JavaFileObject bazSource = JavaFileObjects.forSourceString("test.Baz", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue abstract class Baz {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " abstract String getName();\n" + "}"); JavaFileObject publicInOtherPackage = JavaFileObjects.forSourceString("test2.PublicInOtherPackage", "" + "package test2;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class PublicInOtherPackage {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); // This is generated into a different package and not visible to the factory JavaFileObject notVisibleClass = JavaFileObjects.forSourceString("test2.NotVisibleClass", "" + "package test2;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue abstract class NotVisibleClass {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); // This adapter method generated into a different package and not visible to the factory JavaFileObject notVisibleMethod = JavaFileObjects.forSourceString("test2.NotVisibleMethod", "" + "package test2;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class NotVisibleMethod {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); // This adapter method is private and thus not applicable JavaFileObject privateMethod = JavaFileObjects.forSourceString("test.PrivateMethod", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class PrivateMethod {\n" + " private static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); JavaFileObject factorySource = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public abstract class MyAdapterFactory implements TypeAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); JavaFileObject expected = JavaFileObjects.forSourceString("test.AutoValueGson_MyAdapterFactory", "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import java.lang.Override;\n" + "import java.lang.SuppressWarnings;\n" + "import javax.annotation.Generated;\n" + "import test2.PublicInOtherPackage;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.factory.AutoValueGsonAdapterFactoryProcessor\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValueGson_MyAdapterFactory extends MyAdapterFactory {\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public TypeAdapter create(Gson gson, TypeToken type) {\n" + " Class rawType = type.getRawType();\n" + " if (Bar.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Bar.typeAdapter();\n" + " } else if (Baz.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Baz.typeAdapter(gson);\n" + " } else if (Foo.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Foo.typeAdapter(gson);\n" + " } else if (PublicInOtherPackage.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) PublicInOtherPackage.typeAdapter(gson);\n" + " } else {\n" + " return null;\n" + " }\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(fooSource, barSource, bazSource, publicInOtherPackage, notVisibleClass, notVisibleMethod, privateMethod, factorySource)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void generatesTypeAdapterFactory_notAbstract_shouldFail() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject source2 = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public class MyAdapterFactory implements TypeAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, source2)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .failsToCompile() .withErrorContaining("Must be abstract!"); } @Test public void generatesTypeAdapterFactory_doesNotImplementTypeAdapterFactory_shouldFail() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject source2 = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Bar {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); JavaFileObject source3 = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public abstract class MyAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, source2, source3)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .failsToCompile() .withErrorContaining("Must implement TypeAdapterFactory!"); } @Test public void generatesTypeAdapterFactory_shouldSearchUpComplexAncestry() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject source2 = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Bar {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); JavaFileObject source3 = JavaFileObjects.forSourceString("test.IMyAdapterFactoryBase", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "public interface IMyAdapterFactoryBase extends TypeAdapterFactory {\n" + "}"); JavaFileObject source4 = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public abstract class MyAdapterFactory implements IMyAdapterFactoryBase {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); JavaFileObject expected = JavaFileObjects.forSourceString("test.AutoValueGson_MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import java.lang.Override;\n" + "import java.lang.SuppressWarnings;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.factory.AutoValueGsonAdapterFactoryProcessor\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValueGson_MyAdapterFactory extends MyAdapterFactory {\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public TypeAdapter create(Gson gson, TypeToken type) {\n" + " Class rawType = type.getRawType();\n" + " if (Bar.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Bar.typeAdapter(gson);\n" + " } else if (Foo.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Foo.typeAdapter(gson);\n" + " } else {\n" + " return null;\n" + " }\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, source2, source3, source4)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void generatesInnerClassTypeAdapterFactory() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject source2 = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Bar {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); JavaFileObject source3 = JavaFileObjects.forSourceString("test.InnerClassWrapper", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "public abstract class InnerClassWrapper {\n" + "@GsonTypeAdapterFactory\n" + "public abstract static class MyAdapterFactory implements TypeAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_InnerClassWrapper_MyAdapterFactory();\n" + " }\n" + "}\n" + "}"); JavaFileObject expected = JavaFileObjects.forSourceString("test.AutoValueGson_MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import java.lang.Override;\n" + "import java.lang.SuppressWarnings;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.factory.AutoValueGsonAdapterFactoryProcessor\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValueGson_InnerClassWrapper_MyAdapterFactory extends InnerClassWrapper.MyAdapterFactory {\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public TypeAdapter create(Gson gson, TypeToken type) {\n" + " Class rawType = type.getRawType();\n" + " if (Bar.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Bar.typeAdapter(gson);\n" + " } else if (Foo.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Foo.typeAdapter(gson);\n" + " } else {\n" + " return null;\n" + " }\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, source2, source3)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void generatesTypeAdapterFactory_shouldHaveOrder() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject source2 = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Bar {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + "}"); JavaFileObject factorySource = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public abstract class MyAdapterFactory implements TypeAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); JavaFileObject expected = JavaFileObjects.forSourceString("test.AutoValueGson_MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import java.lang.Override;\n" + "import java.lang.SuppressWarnings;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.factory.AutoValueGsonAdapterFactoryProcessor\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValueGson_MyAdapterFactory extends MyAdapterFactory {\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public TypeAdapter create(Gson gson, TypeToken type) {\n" + " Class rawType = type.getRawType();\n" + " if (Bar.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Bar.typeAdapter(gson);\n" + " } else if (Foo.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Foo.typeAdapter(gson);\n" + " } else {\n" + " return null;\n" + " }\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, source2, factorySource)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .compilesWithoutError() .and() .generatesSources(expected); assertAbout(javaSources()) .that(ImmutableSet.of(source2, source1, factorySource)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .compilesWithoutError() .and() .generatesSources(expected); } @Test public void noAutoValueModels_shouldError() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "public abstract class Foo {\n" + " public static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject factorySource = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public abstract class MyAdapterFactory implements TypeAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, factorySource)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .failsToCompile() .withErrorContaining("no @AutoValue-annotated elements were found on the " + "compilation classpath"); } @Test public void noAutoValueModelsWithAdapterMethods_shouldError() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.Gson;\n" + "@AutoValue public abstract class Foo {\n" + " public abstract String getName();\n" + " public abstract boolean isAwesome();\n" + "}"); JavaFileObject factorySource = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "public abstract class MyAdapterFactory implements TypeAdapterFactory {\n" + " public static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); assertAbout(javaSources()) .that(ImmutableSet.of(source1, factorySource)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .failsToCompile() .withErrorContaining("none of them contain a requisite public static " + "TypeAdapter-returning signature method to opt in to being included in " + "@GsonTypeAdapterFactory-generated factories"); } @Test public void packagePrivateEverything() { JavaFileObject source1 = JavaFileObjects.forSourceString("test.Foo", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue abstract class Foo {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " abstract String getName();\n" + " abstract boolean isAwesome();\n" + "}"); JavaFileObject source2 = JavaFileObjects.forSourceString("test.Bar", "" + "package test;\n" + "import com.google.auto.value.AutoValue;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.Gson;\n" + "@AutoValue abstract class Bar {\n" + " static TypeAdapter typeAdapter(Gson gson) {\n" + " return null;\n" + " }\n" + " abstract String getName();\n" + "}"); JavaFileObject source5 = JavaFileObjects.forSourceString("test.MyAdapterFactory", "" + "package test;\n" + "import com.google.gson.TypeAdapterFactory;\n" + "import com.google.gson.Gson;\n" + "import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory;\n" + "@GsonTypeAdapterFactory\n" + "abstract class MyAdapterFactory implements TypeAdapterFactory {\n" + " static TypeAdapterFactory create() {\n" + " return new AutoValueGson_MyAdapterFactory();\n" + " }\n" + "}"); JavaFileObject expected = JavaFileObjects.forSourceString("test.AutoValueGson_MyAdapterFactory", "\n" + "package test;\n" + "\n" + "import com.google.gson.Gson;\n" + "import com.google.gson.TypeAdapter;\n" + "import com.google.gson.reflect.TypeToken;\n" + "import java.lang.Override;\n" + "import java.lang.SuppressWarnings;\n" + "import javax.annotation.Generated;\n" + "\n" + "@Generated(\n" + " value = \"com.ryanharter.auto.value.gson.factory.AutoValueGsonAdapterFactoryProcessor\",\n" + " comments = \"https://github.com/rharter/auto-value-gson\"\n" + ")\n" + "final class AutoValueGson_MyAdapterFactory extends MyAdapterFactory {\n" + " @Override\n" + " @SuppressWarnings(\"unchecked\")\n" + " public TypeAdapter create(Gson gson, TypeToken type) {\n" + " Class rawType = type.getRawType();\n" + " if (Bar.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Bar.typeAdapter(gson);\n" + " } else if (Foo.class.isAssignableFrom(rawType)) {\n" + " return (TypeAdapter) Foo.typeAdapter(gson);\n" + " } else {\n" + " return null;\n" + " }\n" + " }\n" + "}"); assertAbout(javaSources()).that(ImmutableSet.of(source1, source2, source5)) .processedWith(new AutoValueGsonAdapterFactoryProcessor()) .compilesWithoutError() .and() .generatesSources(expected); } } auto-value-gson-1.3.1/auto-value-gson-runtime/000077500000000000000000000000001375747333100212665ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/build.gradle000066400000000000000000000005531375747333100235500ustar00rootroot00000000000000apply plugin: 'java-library' apply plugin: 'maven-publish' sourceCompatibility = JavaVersion.VERSION_1_8 targetCompatibility = JavaVersion.VERSION_1_8 dependencies { compileOnly deps.errorProneAnnotations compileOnly deps.jetbrainsAnnotations api deps.autotransient api deps.gson } apply from: rootProject.file('gradle/gradle-mvn-push.gradle') auto-value-gson-1.3.1/auto-value-gson-runtime/gradle.properties000066400000000000000000000001471375747333100246440ustar00rootroot00000000000000POM_ARTIFACT_ID=auto-value-gson-runtime POM_NAME=AutoValue: Gson Extension (Runtime) POM_PACKAGING=jar auto-value-gson-1.3.1/auto-value-gson-runtime/src/000077500000000000000000000000001375747333100220555ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/000077500000000000000000000000001375747333100230015ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/000077500000000000000000000000001375747333100237225ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/000077500000000000000000000000001375747333100245005ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/000077500000000000000000000000001375747333100266575ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/000077500000000000000000000000001375747333100276275ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100307435ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100317115ustar00rootroot00000000000000AutoValueGsonBuilder.java000066400000000000000000000007571375747333100365510ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gsonpackage com.ryanharter.auto.value.gson; import static java.lang.annotation.ElementType.METHOD; import static java.lang.annotation.RetentionPolicy.CLASS; import java.lang.annotation.Retention; import java.lang.annotation.Target; /** * If present, indicates that the annotated method should be used for retrieving an instance of the * AutoValue.Builder. Only necessary if there is more than one builder method. */ @Retention(CLASS) @Target(METHOD) public @interface AutoValueGsonBuilder { } GenerateTypeAdapter.java000066400000000000000000000112151375747333100363720ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gsonpackage com.ryanharter.auto.value.gson; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.TypeAdapterFactory; import com.google.gson.reflect.TypeToken; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.Target; import java.lang.reflect.Array; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; import java.util.Collections; import java.util.LinkedHashMap; import java.util.Map; import static java.lang.annotation.ElementType.TYPE; import static java.lang.annotation.RetentionPolicy.RUNTIME; /** * Annotate a given AutoValue-annotated type to indicate to auto-value-gson to generate its * TypeAdapter in a separate class rather than an inner class of an intermediate AutoValue-generated * class hierarchy. */ @Inherited @Retention(RUNTIME) @Target(TYPE) public @interface GenerateTypeAdapter { TypeAdapterFactory FACTORY = new TypeAdapterFactory() { private final Class typeArrayClass = Array.newInstance(Type.class, 0).getClass(); private final Map, Constructor> adapters = Collections.synchronizedMap(new LinkedHashMap<>()); @SuppressWarnings("unchecked") @Override public TypeAdapter create(Gson gson, TypeToken type) { Class rawType = type.getRawType(); if (!rawType.isAnnotationPresent(GenerateTypeAdapter.class)) { return null; } Class superClass = rawType.getSuperclass(); if (superClass.isAnnotationPresent(GenerateTypeAdapter.class)) { // We might be a generated AutoValue_ subtype. Walk up until we hit the first class that // isn't annotated with GenerateTypeAdapter. return (TypeAdapter) gson.getAdapter(superClass); } Constructor constructor = findConstructorForClass(rawType); if (constructor == null) { return null; } //noinspection TryWithIdenticalCatches Resolves to API 19+ only type. try { if (constructor.getParameterTypes().length == 1) { return constructor.newInstance(gson); } else { return constructor.newInstance(gson, ((ParameterizedType) type.getType()).getActualTypeArguments()); } } catch (IllegalAccessException e) { throw new RuntimeException("Unable to invoke " + constructor, e); } catch (InstantiationException e) { throw new RuntimeException("Unable to invoke " + constructor, e); } catch (InvocationTargetException e) { Throwable cause = e.getCause(); if (cause instanceof RuntimeException) { throw (RuntimeException) cause; } if (cause instanceof Error) { throw (Error) cause; } throw new RuntimeException( "Could not create generated TypeAdapter instance for type " + rawType, cause); } } private Constructor findConstructorForClass(Class cls) { Constructor adapterCtor = adapters.get(cls); if (adapterCtor != null) { return adapterCtor; } String clsName = cls.getName(); if (clsName.startsWith("android.") || clsName.startsWith("java.") || clsName.startsWith("kotlin.")) { return null; } try { String nameAdjusted = cls.getName().replace("$", "_"); Class bindingClass = cls.getClassLoader() .loadClass(nameAdjusted + "_GsonTypeAdapter"); try { // Try the gson constructor //noinspection unchecked adapterCtor = (Constructor) bindingClass.getDeclaredConstructor(Gson.class); adapterCtor.setAccessible(true); } catch (NoSuchMethodException e) { // Try the gson + type[] constructor //noinspection unchecked adapterCtor = (Constructor) bindingClass.getDeclaredConstructor(Gson.class, typeArrayClass); adapterCtor.setAccessible(true); } } catch (ClassNotFoundException e) { Constructor superClassAdapter = findConstructorForClass(cls.getSuperclass()); if (superClassAdapter != null) { superClassAdapter.setAccessible(true); } adapterCtor = superClassAdapter; } catch (NoSuchMethodException e) { throw new RuntimeException("Unable to find binding constructor for " + clsName, e); } adapters.put(cls, adapterCtor); return adapterCtor; } }; } GsonTypeAdapterFactory.java000066400000000000000000000014421375747333100370770ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gsonpackage com.ryanharter.auto.value.gson; import com.google.gson.TypeAdapterFactory; import java.lang.annotation.Retention; import java.lang.annotation.Target; import static java.lang.annotation.ElementType.TYPE; import static java.lang.annotation.RetentionPolicy.CLASS; /** * Annotation to indicate that a given class should generate a concrete implementation of a * {@link TypeAdapterFactory} that handles all the publicly denoted adapter implementations of this * project. *

*

 *   @GsonTypeAdapterFactory
 *   public abstract class Factory implements TypeAdapterFactory {
 *     public static Factory create() {
 *       return new AutoValueGson_Factory();
 *     }
 *   }
 * 
*/ @Retention(CLASS) @Target(TYPE) public @interface GsonTypeAdapterFactory { } auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gson/internal/000077500000000000000000000000001375747333100335255ustar00rootroot00000000000000Util.java000066400000000000000000000251741375747333100352370ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gson/internal/* * Copyright (C) 2014 Square, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.ryanharter.auto.value.gson.internal; import com.google.gson.FieldNamingPolicy; import com.google.gson.FieldNamingStrategy; import java.lang.reflect.GenericArrayType; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; import java.lang.reflect.WildcardType; import java.util.Arrays; import java.util.LinkedHashMap; import java.util.List; import java.util.Locale; import java.util.Map; import org.jetbrains.annotations.Nullable; public final class Util { static final Type[] EMPTY_TYPE_ARRAY = new Type[] {}; private Util() { } public static Map renameFields(Class targetClass, List names, FieldNamingStrategy fieldNamingStrategy) { Map renamedFields = new LinkedHashMap<>(); for (String fieldName : names) { if (fieldNamingStrategy instanceof FieldNamingPolicy) { switch ((FieldNamingPolicy) fieldNamingStrategy) { case UPPER_CAMEL_CASE: renamedFields.put(fieldName, upperCaseFirstLetter(fieldName)); break; case UPPER_CAMEL_CASE_WITH_SPACES: renamedFields.put(fieldName, upperCaseFirstLetter(separateCamelCase(fieldName, " "))); break; case LOWER_CASE_WITH_UNDERSCORES: renamedFields.put(fieldName, separateCamelCase(fieldName, "_").toLowerCase(Locale.ENGLISH)); break; case LOWER_CASE_WITH_DASHES: renamedFields.put(fieldName, separateCamelCase(fieldName, "-").toLowerCase(Locale.ENGLISH)); break; case LOWER_CASE_WITH_DOTS: renamedFields.put(fieldName, separateCamelCase(fieldName, ".").toLowerCase(Locale.ENGLISH)); break; default: renamedFields.put(fieldName, fieldName); } } else { try { renamedFields.put(fieldName, fieldNamingStrategy.translateName(targetClass.getDeclaredField(fieldName))); } catch (NoSuchFieldException E) { renamedFields.put(fieldName, fieldName); } } } return renamedFields; } private static String separateCamelCase(String name, String separator) { StringBuilder translation = new StringBuilder(); for (int i = 0, length = name.length(); i < length; i++) { char character = name.charAt(i); if (Character.isUpperCase(character) && translation.length() != 0) { translation.append(separator); } translation.append(character); } return translation.toString(); } private static String modifyString(char firstCharacter, String srcString, int indexOfSubstring) { return (indexOfSubstring < srcString.length()) ? firstCharacter + srcString.substring(indexOfSubstring) : String.valueOf(firstCharacter); } private static String upperCaseFirstLetter(String name) { StringBuilder fieldNameBuilder = new StringBuilder(); int index = 0; char firstCharacter = name.charAt(index); int length = name.length(); while (index < length - 1) { if (Character.isLetter(firstCharacter)) { break; } fieldNameBuilder.append(firstCharacter); firstCharacter = name.charAt(++index); } if (!Character.isUpperCase(firstCharacter)) { String modifiedTarget = modifyString(Character.toUpperCase(firstCharacter), name, ++index); return fieldNameBuilder.append(modifiedTarget).toString(); } else { return name; } } /** * Returns a type that is functionally equal but not necessarily equal according to {@link * Object#equals(Object) Object.equals()}. */ private static Type canonicalize(Type type) { if (type instanceof Class) { Class c = (Class) type; return c.isArray() ? new GenericArrayTypeImpl(canonicalize(c.getComponentType())) : c; } else if (type instanceof ParameterizedType) { if (type instanceof ParameterizedTypeImpl) return type; ParameterizedType p = (ParameterizedType) type; return new ParameterizedTypeImpl(p.getOwnerType(), p.getRawType(), p.getActualTypeArguments()); } else if (type instanceof GenericArrayType) { if (type instanceof GenericArrayTypeImpl) return type; GenericArrayType g = (GenericArrayType) type; return new GenericArrayTypeImpl(g.getGenericComponentType()); } else if (type instanceof WildcardType) { if (type instanceof WildcardTypeImpl) return type; WildcardType w = (WildcardType) type; return new WildcardTypeImpl(w.getUpperBounds(), w.getLowerBounds()); } else { return type; // This type is unsupported! } } private static int hashCodeOrZero(@Nullable Object o) { return o != null ? o.hashCode() : 0; } private static String typeToString(Type type) { return type instanceof Class ? ((Class) type).getName() : type.toString(); } private static void checkNotPrimitive(Type type) { if ((type instanceof Class) && ((Class) type).isPrimitive()) { throw new IllegalArgumentException("Unexpected primitive " + type + ". Use the boxed type."); } } public static final class ParameterizedTypeImpl implements ParameterizedType { private final @Nullable Type ownerType; private final Type rawType; final Type[] typeArguments; ParameterizedTypeImpl(@Nullable Type ownerType, Type rawType, Type... typeArguments) { // Require an owner type if the raw type needs it. if (rawType instanceof Class) { Class enclosingClass = ((Class) rawType).getEnclosingClass(); if (ownerType != null) { if (enclosingClass == null || WildcardUtil.getRawType(ownerType) != enclosingClass) { throw new IllegalArgumentException( "unexpected owner type for " + rawType + ": " + ownerType); } } else if (enclosingClass != null) { throw new IllegalArgumentException( "unexpected owner type for " + rawType + ": null"); } } this.ownerType = ownerType == null ? null : canonicalize(ownerType); this.rawType = canonicalize(rawType); this.typeArguments = typeArguments.clone(); for (int t = 0; t < this.typeArguments.length; t++) { if (this.typeArguments[t] == null) throw new NullPointerException(); checkNotPrimitive(this.typeArguments[t]); this.typeArguments[t] = canonicalize(this.typeArguments[t]); } } @Override public Type[] getActualTypeArguments() { return typeArguments.clone(); } @Override public Type getRawType() { return rawType; } @Override public @Nullable Type getOwnerType() { return ownerType; } @Override public boolean equals(Object other) { return other instanceof ParameterizedType && WildcardUtil.equals(this, (ParameterizedType) other); } @Override public int hashCode() { return Arrays.hashCode(typeArguments) ^ rawType.hashCode() ^ hashCodeOrZero(ownerType); } @Override public String toString() { StringBuilder result = new StringBuilder(30 * (typeArguments.length + 1)); result.append(typeToString(rawType)); if (typeArguments.length == 0) { return result.toString(); } result.append("<").append(typeToString(typeArguments[0])); for (int i = 1; i < typeArguments.length; i++) { result.append(", ").append(typeToString(typeArguments[i])); } return result.append(">").toString(); } } public static final class GenericArrayTypeImpl implements GenericArrayType { private final Type componentType; GenericArrayTypeImpl(Type componentType) { this.componentType = canonicalize(componentType); } @Override public Type getGenericComponentType() { return componentType; } @Override public boolean equals(Object o) { return o instanceof GenericArrayType && WildcardUtil.equals(this, (GenericArrayType) o); } @Override public int hashCode() { return componentType.hashCode(); } @Override public String toString() { return typeToString(componentType) + "[]"; } } /** * The WildcardType interface supports multiple upper bounds and multiple lower bounds. We only * support what the Java 6 language needs - at most one bound. If a lower bound is set, the upper * bound must be Object.class. */ public static final class WildcardTypeImpl implements WildcardType { private final Type upperBound; private final @Nullable Type lowerBound; WildcardTypeImpl(Type[] upperBounds, Type[] lowerBounds) { if (lowerBounds.length > 1) throw new IllegalArgumentException(); if (upperBounds.length != 1) throw new IllegalArgumentException(); if (lowerBounds.length == 1) { if (lowerBounds[0] == null) throw new NullPointerException(); checkNotPrimitive(lowerBounds[0]); if (upperBounds[0] != Object.class) throw new IllegalArgumentException(); this.lowerBound = canonicalize(lowerBounds[0]); this.upperBound = Object.class; } else { if (upperBounds[0] == null) throw new NullPointerException(); checkNotPrimitive(upperBounds[0]); this.lowerBound = null; this.upperBound = canonicalize(upperBounds[0]); } } @Override public Type[] getUpperBounds() { return new Type[] { upperBound }; } @Override public Type[] getLowerBounds() { return lowerBound != null ? new Type[] { lowerBound } : EMPTY_TYPE_ARRAY; } @Override public boolean equals(Object other) { return other instanceof WildcardType && WildcardUtil.equals(this, (WildcardType) other); } @Override public int hashCode() { // This equals Arrays.hashCode(getLowerBounds()) ^ Arrays.hashCode(getUpperBounds()). return (lowerBound != null ? 31 + lowerBound.hashCode() : 1) ^ (31 + upperBound.hashCode()); } @Override public String toString() { if (lowerBound != null) { return "? super " + typeToString(lowerBound); } else if (upperBound == Object.class) { return "?"; } else { return "? extends " + typeToString(upperBound); } } } } WildcardUtil.java000066400000000000000000000133551375747333100367070ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/java/com/ryanharter/auto/value/gson/internal/* * Copyright (C) 2008 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.ryanharter.auto.value.gson.internal; import com.google.errorprone.annotations.CheckReturnValue; import com.ryanharter.auto.value.gson.internal.Util.ParameterizedTypeImpl; import com.ryanharter.auto.value.gson.internal.Util.WildcardTypeImpl; import java.lang.reflect.Array; import java.lang.reflect.GenericArrayType; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; import java.lang.reflect.TypeVariable; import java.lang.reflect.WildcardType; import java.util.Arrays; import org.jetbrains.annotations.Nullable; import static com.ryanharter.auto.value.gson.internal.Util.EMPTY_TYPE_ARRAY; /** Factory methods for types. Adapted from Moshi. */ @CheckReturnValue public final class WildcardUtil { private WildcardUtil() { } /** * Returns a type that represents an unknown type that extends {@code bound}. For example, if * {@code bound} is {@code CharSequence.class}, this returns {@code ? extends CharSequence}. If * {@code bound} is {@code Object.class}, this returns {@code ?}, which is shorthand for {@code * ? extends Object}. */ public static WildcardType subtypeOf(Type bound) { return new WildcardTypeImpl(new Type[] { bound }, EMPTY_TYPE_ARRAY); } /** * Returns a type that represents an unknown supertype of {@code bound}. For example, if {@code * bound} is {@code String.class}, this returns {@code ? super String}. */ public static WildcardType supertypeOf(Type bound) { return new WildcardTypeImpl(new Type[] { Object.class }, new Type[] { bound }); } static Class getRawType(Type type) { if (type instanceof Class) { // type is a normal class. return (Class) type; } else if (type instanceof ParameterizedType) { ParameterizedType parameterizedType = (ParameterizedType) type; // I'm not exactly sure why getRawType() returns Type instead of Class. Neal isn't either but // suspects some pathological case related to nested classes exists. Type rawType = parameterizedType.getRawType(); return (Class) rawType; } else if (type instanceof GenericArrayType) { Type componentType = ((GenericArrayType) type).getGenericComponentType(); return Array.newInstance(getRawType(componentType), 0).getClass(); } else if (type instanceof TypeVariable) { // We could use the variable's bounds, but that won't work if there are multiple. having a raw // type that's more general than necessary is okay. return Object.class; } else if (type instanceof WildcardType) { return getRawType(((WildcardType) type).getUpperBounds()[0]); } else { String className = type == null ? "null" : type.getClass().getName(); throw new IllegalArgumentException("Expected a Class, ParameterizedType, or " + "GenericArrayType, but <" + type + "> is of type " + className); } } /** Returns true if {@code a} and {@code b} are equal. */ static boolean equals(@Nullable Type a, @Nullable Type b) { if (a == b) { return true; // Also handles (a == null && b == null). } else if (a instanceof Class) { if (b instanceof GenericArrayType) { return equals(((Class) a).getComponentType(), ((GenericArrayType) b).getGenericComponentType()); } return a.equals(b); // Class already specifies equals(). } else if (a instanceof ParameterizedType) { if (!(b instanceof ParameterizedType)) return false; ParameterizedType pa = (ParameterizedType) a; ParameterizedType pb = (ParameterizedType) b; Type[] aTypeArguments = pa instanceof ParameterizedTypeImpl ? ((ParameterizedTypeImpl) pa).typeArguments : pa.getActualTypeArguments(); Type[] bTypeArguments = pb instanceof ParameterizedTypeImpl ? ((ParameterizedTypeImpl) pb).typeArguments : pb.getActualTypeArguments(); return equals(pa.getOwnerType(), pb.getOwnerType()) && pa.getRawType().equals(pb.getRawType()) && Arrays.equals(aTypeArguments, bTypeArguments); } else if (a instanceof GenericArrayType) { if (b instanceof Class) { return equals(((Class) b).getComponentType(), ((GenericArrayType) a).getGenericComponentType()); } if (!(b instanceof GenericArrayType)) return false; GenericArrayType ga = (GenericArrayType) a; GenericArrayType gb = (GenericArrayType) b; return equals(ga.getGenericComponentType(), gb.getGenericComponentType()); } else if (a instanceof WildcardType) { if (!(b instanceof WildcardType)) return false; WildcardType wa = (WildcardType) a; WildcardType wb = (WildcardType) b; return Arrays.equals(wa.getUpperBounds(), wb.getUpperBounds()) && Arrays.equals(wa.getLowerBounds(), wb.getLowerBounds()); } else if (a instanceof TypeVariable) { if (!(b instanceof TypeVariable)) return false; TypeVariable va = (TypeVariable) a; TypeVariable vb = (TypeVariable) b; return va.getGenericDeclaration() == vb.getGenericDeclaration() && va.getName().equals(vb.getName()); } else { // This isn't a supported type. return false; } } } auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/resources/000077500000000000000000000000001375747333100250135ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/resources/META-INF/000077500000000000000000000000001375747333100261535ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/resources/META-INF/proguard/000077500000000000000000000000001375747333100277765ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson-runtime/src/main/resources/META-INF/proguard/autovaluegson.pro000066400000000000000000000002231375747333100334110ustar00rootroot00000000000000# Annotations are for embedding static analysis information. -dontwarn org.jetbrains.annotations.** -dontwarn com.google.errorprone.annotations.** auto-value-gson-1.3.1/auto-value-gson/000077500000000000000000000000001375747333100176055ustar00rootroot00000000000000auto-value-gson-1.3.1/auto-value-gson/build.gradle000066400000000000000000000005011375747333100220600ustar00rootroot00000000000000plugins { id 'java-library' id 'maven-publish' } sourceCompatibility = JavaVersion.VERSION_1_8 targetCompatibility = JavaVersion.VERSION_1_8 dependencies { api project(':auto-value-gson-extension') api project(':auto-value-gson-factory') } apply from: rootProject.file('gradle/gradle-mvn-push.gradle') auto-value-gson-1.3.1/auto-value-gson/gradle.properties000066400000000000000000000001471375747333100231630ustar00rootroot00000000000000POM_ARTIFACT_ID=auto-value-gson POM_NAME=AutoValue: Gson Extension (Legacy Artifact) POM_PACKAGING=jar auto-value-gson-1.3.1/build.gradle000066400000000000000000000030661375747333100170430ustar00rootroot00000000000000// Top-level build file where you can add configuration options common to all sub-projects/modules. buildscript { repositories { maven { url "https://plugins.gradle.org/m2/" } jcenter() } dependencies { classpath "net.ltgt.gradle:gradle-apt-plugin:0.19" } } allprojects { project -> project.repositories { maven { url 'http://oss.sonatype.org/content/repositories/snapshots' } mavenCentral() } project.plugins.withId("net.ltgt.apt") { try { // automatically apply net.ltgt.apt-idea whenever net.ltgt.apt is used project.apply plugin: "net.ltgt.apt-idea" // disable addAptDependencies (if you delegate build actions to Gradle) project.plugins.withType(JavaPlugin) { project.afterEvaluate { project.idea.module.apt.addAptDependencies = false } } } catch (ignored) { // ignore, in case an older version of net.ltgt.apt is being used // that doesn't come with net.ltgt.apt-idea. } } } ext.deps = [ autotransient: 'io.sweers.autotransient:autotransient:1.0.0', errorProneAnnotations: 'com.google.errorprone:error_prone_annotations:2.3.1', jetbrainsAnnotations: 'org.jetbrains:annotations-java5:16.0.2', gson: 'com.google.code.gson:gson:2.8.5' ] task wrapper(type: Wrapper) { gradleVersion = '4.10.2' distributionUrl = "https://services.gradle.org/distributions/gradle-$gradleVersion-all.zip" } auto-value-gson-1.3.1/example/000077500000000000000000000000001375747333100162125ustar00rootroot00000000000000auto-value-gson-1.3.1/example/build.gradle000066400000000000000000000012571375747333100204760ustar00rootroot00000000000000apply plugin: 'java' tasks.withType(JavaCompile) { options.compilerArgs += "-Aautovaluegson.useFieldNamePolicy" } dependencies { annotationProcessor 'com.google.auto.value:auto-value:1.7' annotationProcessor project(':auto-value-gson') compileOnly project(':auto-value-gson') compileOnly 'com.google.auto.value:auto-value-annotations:1.7' compile 'com.google.code.gson:gson:2.8.6' compile project(':auto-value-gson-runtime') testCompile 'junit:junit:4.12' testAnnotationProcessor 'com.google.auto.value:auto-value:1.7' testAnnotationProcessor project(':auto-value-gson') testCompileOnly 'com.google.auto.value:auto-value-annotations:1.7' } auto-value-gson-1.3.1/example/src/000077500000000000000000000000001375747333100170015ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/000077500000000000000000000000001375747333100177255ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/000077500000000000000000000000001375747333100206465ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/000077500000000000000000000000001375747333100214245ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/000077500000000000000000000000001375747333100236035ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/000077500000000000000000000000001375747333100245535ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100256675ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100266355ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/000077500000000000000000000000001375747333100302705ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/Address.java000066400000000000000000000021541375747333100325220ustar00rootroot00000000000000package com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.annotations.SerializedName; import com.ryanharter.auto.value.gson.AutoValueGsonBuilder; @AutoValue public abstract class Address { public static Address create(String streetName, String city) { return new AutoValue_Address(streetName, city); } public static TypeAdapter
typeAdapter(Gson gson) { return new AutoValue_Address.GsonTypeAdapter(gson); } @SerializedName("street-name") public abstract String streetName(); public abstract String city(); // This builder should never be invoked public static Builder badBuilder() { throw new RuntimeException("Wrong builder instance used!"); } @AutoValueGsonBuilder public static Builder builder() { return new AutoValue_Address.Builder(); } @AutoValue.Builder public interface Builder { Builder streetName(String streetName); Builder city(String city); Address build(); } } BirthdateAdapter.java000066400000000000000000000020321375747333100342600ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/examplepackage com.ryanharter.auto.value.gson.example; import com.google.gson.TypeAdapter; import com.google.gson.stream.JsonReader; import com.google.gson.stream.JsonToken; import com.google.gson.stream.JsonWriter; import java.io.IOException; import java.text.DateFormat; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; import java.util.Locale; public class BirthdateAdapter extends TypeAdapter { private final DateFormat df = new SimpleDateFormat("yyyy-MM-dd", Locale.getDefault()); @Override public void write(JsonWriter out, Date value) throws IOException { if (value == null) { out.nullValue(); } else { String date = df.format(value); out.value(date); } } @Override public Date read(JsonReader in) throws IOException { if (in.peek() == JsonToken.NULL) { in.nextNull(); return null; } String stringDate = in.nextString(); try { return df.parse(stringDate); } catch (ParseException e) { return new Date(); } } } GenericsExample.java000066400000000000000000000014641375747333100341340ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/examplepackage com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import java.lang.reflect.Type; @AutoValue public abstract class GenericsExample { public abstract A a(); public abstract B b(); public abstract C c(); @AutoValue.Builder public interface Builder { Builder a(A a); Builder b(B b); Builder c(C c); GenericsExample build(); } public static Builder builder() { return new AutoValue_GenericsExample.Builder(); } public static TypeAdapter> typeAdapter(Gson gson, Type[] types) { return new AutoValue_GenericsExample.GsonTypeAdapter<>(gson, types); } }auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/Person.java000066400000000000000000000025111375747333100324000ustar00rootroot00000000000000package com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import java.util.Date; @AutoValue public abstract class Person { public abstract String name(); public abstract int gender(); public abstract int age(); public abstract Date birthdate(); @Nullable public abstract Address address(); public static Builder builder() { return new AutoValue_Person.Builder() .name("Jane Doe") .gender(0); } public static TypeAdapter typeAdapter(Gson gson) { return new AutoValue_Person.GsonTypeAdapter(gson); } @AutoValue.Builder public static abstract class Builder { public abstract Builder name(String name); public abstract Builder gender(int gender); public abstract Builder age(int age); public abstract Builder birthdate(Date birthdate); public abstract Builder address(Address address); public abstract Person autoBuild(); public Person build() { Person person = autoBuild(); if (person.age() < 0) { throw new IllegalArgumentException("age cannot be negative"); } return person; } } public @interface Nullable {} } SampleAdapterFactory.java000066400000000000000000000005621375747333100351310ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/examplepackage com.ryanharter.auto.value.gson.example; import com.google.gson.TypeAdapterFactory; import com.ryanharter.auto.value.gson.GsonTypeAdapterFactory; @GsonTypeAdapterFactory public abstract class SampleAdapterFactory implements TypeAdapterFactory { public static SampleAdapterFactory create() { return new AutoValueGson_SampleAdapterFactory(); } } auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/TestNames.java000066400000000000000000000030171375747333100330370ustar00rootroot00000000000000package com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.annotations.SerializedName; @AutoValue public abstract class TestNames { public abstract int lowerCamel(); public abstract int UpperCamel(); public abstract int _lowerCamelLeadingUnderscore(); public abstract int _UpperCamelLeadingUnderscore(); public abstract int lower_words(); public abstract int UPPER_WORDS(); @SerializedName("annotatedName") public abstract int annotated(); public abstract int lowerId(); public static Builder builder() { return new AutoValue_TestNames.Builder(); } @AutoValue.Builder public static abstract class Builder { public abstract Builder lowerCamel(int lowerCamel); public abstract Builder UpperCamel(int UpperCamel); public abstract Builder _lowerCamelLeadingUnderscore(int _lowerCamelLeadingUnderscore); public abstract Builder _UpperCamelLeadingUnderscore(int _UpperCamelLeadingUnderscore); public abstract Builder lower_words(int lower_words); public abstract Builder UPPER_WORDS(int UPPER_WORDS); @SerializedName("annotatedName") public abstract Builder annotated(int annotated); public abstract Builder lowerId(int lowerId); public abstract TestNames build(); } public static TypeAdapter typeAdapter(Gson gson) { return new AutoValue_TestNames.GsonTypeAdapter(gson); } } auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/User.java000066400000000000000000000007571375747333100320620ustar00rootroot00000000000000package com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; @AutoValue public abstract class User { abstract String firstname(); abstract String lastname(); public static TypeAdapter typeAdapter(Gson gson) { return new AutoValue_User.GsonTypeAdapter(gson); } public static User with(String firstname, String lastname) { return new AutoValue_User(firstname, lastname); } } auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/WebResponse.java000066400000000000000000000011721375747333100333700ustar00rootroot00000000000000package com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.reflect.TypeToken; import java.lang.reflect.Type; import java.util.List; import java.util.Map; @AutoValue public abstract class WebResponse { public abstract int status(); public abstract T data(); public abstract List dataList(); public abstract Map> dataMap(); public static TypeAdapter> typeAdapter(Gson gson, Type[] types) { return new AutoValue_WebResponse.GsonTypeAdapter<>(gson, types); } } WebResponseNoStatic.java000066400000000000000000000010221375747333100347500ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/examplepackage com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.TypeAdapter; import com.google.gson.reflect.TypeToken; import com.ryanharter.auto.value.gson.GenerateTypeAdapter; import java.util.List; import java.util.Map; @AutoValue @GenerateTypeAdapter public abstract class WebResponseNoStatic { public abstract int status(); public abstract T data(); public abstract List dataList(); public abstract Map> dataMap(); } auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/wildgenerics/000077500000000000000000000000001375747333100327475ustar00rootroot00000000000000AddressGenericBase.java000066400000000000000000000013001375747333100372020ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/wildgenericspackage com.ryanharter.auto.value.gson.example.wildgenerics; import com.google.auto.value.AutoValue; import com.ryanharter.auto.value.gson.GenerateTypeAdapter; import com.ryanharter.auto.value.gson.example.Address; @GenerateTypeAdapter @AutoValue public abstract class AddressGenericBase extends GenericBase
{ public abstract String topLevelNonGeneric(); public static Builder builder() { return new AutoValue_AddressGenericBase.Builder(); } @AutoValue.Builder public abstract static class Builder implements GenericBase.Builder { public abstract Builder topLevelNonGeneric(String topLevel); public abstract AddressGenericBase build(); } } DoubleGeneric.java000066400000000000000000000016051375747333100362440ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/wildgenericspackage com.ryanharter.auto.value.gson.example.wildgenerics; import com.google.auto.value.AutoValue; import com.ryanharter.auto.value.gson.GenerateTypeAdapter; import java.util.List; @GenerateTypeAdapter @AutoValue public abstract class DoubleGeneric extends GenericBase { public abstract String topLevelNonGeneric(); public abstract T topLevelGeneric(); public abstract List topLevelGenericCollection(); public static Builder builder() { return new AutoValue_DoubleGeneric.Builder(); } @AutoValue.Builder public abstract static class Builder implements GenericBase.Builder> { public abstract Builder topLevelNonGeneric(String topLevel); public abstract Builder topLevelGeneric(T topLevel); public abstract Builder topLevelGenericCollection(List topLevel); public abstract DoubleGeneric build(); } } GenericBase.java000066400000000000000000000006251375747333100357050ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/main/java/com/ryanharter/auto/value/gson/example/wildgenericspackage com.ryanharter.auto.value.gson.example.wildgenerics; import java.util.List; public abstract class GenericBase { public abstract T generic(); public abstract String notGeneric(); public abstract List collection(); public interface Builder> { B generic(T generic); B notGeneric(String notGeneric); B collection(List collection); } } auto-value-gson-1.3.1/example/src/test/000077500000000000000000000000001375747333100177605ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/000077500000000000000000000000001375747333100207015ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/000077500000000000000000000000001375747333100214575ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/000077500000000000000000000000001375747333100236365ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/000077500000000000000000000000001375747333100246065ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/000077500000000000000000000000001375747333100257225ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/000077500000000000000000000000001375747333100266705ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/example/000077500000000000000000000000001375747333100303235ustar00rootroot00000000000000FieldNamingTest.java000066400000000000000000000124271375747333100341320ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/example/* * Copyright (C) 2011 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.ryanharter.auto.value.gson.example; import com.google.gson.FieldNamingPolicy; import com.google.gson.FieldNamingStrategy; import com.google.gson.Gson; import com.google.gson.GsonBuilder; import junit.framework.TestCase; import java.lang.reflect.Field; import static com.google.gson.FieldNamingPolicy.IDENTITY; import static com.google.gson.FieldNamingPolicy.LOWER_CASE_WITH_DASHES; import static com.google.gson.FieldNamingPolicy.LOWER_CASE_WITH_DOTS; import static com.google.gson.FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES; import static com.google.gson.FieldNamingPolicy.UPPER_CAMEL_CASE; import static com.google.gson.FieldNamingPolicy.UPPER_CAMEL_CASE_WITH_SPACES; public final class FieldNamingTest extends TestCase { public void testIdentity() { Gson gson = getGsonWithNamingPolicy(IDENTITY); assertEquals("{'lowerCamel':1,'UpperCamel':2,'_lowerCamelLeadingUnderscore':3," + "'_UpperCamelLeadingUnderscore':4,'lower_words':5,'UPPER_WORDS':6," + "'annotatedName':7,'lowerId':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } public void testUpperCamelCase() { Gson gson = getGsonWithNamingPolicy(UPPER_CAMEL_CASE); assertEquals("{'LowerCamel':1,'UpperCamel':2,'_LowerCamelLeadingUnderscore':3," + "'_UpperCamelLeadingUnderscore':4,'Lower_words':5,'UPPER_WORDS':6," + "'annotatedName':7,'LowerId':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } public void testUpperCamelCaseWithSpaces() { Gson gson = getGsonWithNamingPolicy(UPPER_CAMEL_CASE_WITH_SPACES); assertEquals("{'Lower Camel':1,'Upper Camel':2,'_Lower Camel Leading Underscore':3," + "'_ Upper Camel Leading Underscore':4,'Lower_words':5,'U P P E R_ W O R D S':6," + "'annotatedName':7,'Lower Id':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } public void testLowerCaseWithUnderscores() { Gson gson = getGsonWithNamingPolicy(LOWER_CASE_WITH_UNDERSCORES); assertEquals("{'lower_camel':1,'upper_camel':2,'_lower_camel_leading_underscore':3," + "'__upper_camel_leading_underscore':4,'lower_words':5,'u_p_p_e_r__w_o_r_d_s':6," + "'annotatedName':7,'lower_id':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } public void testLowerCaseWithDashes() { Gson gson = getGsonWithNamingPolicy(LOWER_CASE_WITH_DASHES); assertEquals("{'lower-camel':1,'upper-camel':2,'_lower-camel-leading-underscore':3," + "'_-upper-camel-leading-underscore':4,'lower_words':5,'u-p-p-e-r_-w-o-r-d-s':6," + "'annotatedName':7,'lower-id':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } public void testLowerCaseWithDots() { Gson gson = getGsonWithNamingPolicy(LOWER_CASE_WITH_DOTS); assertEquals("{'lower.camel':1,'upper.camel':2,'_lower.camel.leading.underscore':3," + "'_.upper.camel.leading.underscore':4,'lower_words':5,'u.p.p.e.r_.w.o.r.d.s':6," + "'annotatedName':7,'lower.id':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } public void testCustomFieldNamingStrategy() { Gson gson = new GsonBuilder() .setFieldNamingStrategy(new xPrefixStrategy()) .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); assertEquals("{'xlowerCamel':1,'xUpperCamel':2,'x_lowerCamelLeadingUnderscore':3," + "'x_UpperCamelLeadingUnderscore':4,'xlower_words':5,'xUPPER_WORDS':6," + "'annotatedName':7,'xlowerId':8}", gson.toJson(getTestNames()).replace('\"', '\'')); } private Gson getGsonWithNamingPolicy(FieldNamingPolicy fieldNamingPolicy){ return new GsonBuilder() .setFieldNamingPolicy(fieldNamingPolicy) .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); } private TestNames getTestNames() { return TestNames.builder() .lowerCamel(1) .UpperCamel(2) ._lowerCamelLeadingUnderscore(3) ._UpperCamelLeadingUnderscore(4) .lower_words(5) .UPPER_WORDS(6) .annotated(7) .lowerId(8) .build(); } private class xPrefixStrategy implements FieldNamingStrategy { @Override public String translateName(Field f) { return "x" + f.getName(); } } } GenerateTypeAdapterTest.java000066400000000000000000000035741375747333100356550ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/examplepackage com.ryanharter.auto.value.gson.example; import com.google.auto.value.AutoValue; import com.google.gson.Gson; import com.google.gson.GsonBuilder; import com.google.gson.TypeAdapter; import com.ryanharter.auto.value.gson.GenerateTypeAdapter; import java.io.IOException; import org.junit.Test; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertSame; public final class GenerateTypeAdapterTest { @Test public void smokeTest() throws IOException { //language=JSON String json = "{\"seasoning\":\"spicy\"}"; Gson gson = new GsonBuilder() .registerTypeAdapterFactory(GenerateTypeAdapter.FACTORY) .create(); Taco expectedTaco = Taco.create("spicy"); // Basic read Taco taco = gson.fromJson(json, Taco.class); assertEquals(expectedTaco, taco); // Basic write String encoded = gson.toJson(taco); assertEquals(json, encoded); // The same as above, but with the adapter looked up first TypeAdapter adapter = gson.getAdapter(Taco.class); // Basic read Taco taco2 = adapter.fromJson(json); assertEquals(expectedTaco, taco2); // Basic write String encoded2 = adapter.toJson(taco); assertEquals(json, encoded2); // Ensure the generated AutoValue_Taco class still gets translated back into a taco adapter // Assert they're the same to ensure we've properly delegated up the gson chain and avoid // duplicating adapter instances. TypeAdapter generatedClassAdapter = gson.getAdapter(AutoValue_GenerateTypeAdapterTest_Taco.class); assertSame(adapter, generatedClassAdapter); } @GenerateTypeAdapter @AutoValue public static abstract class Taco { public abstract String seasoning(); public static Taco create(String seasoning) { return new AutoValue_GenerateTypeAdapterTest_Taco(seasoning); } } } auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/example/PersonTest.java000066400000000000000000000055321375747333100333010ustar00rootroot00000000000000package com.ryanharter.auto.value.gson.example; import com.google.gson.Gson; import com.google.gson.GsonBuilder; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.Date; import java.util.Locale; import org.junit.Assert; import org.junit.Rule; import org.junit.Test; import org.junit.rules.ExpectedException; public class PersonTest { @Rule public ExpectedException expectedException = ExpectedException.none(); @Test public void testGson() throws Exception { final DateFormat df = new SimpleDateFormat("yyyy-MM-dd", Locale.getDefault()); String birthdate = "2007-11-11"; Date date = df.parse(birthdate); Gson gson = new GsonBuilder() .registerTypeAdapter(Date.class, new BirthdateAdapter()) .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); Person person = Person.builder() .name("Piasy") .gender(1) .age(23) .birthdate(date) .address(Address.create("street", "city")) .build(); //language=json String json = "{\"name\":\"Piasy\",\"gender\":1,\"age\":23,\"birthdate\":\"" + birthdate + "\",\"address\":{\"street-name\":\"street\",\"city\":\"city\"}}"; String toJson = gson.toJson(person, Person.class); Assert.assertEquals(json, toJson); Person fromJson = gson.fromJson(json, Person.class); Assert.assertEquals("Piasy", fromJson.name()); Assert.assertEquals(23, fromJson.age()); Assert.assertEquals(1, fromJson.gender()); Assert.assertEquals(date, fromJson.birthdate()); } @Test public void testGsonWithValidation() { expectedException.expect(IllegalArgumentException.class); expectedException.expectMessage("age cannot be negative"); Gson gson = new GsonBuilder() .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); //language=json String json = "{\"name\":\"Piasy\",\"gender\":1,\"age\":-1,\"birthdate\":\"2007-11-11\",\"address\":{\"street-name\":\"street\",\"city\":\"city\"}}"; gson.fromJson(json, Person.class); } @Test public void testGsonWithDefaults() { Gson gson = new GsonBuilder() .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); // "name" and "gender" are unspecified. Should default to "Jane Doe" and 23 //language=json String json = "{\"age\":23,\"birthdate\":\"2007-11-11\",\"address\":{\"street-name\":\"street\",\"city\":\"city\"}}"; Person fromJson = gson.fromJson(json, Person.class); Assert.assertEquals("Jane Doe", fromJson.name()); Assert.assertEquals(23, fromJson.age()); Assert.assertEquals(0, fromJson.gender()); } } WebResponseTest.java000066400000000000000000000065661375747333100342200ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/examplepackage com.ryanharter.auto.value.gson.example; import com.google.gson.Gson; import com.google.gson.GsonBuilder; import com.google.gson.reflect.TypeToken; import com.ryanharter.auto.value.gson.GenerateTypeAdapter; import java.lang.reflect.Type; import org.junit.Test; import static org.junit.Assert.assertEquals; public class WebResponseTest { @Test public void handlesBasicTypes() { String json = "{\"status\":200,\"data\":\"string\"," + "\"dataList\":[\"string\"]," + "\"dataMap\":{\"key\":[\"string\"]}}"; Gson gson = new GsonBuilder() .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); Type responseType = new TypeToken>(){}.getType(); WebResponse response = gson.fromJson(json, responseType); assertEquals("string", response.data()); assertEquals("string", response.dataList().get(0)); assertEquals("string", response.dataMap().get("key").get(0)); assertEquals(200, response.status()); } @Test public void handlesComplexTypes() { String json = "{\"status\":200,\"data\":{\"firstname\":\"Ryan\",\"lastname\":\"Harter\"}, " + "\"dataList\":[{\"firstname\":\"Ryan\",\"lastname\":\"Harter\"}]," + "\"dataMap\":{\"key\":[{\"firstname\":\"Ryan\",\"lastname\":\"Harter\"}]}}"; Gson gson = new GsonBuilder() .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); Type responseType = new TypeToken>(){}.getType(); WebResponse response = gson.fromJson(json, responseType); User expected = User.with("Ryan", "Harter"); assertEquals(expected, response.data()); assertEquals(expected, response.dataList().get(0)); assertEquals(expected, response.dataMap().get("key").get(0)); } @Test public void handlesBasicTypesExternalAdapter() { String json = "{\"status\":200,\"data\":\"string\"," + "\"dataList\":[\"string\"]," + "\"dataMap\":{\"key\":[\"string\"]}}"; Gson gson = new GsonBuilder() .registerTypeAdapterFactory(GenerateTypeAdapter.FACTORY) .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); Type responseType = new TypeToken>(){}.getType(); WebResponseNoStatic response = gson.fromJson(json, responseType); assertEquals("string", response.data()); assertEquals("string", response.dataList().get(0)); assertEquals("string", response.dataMap().get("key").get(0)); assertEquals(200, response.status()); } @Test public void handlesComplexTypesExternalAdapter() { String json = "{\"status\":200,\"data\":{\"firstname\":\"Ryan\",\"lastname\":\"Harter\"}, " + "\"dataList\":[{\"firstname\":\"Ryan\",\"lastname\":\"Harter\"}]," + "\"dataMap\":{\"key\":[{\"firstname\":\"Ryan\",\"lastname\":\"Harter\"}]}}"; Gson gson = new GsonBuilder() .registerTypeAdapterFactory(GenerateTypeAdapter.FACTORY) .registerTypeAdapterFactory(SampleAdapterFactory.create()) .create(); Type responseType = new TypeToken>(){}.getType(); WebResponseNoStatic response = gson.fromJson(json, responseType); User expected = User.with("Ryan", "Harter"); assertEquals(expected, response.data()); assertEquals(expected, response.dataList().get(0)); assertEquals(expected, response.dataMap().get("key").get(0)); } } auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/example/wildgenerics/000077500000000000000000000000001375747333100330025ustar00rootroot00000000000000WildGenericsTest.java000066400000000000000000000065221375747333100370120ustar00rootroot00000000000000auto-value-gson-1.3.1/example/src/test/java/com/ryanharter/auto/value/gson/example/wildgenericspackage com.ryanharter.auto.value.gson.example.wildgenerics; import com.google.gson.Gson; import com.google.gson.GsonBuilder; import com.google.gson.TypeAdapter; import com.google.gson.reflect.TypeToken; import com.ryanharter.auto.value.gson.GenerateTypeAdapter; import com.ryanharter.auto.value.gson.example.Address; import com.ryanharter.auto.value.gson.example.SampleAdapterFactory; import java.io.IOException; import org.junit.Test; import static com.google.gson.reflect.TypeToken.getParameterized; import static java.util.Collections.singletonList; import static org.junit.Assert.assertEquals; public final class WildGenericsTest { @Test public void addressGenericBase() throws IOException { //language=JSON String json = "{\"generic\":{\"street-name\":\"genericStreet\",\"city\":\"genericCity\"}," + "\"notGeneric\":\"notGeneric\",\"collection\":[{\"street-name\":\"fooStreet\"," + "\"city\":\"barCity\"}],\"topLevelNonGeneric\":\"topLevelNonGeneric\"}"; Gson gson = new GsonBuilder().registerTypeAdapterFactory(SampleAdapterFactory.create()) .registerTypeAdapterFactory(GenerateTypeAdapter.FACTORY) .create(); TypeAdapter adapter = gson.getAdapter(AddressGenericBase.class); AddressGenericBase instance = adapter.fromJson(json); AddressGenericBase testInstance = AddressGenericBase.builder() .topLevelNonGeneric("topLevelNonGeneric") .collection(singletonList(Address.create("fooStreet", "barCity"))) .generic(Address.create("genericStreet", "genericCity")) .notGeneric("notGeneric") .build(); assertEquals(instance, testInstance); assertEquals(json, adapter.toJson(testInstance)); } @Test public void doubleGeneric() throws IOException { //language=JSON String json = "{\"generic\":{\"street-name\":\"genericStreet\",\"city\":\"genericCity\"}," + "\"notGeneric\":\"notGeneric\",\"collection\":[{\"street-name\":\"fooStreet\"," + "\"city\":\"barCity\"}],\"topLevelNonGeneric\":\"topLevelNonGeneric\"," + "\"topLevelGeneric\":{\"street-name\":\"topStreet\",\"city\":\"topCity\"}," + "\"topLevelGenericCollection\":[{\"street-name\":\"topCollectionStreet\"," + "\"city\":\"topCollectionCity\"}]}"; Gson gson = new GsonBuilder().registerTypeAdapterFactory(SampleAdapterFactory.create()) .registerTypeAdapterFactory(GenerateTypeAdapter.FACTORY) .create(); //noinspection unchecked TypeToken> token = (TypeToken>) getParameterized(DoubleGeneric.class, Address.class); TypeAdapter> adapter = gson.getAdapter(token); DoubleGeneric
instance = adapter.fromJson(json); DoubleGeneric
testInstance = DoubleGeneric.
builder().topLevelNonGeneric("topLevelNonGeneric") .topLevelGeneric(Address.create("topStreet", "topCity")) .topLevelGenericCollection(singletonList(Address.create("topCollectionStreet", "topCollectionCity"))) .collection(singletonList(Address.create("fooStreet", "barCity"))) .generic(Address.create("genericStreet", "genericCity")) .notGeneric("notGeneric") .build(); assertEquals(instance, testInstance); assertEquals(json, adapter.toJson(testInstance)); } } auto-value-gson-1.3.1/gradle.properties000066400000000000000000000011531375747333100201330ustar00rootroot00000000000000GROUP=com.ryanharter.auto.value VERSION_NAME=1.3.1 POM_DESCRIPTION=AutoValue extension that creates a Gson TypeAdapter for JSON serialization of AutoValue classes. POM_URL=https://github.com/rharter/auto-value-gson/ POM_SCM_URL=https://github.com/rharter/auto-value-gson/ POM_SCM_CONNECTION=scm:git:git://github.com/rharter/auto-value-gson POM_SCM_DEV_CONNECTION=scm:git:ssh://git@github.com/rharter/auto-value-gson POM_LICENCE_NAME=The Apache Software License, Version 2.0 POM_LICENCE_URL=http://www.apache.org/licenses/LICENSE-2.0.txt POM_LICENCE_DIST=repo POM_DEVELOPER_ID=rharter POM_DEVELOPER_NAME=Ryan Harter auto-value-gson-1.3.1/gradle/000077500000000000000000000000001375747333100160155ustar00rootroot00000000000000auto-value-gson-1.3.1/gradle/gradle-mvn-push.gradle000066400000000000000000000155061375747333100222150ustar00rootroot00000000000000/* * Copyright 2013 Chris Banes * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ apply plugin: 'maven' apply plugin: 'signing' version = VERSION_NAME group = GROUP def isReleaseBuild() { return VERSION_NAME.contains("SNAPSHOT") == false } def getReleaseRepositoryUrl() { return hasProperty('RELEASE_REPOSITORY_URL') ? RELEASE_REPOSITORY_URL : "https://oss.sonatype.org/service/local/staging/deploy/maven2/" } def getSnapshotRepositoryUrl() { return hasProperty('SNAPSHOT_REPOSITORY_URL') ? SNAPSHOT_REPOSITORY_URL : "https://oss.sonatype.org/content/repositories/snapshots/" } def getRepositoryUsername() { return hasProperty('SONATYPE_NEXUS_USERNAME') ? SONATYPE_NEXUS_USERNAME : "" } def getRepositoryPassword() { return hasProperty('SONATYPE_NEXUS_PASSWORD') ? SONATYPE_NEXUS_PASSWORD : "" } afterEvaluate { project -> uploadArchives { repositories { mavenDeployer { beforeDeployment { MavenDeployment deployment -> signing.signPom(deployment) } pom.groupId = GROUP pom.artifactId = POM_ARTIFACT_ID pom.version = VERSION_NAME repository(url: getReleaseRepositoryUrl()) { authentication(userName: getRepositoryUsername(), password: getRepositoryPassword()) } snapshotRepository(url: getSnapshotRepositoryUrl()) { authentication(userName: getRepositoryUsername(), password: getRepositoryPassword()) } pom.project { name POM_NAME packaging POM_PACKAGING description POM_DESCRIPTION url POM_URL scm { url POM_SCM_URL connection POM_SCM_CONNECTION developerConnection POM_SCM_DEV_CONNECTION } licenses { license { name POM_LICENCE_NAME url POM_LICENCE_URL distribution POM_LICENCE_DIST } } developers { developer { id POM_DEVELOPER_ID name POM_DEVELOPER_NAME } } } } } } signing { required { isReleaseBuild() && gradle.taskGraph.hasTask("uploadArchives") } sign configurations.archives } if (project.getPlugins().hasPlugin('com.android.application') || project.getPlugins().hasPlugin('com.android.library')) { task install(type: Upload, dependsOn: assemble) { repositories.mavenInstaller { configuration = configurations.archives pom.groupId = GROUP pom.artifactId = POM_ARTIFACT_ID pom.version = VERSION_NAME pom.project { name POM_NAME packaging POM_PACKAGING description POM_DESCRIPTION url POM_URL scm { url POM_SCM_URL connection POM_SCM_CONNECTION developerConnection POM_SCM_DEV_CONNECTION } licenses { license { name POM_LICENCE_NAME url POM_LICENCE_URL distribution POM_LICENCE_DIST } } developers { developer { id POM_DEVELOPER_ID name POM_DEVELOPER_NAME } } } } } task androidJavadocs(type: Javadoc) { source = android.sourceSets.main.java.source classpath += project.files(android.getBootClasspath().join(File.pathSeparator)) } task androidJavadocsJar(type: Jar, dependsOn: androidJavadocs) { classifier = 'javadoc' from androidJavadocs.destinationDir } task androidSourcesJar(type: Jar) { classifier = 'sources' from android.sourceSets.main.java.source } } else { install { repositories.mavenInstaller { pom.groupId = GROUP pom.artifactId = POM_ARTIFACT_ID pom.version = VERSION_NAME pom.project { name POM_NAME packaging POM_PACKAGING description POM_DESCRIPTION url POM_URL scm { url POM_SCM_URL connection POM_SCM_CONNECTION developerConnection POM_SCM_DEV_CONNECTION } licenses { license { name POM_LICENCE_NAME url POM_LICENCE_URL distribution POM_LICENCE_DIST } } developers { developer { id POM_DEVELOPER_ID name POM_DEVELOPER_NAME } } } } } task sourcesJar(type: Jar, dependsOn:classes) { classifier = 'sources' from sourceSets.main.allSource } task javadocJar(type: Jar, dependsOn:javadoc) { classifier = 'javadoc' from javadoc.destinationDir } } if (JavaVersion.current().isJava8Compatible()) { allprojects { tasks.withType(Javadoc) { options.addStringOption('Xdoclint:none', '-quiet') } } } artifacts { if (project.getPlugins().hasPlugin('com.android.application') || project.getPlugins().hasPlugin('com.android.library')) { archives androidSourcesJar archives androidJavadocsJar } else { archives sourcesJar archives javadocJar } } }auto-value-gson-1.3.1/gradle/wrapper/000077500000000000000000000000001375747333100174755ustar00rootroot00000000000000auto-value-gson-1.3.1/gradle/wrapper/gradle-wrapper.jar000066400000000000000000001527131375747333100231200ustar00rootroot00000000000000PKA META-INF/PKAE?cDAVMETA-INF/MANIFEST.MFMLK-. K-*ϳR03-IM+I, dZ)%bµrrPKAorg/PKA org/gradle/PKAorg/gradle/wrapper/PKAzZ -org/gradle/wrapper/BootstrapMainStarter.classVYwV˖#DgH !bSJiIh @c5žv2$ڗSz?B;Wopsٿ;#g^O2Bx[aKXpA .X%˒ e\U &Ẅ)XpCƻHXp+|y/wPe#+!ǼC(H8=' f Ni[.יuM]׉JYU_U-kDJb)*$ Y⁥nn2+q4M%U3Ҷj̚/񣀣ѵ]UHۖff]f&4:V< ./EQY@˽Z}4le'uTJj1jN\_IձrvHz|;$󗘽a X,py$9ؘM/yfLq[YikQQ)Ke#,p)OżTGd`3¢r,[Yh.-(8Q3 ^:J=G]ܓ+(`* ,q_l0zFqjq8~XlC`5eOGX@.]O|Ϩ^Yb >Ǩr-? BJC|Mݸ(+o|z^)Vd;04z.bvD`".g3rJD3V+?~0?{g7y}^@O4V%\IZ$S$9T.٬qf ]}UA!  + ԏ{e n]*d0}Nε0mJ-kмYqZNeex45yz7n#4b$wѦ9z'e42l$ X)L#jSCŁv)I@EBM@ mg0}y[vvDc:t`րC܏c*2&H+?'9TOˈ@+WH?TA XR*;C]$ݝ']AL &w#,G\7 xϣ;NZr?a\VmˈãFX&1 cqBJCl"[F$k'UBmk$u : Hd VcO1 bᥩgsSxgL 'x* #_/2a/O0REϝ [&H| qs6!0~tMu4YO?G'N";8IPKAhdf#org/gradle/wrapper/Download$1.class}M 0h5Z+v/ ׆p!.Mˎµqd#c-l̈0b\7pK^\dNPKA&!org/gradle/wrapper/Download.classY `u=OFVH`b- FZ$`l#ia5#r[7C4Mܴq>Nb'NZ_ N-n7m~6iM?i~Zm}}{Ͻ).HກaTgdYټ,W/_/ienxbWyu[ߓ˿/???13y E}Y_Kk[7AknU3rN_3_RͿd|]6)/mG7to؇t|_&*HT8""$4]蚨S.juajx& Q/t]M5ѤhD.uFj"n&]lͺb.nV6MlĭM^J@xs/-fYSzc b9& ]'[?a2v@mCcOT ܙlrJ5?o{Cjq5gSs;e931K:3\6yLrހ奒#;J8TR Yxʤ,/X~fe6Ge0};^t=%l;Lw0܄$dέ oRq.s MNzSijs3ޔ-0.`Kc"͛Ma&|dpʹݬw< 5S9U2϶vMtStiCi )v~ QѧBN4C.IeGwc#x xw*kb)L_gW}51`C)HŞ\mR/3 M511$T E D ^.هM1"kb'Ę&b8)M1!LqJ,)ބMik3wlβ@P*mCO  葓'G*ADg{Cms~t) D}7ZHQ2E+GQdFtEGS6ӃdK'+G|_&Rʩ'Cl~ ϙbRrJ$LaiE>41cYLi,&69q>RmE\9tʋ*M8HbN: 0W/+6UBQ9{\k2$Z/Ltis?ZDori1^H9dϳIWM:z1Ϯd:~4wC"ʹ&щaهrAޕrFnOg 8.SCn84\4[6mLq{{{:ں=Wk _dHJNb^:ielK%(-&ZB XU[2Jo#mi+⽱ZF0K#Ojayɍ R;HziUٝ&}A RYuemi\,gNILIJ IcgX8. EI))JLL6,M\z]oˎPR$0 !l9g? s;%caD`?,_ |+k".]hAik/mQUl){k{}w%HUzRXh^\mn[DU[UTu^E-`[U- VJb-nC؄J]l.YJ:#zusM띊o@꼁,֞\4kx&{_Uߜh樇}:{1!10hGcD02'qO8k~ԟ#k 8/ q|saR75+x~;U/.B!5(gdCxj)$G^'pb/*Q>T GPx?>@,M/\bp#|T8W7a(ٞ@ p#=H p=|,aE] O#*g_,*S3s2BW/Qy=%ΆBCJrO0lO(<٨+XCB1| u#9G IR%̈Ad% rݸ^R?YU"Ta^H?PKAN/ݡ1org/gradle/wrapper/DownloadProgressListener.classu @E+jDEкe$B~҇)̘֢"n~<0y&\8VfsE1a< 2y`+Χ0TxLnQD1Aj,IX(D8gQkY?0&/׌##Ѕ>v~PKA~0^ 3org/gradle/wrapper/ExclusiveFileAccessManager.classV[s~>Y5685D2؂@*$FT sqB҇xVKOIKڤmzJچrӛ"әf&τ\3nMgzQ|;}{WG8:¹< 5Lj8#) _ӑ9yA_R'u5\T% /R=(4U qYV%:`+:ᴣ:Q7tPO@u}WqMu )SoD)H=喽J$m׹Uw45.^A>PoeIKeٷ1#Y"Y lrTg]+̭MTI#=+I2qvŗt\fdjlݖt=^DE'/;JK[֑a4^/5Rp~00Fkho <NY^!OҼWoi557Y6].p0՝zYHj$%=T2/iG I'E%)4Pĸ9~!9ӻlX~glDiNk0K17k^&gj_ nhwpR- Va-a~n7!"^ 3ܓά~ENY^nIgkI-UzZ&H`/c!MVY׈/]ˮvڵM35ϩ l]/u6;7`v]-%#kD͑>H6 9=#\rb Z0͠)nC id 0] :zUW q ! wӔiS;?Dl`LJh{?T>9s܃&<1#5|_~nDRb\_H ,">"wq@GΡ;0N 6O躅Cj:} nn{t/r K2- v:\G/Ug}[qx[{\"h\2l  >{*yR 4v.ebT.ٻ+`9=34HTM| )׹L"hgOE7s-6ep_̴kH="sEVb˔z5UVMmm|G0Z(=ç8}'Ei Q}(8!v]wq&2{ 4zmOgoG#|gmH"aAG5=R+ȃɹws??~XªQx)I)`^F\F ṂzQFRhMK K [A*_ɮoANϖvӟtp854˰ZsM0ݍ+e錞K{zahӱa{jr⿅ >4fڦ?(06 %L7k}8e*)v0 DqZ5*>F]m4xqNuj}g'-mZ0Zjw䜦[b!ڋ3)UD0A\>yIA$Rf MxfFӴ*e]ӫxwԯ x wuH𘗽P`{!!}%nx/ q}Jhͮ0,މ=q@{,Qzii“G7 !8CH3 `_[(`+8$U)<$4OZd4}/z@:CYׅ"D "Vv I (&%꿮)[|SW/9s ,n%BrUv/PKAj 4*org/gradle/wrapper/GradleWrapperMain.classX|?}ll0" A/l؄Ա dlcbHBE$Nfu$It&3I'I[ٍδN+{;˲t?޷^Ï 6؆UlKRLi%\+xUrj=~kNEĪToRf5-*㭒vC}_ ǻ$*= ޫbާb(xP><~|@E>" 5)a?xDc ΪxV|Tq >S*IgWEyz2/LHZR-WGb kmQ-6F^y%1Wk<ǂ<2*5[qU<((#p.<.m {9Dev4vG *Z_x;jsH0de[׽΅ .y:öOνyu/r. ]T9x8S;'e, Wá2JRilSƶh8iūt--إb֩p!Ů8?r/>ЊΜߒsS$7&drM-2yayyIR ³Ϥ @t̰Ʊ;Wh[":xG0zxr]CvYcN/s >gh[Ry.ϴS|l=1Ѹ)/@nE]6}y1xuss(c;6b7f CX 5Cy{y +I!!kqAR7p82D1B,WN㓘i.) OeJx~l`P(BWHpB)K$ 1@k$[Bhj(Yy7O"(7K[OV56Napl ˹VLbBSX5O`.rZźw!u-ݼ] E# ;тt5@lAI\۰w/Vƕ8vww#|6џe@_2 9nDX{a"$}~3ɥRP( ^q8i7_on5uh[낚@3b@'ElZMTQ(U]yUNvMο3՝]_@XxY_s% l(dדǮ6ǎtt{W56f)Y\eCEd [&pYh e^,Gmx~$3κʴO3&+( lA`Sg?V[=Y0xsi9]_]f0K;DYtn|f]:ۙ;"MigS;]=1vK<4i zY}g}%je]cFz* lLL5L&z9L_1w1uw3mČ^ݪ _I ΂ hqk;@Y{A͢_z39E ,e! $w wܷ+0H0}uSJ`IPKAXs"org/gradle/wrapper/IDownload.classE 0  ^b AP^26J;t>;ɗ|{z~+%5O&WΔ(a_4[gR#!XbQVg={}1AYCX'R5c/J$S@pP\mKulPKA6"org/gradle/wrapper/Install$1.classW{W~O2fZp[hF, P mC@Bcu;l&3,jVjTw%mW7ly9||;9y߯` [ C22LɐO"3Nj(&DAd%pJ"eMb$> /H|T8#&g8"yGEc|':I|JO'pAğI\`BO%<>#KIBg5<+ ]g v}*=h٦Š)㌑ܬw+tN*Ö򬩊oθg9lFD-:'b3DO7zl3 =e'yY0cFc0m8" U٪]-zc{.R\v^ozSQSf}er̔59)1ód]Fe PGl\6)kl㾣tBu{[gJՏVm"ogG|p=Q1Ӧ/wLPv:k 2l4[n)g1mx+=,R}ZLnd`;p.o$fYWn:e%Ҳ1~ţ}癎bHcn˛aJZַ {n>d.[t<n_v;q`A|C&TXJ.xAǷmVq{Zwd=u|}~ C ?2.)Ķ cyQ g~_ᲆWtq:aN5Ẏߊ߁^=lڦ/NH*&kx]:~?h?U*.w&xKl~R,'Zk~bdd< w8=+jF6E_^$-`y+$1cҷķï/ݦ,AÄ"{zZkspyǭrs%q*v74cf]x+pkm4ruggI,6z2 Jvm_:ڶ'W-.(C,U 1ǶޮCgl >FFǻ[R4+AG]-ao߰[+J۪M<:ҽϬRrf("S}V`rn Mt(7X7vGoZ3snK)vWfWru<+"l}5VĻ8o<2f0 fGZt@1y*s,:2}dRYD3,bT|UhPp^z0 Hw"N/g|h*Z >wTST*UqgrOEa}h:9=<KŮᝊ=.7ARq%DS ϡg^yU*x7ihQwo9 L3wyqcs\;]!}k9#MN"rd$yT ɢF]aߤ߱a&^J2xX ᠪuU/z 2y1N0'Y4dySTZ=GVaR$CWA(apL2*zc5^(%QJƻB_7ՠ̻8ǃ U4/PKA @$) org/gradle/wrapper/Install.classY |?'ͰB#v4( ! a1 $w%ΝKVVZuwUT4*uZgk}}o{o)fnnnAofs}yoO=KD q@E4y҄Kx'~.D!(c)d"MarQni҄) VU!Z^^Z(5Өy .MA^+!:I>[By<7ISͼE nI[C/Tx{Zo3_q4 !j6Cۂ 3. NyFEuh͗Jc) NnƖ)w)n^Y'^y]!>\+Im .'F)\߰vS-74ڴq]04]ҫzŶx quxf=.W1Nbє:;-=5{,۰A F=hB0X.ݺU cmQx_u<0u%`6FU*Иiصf2;m!ozcSz#nݮӂ:V7SGB{ڨgRreOG_]Ѿ3)YZ2VjTb`CNQyDj7<FmD`ԅTӼᐭ_ymq hZθn,c F0ɴa7TXqadRPMmLf0~FdGܜ#zuŠLG@ 8˨뭄dU)3RžFNceoEΊ8XZn?{JMk1~%b1=(S"axLn1ؗt+L$uY1OR$S~ylbhO!_fB9\#n[}_8vj$GdruY焔)}Qx*fd1dP.J|Ec!:=ݗJL&$Ui WZfbi 3Pc _v U~m˴By'">*?Qi/ݣ߃U)v'kKR/sbz֥26Èk [1S[1m46߳ܞu[ ,4XOxHL#*=5)Z53 X#%nAt6FvZif[-D1-PShPZԤ!;ctmW8 dZ@3#GU>ȏ)Oq?YkYzN*?%oLj3)} `,kiZaaC=\i Fϫ1U;H4PSUރbA!д՚ʯ˴ٟ^1%Rш(z}Г3MNp:"p0أm@I$TU O-eoT~E Y;¿P=A/!KT|̿u#i)9{H[?T7|ʿۑ['ꑐGOCuܗ3R46s ^js$%ouƞRy֎ӎL vN+envgNX9z,2 SqIp7W(8S.2p`~v`^z `r۪Fh趔?paE ɤɹ%̷k/K K=qr@xt9nJűuCrYC)3q> t8i9Čm{<)L%e0.cƥHQTɑ=I_ۖLDS!4rCe?EdCdn[rd]I}yr/#v=J\ "8wDTqrQW%7N-ǒن4S6&ݴ:ǚKV9o'VN(fPQVy8p/ZMJ_IoP±ɹ :9FIeFtt d-7TZ2X䀦D' as6H[ JPi0G0ZT7=\9z$UP@,Yhhw>=T 9i v"9XA zu꧁NµE+~X`^Gʖ@/>FKOKipur^E)0tįopb >xG$<鬂@5)'ț?yt$[ԛxJr`t 3x4W>[[hE5pVAjE8$C[vT fCfu |s+m!t6 'HP9=KG~p+eM1j⽜ܞCcyzM>c ,䊃TZ^r^̓tNkm 8e ;N: N50Vcy9q4OkAjnu2?@$S`E̷&".~ڲyEhv+^ lG3r"#qlQ{`{a<bD} B̷!{W#GLE<(:ޏqXbcp< v=%Xjl?/+งfѫb,z^kh'ހOwy{ӁL?u:V@˳Cd̏@c=%{aZ(}|OE7~Hmc-V@`2! &˶bf'`)Gu\^u._Q= Eɬ'Ϡ؀9RV-8\3;5PKDfv G:XYI4't8pb k "ޫpd Zz8I:GKy=#I'32/ (zvv O No6λ p޴a aQ-sNAsn㩨PKAy0Vorg/gradle/wrapper/Logger.classoPǿ*1pL EWo1&d  f&Ki Xs9* OHfp+ۨ&Q,D];Bܕp A`w 32 I^m{]⯧>wީ}N6Ǫc;8ƄԵ]u`r壣FQs5mP힩TT-]鹎a-PDս +TYb|Tuz|<[(N$&cbXC3RWu:+mf=mdϞ:?2RlDLBZB3{/AQHGB< 2og\+j/>M\> {JQV \ZWHcHXŠ\_)Gt } РF If:B~@R &~ 6H#J2b&J9򼉖=W-ĝPKAbi+&org/gradle/wrapper/PathAssembler.classV_.HZYm,c;`HeL!mM++NB눕WzM+=Һ}֑O{]K&A{7̛p'>ENs2|^m_1$R_/1UeFߌ[1|/G(''_V}?Ppl'Mǵ͵k,Zֲ {;(8uy~lrzjuiaj~lnjuaq>;sYA|~SuY0Dr\rbPpdn~#[%ǰ6E7c2E&"^RО\V(vtڴa/kEC e6el Kv!ܲrٰ3s1ƨ6iѱ~hple)fBEwt-pl}-ʹt ;ƌ.DF4v0l\U`6콃8[d^+AC-<|xp`LqJ^Mw;7kۮTg۵q c<ֺQ8 &66K7-װ屣 %K΂-,gY\"Me]AFǙ!hTHʃ n&+}_ xTqQ+&C 𴆧0!i^佖5w4<Wx, [ÏE')hFs 闸;p+0Y#*~kO.&eBr(⯂hGV؝XiwT)VMl0U 4Fw w񚆪D|Wu67@ 5cƋWa3ɛhx]0]֞db0Dz*"[QH`fnj<ƽJf9#+~[w:()Pni!gn Ca1 XVk{Qp'dlX ෯PrPXR.=d$$ΉdeRxcgq.ݒы:pH!vkLf[BE/:Mm56}^ndgUt_`# I?g}r{l꽜T ʫ%1@BxJ08ݘH_C[?Byv`I$% &0/=oUʥjϤ5D.v_"RCWZG8t8Tq΋zPPdWRH ];pzd(8JFO!MC°t=ۘ"=~^f.x&.S] 5 RxR|2\85iD4%wÖmԱ x8OKm)tlA8n ^^4O2SӴ#㳞 : =W 'KqhJOkalq8Ub4^"/YwI%F&d t<:{`.w$-AGX'^dW{hB4z}9Xeu{d$ւ {̠T_FVPKAdzߒ= 0org/gradle/wrapper/SystemPropertiesHandler.classV[S[U6I8!ʽ4H[[R/H%*$%^!&sIoZ_go8ԗ0Ȍ}GN"dk}{N?o}Ť)0uL+#o`VN"Rms }h¤Q9.H(MT,Ixx+Xk>ƻR'A#CuZlyz"Ls}-D(؆ 4MYfLgEKeu 4k{ڠy0hvk].Yq-نn'id,;JFJmZ:ۡnѷm3IS8(kX# ӬcBZ뼆cږ 1Y"L0#daz'2swH-t9ZXQuE#nk%?>YnMCyE.LV×.91a({7'\mƀ 1#ajNfgs}☾zT<(Z2I b\Q TZF" CnQrėcl%ήSMfԆn \mNfo6=[]rU&IRS&dH# Y/F戠]3お/𥂯T|oOR-X@-mn+6Pyds 9vDx)+#tLbGbT$5\/۳^*x;@"kdPbYNֶ|xXfMًeǠ*$LTaS?zeZ*g?e̪$"OfBX:]w"5Yr%.G5\k p82z ϯ%ZBG>yH?e}GR-0A 0Q\,HCԭ57GxCC!5FFGrh׷u-zPDZ5}?ΡёxT~wg9t?z=~9]g=9V9z Gq9?1q4#E;Yxq䱕 _Yg9Q'-aش> aWI\b>t.awCtGzm2)p+G%qg\k,s0K0Q,Nwڵ|_F^{_PKA ^F-org/gradle/wrapper/WrapperConfiguration.classmOAփ>"!3 U1U H|Cp#]5Si"1e[733\_(K`6 q?<7yBi~yVb޼[_2DO O[Bxko 2;d4q.wXs $.{{yxsx>;Շx0Ga|O>gـϱeċ"ƗWC_g3! c}I7M, eq5oIh9ܛO zG%DNΜL[: aI/Zj:J*CB53uh>zh|=~ zSZDN.FA3-]+JXi^\g'+RGhKF'$Q1-}99eٜyT t=w ZCMIzNjnhvŪsV˔,YOk V'HQsTSgdSL8ZP6X-$Htٓ=g3Z8`Msie ֋H%1u b[ly (%3S Gv ("墩Yi9:3lʓa=O3toP!,P4kYE;UW,!! ɉR/U.2 S *{f 䣃mH mwR(ańm [1C>/9@U)vq[UڜPeϹEjj)wG+1_*QIg$Oې͔φi"U"y[QS[_]2䂷F%jm!V4My/0e\\qxͭ0 IǦh<:ID7o,nf "Rl` NpC ; S۹QVZQ*xŘ').) ~Ě$R5, 3JoN4'6FIOYLyT/"4[Ϙ* fM>}mrY(nN3>15ӭ_)g_oW%lU̱|qz`c!Oc=PG)?+ 'Ec!UCFT$|nMGf/>k٨ufP^[YO|ܮcK5<1UT1_UME?Zο[e5nx[LN.z c;߹ONQh tfZ"[vs?*_bͮxη k ZŵhD22܋af&g( [#F$~_/{Xɮ%|nq1i@ D:l#^Et^Ctz==z#"ܠi-f4/!}'QO+eFEe-М܄} rBHy4FKX<( %XYOG#'ѕt?. hg.^F#-%eWwvYBHhBem` Cߢ/42YPa9{s?p# 5ϕBy9yEx\Ye ZpՆaEjw I7|ZU c:tiw]HaٲzQdu;BrQfظ&DV\%6# ǹƙun]s̷X3ػe9H%ҲIk.=,ȍ-1ak^1蔕)9(id0^}y_7бp:%`i6t*㰊jBL F vUWm_ImuוD I8H7Լ&%DDՄ3qONB :ACvsπb0\8y 3CEw&T*kY+$@B!K͡5IYF6VANwh`+&XEKH,έQFV# J#hRq +dAy ~#TN*)oްOOSxΑ 86'??'k<ų@zV`PKAy0L ;org/gradle/cli/AbstractPropertiesCommandLineConverter.classV[WUN2a`rԄ IєR Mik$d&L0 ]w_ۗQ\KWuIH&Ҕ%Ys}g?!>񉄋H厄Uܕ{K3 #;ul c[|b_J!7ØWˎ"1 4]W6jTѬjA5( qYTSt)p#kDTJ5QhtMhoFM5mMVjUK쪡S5 eDhfk0fC%QQr"g^&I]VZb*ZPr$iж"\<50jy6ZΒn"#iLdC+LF*ur52TѶRkZ )+>m2a*CQl#6zDE U4vryw0oQTE<洲nux|6zNjnԢzY ^:^79n{QN^dIvew' 9ؓQƾMȨBaFw@M lu{ ]ʈbعJwHrv2̼j<^.T߆ڰCzs.k%ꕊ uRj9M:d|",}@\xQ ک7X9|ue0 ޷. |4lk;ɼIi ߮?+&Dw^z BuN2#iZk"pOag$7J(Bڌ"cܯ!16Cfkt[¤q0~xȱc3B( p1YITةQcBW~ͿDcD y f@]tӻLA^%6uV18(c]3 2gy=ݴoa̝̩ kqv>PKA'H g)org/gradle/cli/CommandLineConverter.classQMK@}ԯ'"4 FM)HQ ޷lI7ݔ6MBya=t$S l)8A {Oyb :˄3I5' JXdT"qx{a/4OR1=Q615 ڹ6ƇEWbRh{'qjհOJs>j=wSC,\Kh _kF{QOٲÑ΋i BSWβGAAp0-C*8KT!: Lm%z( 4l:n :YA_3'1:MX^+^Q0k6jcŖs?y ]q 鄷fkRL-c ÛfiK'?fxCDlnmMJ]k'iu#0BWՔ!f,By@wZ͕t!BI]#HI9|g|{ -|PKAF= ;org/gradle/cli/CommandLineParser$AfterFirstSubCommand.classVRPN[J;be-x)U.R(XE_( |gtPq猏C8I K0㟳{v|gw9| E͈4тHA/pK\3*B;ExqOH0ԛoB Нҍ\$gȫy%ͫ!k)USdqW5ռ0l%^--1x[+T^ɓğҳr~I6T/ =]kS1&U`fV Ҍ)F2/ Y88f-IMy$ M1eLo H2ڼ})MzٛZʑ"P\\y6O\}9ݲ'JBKeվm gËYzK/ Z1}**2 ~CG&1%`Z 3t9eaёgںokǔ@g%Ku`|G%E5e uS閻5ϗ4ڪ۲b}B\EU6:YPFJjh>3L578")&COH)YslH#?nA;ΙZwQW+S ^ͼ8W'AE[=|>>Јs6MMT > wH{im::^A:HG8d"Fx&O+\AX!vѸq HW69Z e+˝}с; \G7Ĺ(( ]LSh`cB p.9Bj1 D'($U<"z@'bj']GZ f<_PKA_>ң)3org/gradle/cli/CommandLineParser$AfterOptions.classmOP( @ 1Q|Dt1wkծ5wE#]|!Je<1Mz99s۟PƊ'IW1!y &TU,2~=x6x !fu'\lkug!s6tL![/Hvfe׹[‘zdL[N!e`oyXuyej&!&)bA02]yr^5Vf6R\4dr] C0ԖQi26{{J.-ߏ޺&Ww{Ֆ/IU0ᘓ*3Le o13/#|]ˣ%D3zaS2u7XMc qaA78yWN*%1|avť-N x!*JB)>#iIShln|AjTDVИF(J9DRR; %܊1C!˩1(|GFJfKmt$F5kj|EFߧjr dY+!kmrQ>э=ep2n&0ft9\9E'tN* iϤK49hPKAC| <org/gradle/cli/CommandLineParser$BeforeFirstSubCommand.classVmSW~npqhH H@^A@ m6طMXdl?δ[g?swc&q_˹=Ϟ{o/q_Y̝f^Fse,ZK,s 2V9qS>1M8zta,gZTRwZ\SYVSݜnhUYB/n/1Oz_G86\fjAʚajJuG-UʙeZ7^[u(5}2(Y٪Zk$N(@yA(تeٺip|0Xh5UUJ`[QtZw&~ժ:ѷ3<MSn9j*ߌq\6 G J*yN;{2#hfc f*SL Èaq!y" Ǘ axIbtxh.x%0&X}?:Ui4^}R֚Be*"nE& ̗he;*J"b68pÕ7rC.?V-0fO~^y뱉6{*B^*Y-z=:7CRyʍհjHC*dfN͔W{M ^pCMPzkt_gBR,]x;^Vw+eM+" ;zf7i^/O3}xb]C[B]d5c|Q%G7GO=m~} ҈'ҮOC.i^yb;30rؑ^!yڢ{f U5$:2lދ9eX Lm1 }IbR$Yy|M7PKA4:Forg/gradle/cli/CommandLineParser$CaseInsensitiveStringComparator.classS]oQ=wXXE-_vK_dbMh~ fmvƿŗ`|G.̝=sf_hĀ:Rx:UΐnIWmDrΐxCѕx;E ]osKu :<7ЗM.Cg0m~A (uCGXGZpw~ &>na{pwl͑HŦ2Pz'l(G#f`?_A\Tlתk4հ A3d{rp4kHem2z7RR#d-GLbߤ2-4ʗ2c bZ`68Bg^&j) 4:zNegD7nY}"GCδ ji&ߜԮd 5ݏHΔ- eֲ04fFyf̪mJ%OL}$N\9r7FzsyݘY-VV.Rx=z~>ƝgF\*;x@OPKA?h=org/gradle/cli/CommandLineParser$KnownOptionParserState.classXkx~dY&Vdsд!@LawL73ۙJ[^6V{zJB^֢Oy<}ߟ=gfvݰ?ܾs@;ޕw%!cL!XIy^l^^\OIcB w10X/LtL{$|Uz|={el}2oɷd|a=_P8? cu 'yy(agL K%yT¯$PkXk戄ٗmivOBsh['Xh43g~vt&@A˰qlŀ1bjn&5 wfbqԴ]L0 KB}qeF;zylXqLҟ;C Dl2XwiMU9^܀¬[2ԘnհT ky|qG\R-VlFtw[^ZpwL&tJ%Fڬuu[s-ά }A*I/m^E> @YK#m^:y}` b(`Vʎ7hZ8h W '̇۶g vGVlWpv(؉)BA 4.nS:vA :WV:ujմ\UTBq>:ȲI6)Oݦ%ݣ6uBsԤmq=ajl-N+8*7+xC bH”.k8e9tTij$ *9 cNsxF ^ ~ڛJjFN趾d/?BQ2xEE*~!Ů656ד#f"zHxS[,mUbj%pAMi /}2h4i-lӨo/F[Xulxz o,<#^Z<;^=F݉5y豖 g7eÿ%8#^+XLw6j=K֕ xr^ i/=Tqܼ42b6vc&>ܰh0g|,eŨCgm}C~Rqᦊl>hAdf^]2gbV% %jO2!ǃL&~spfc9OvtYSr\w 7V*9dpŮ[ [3Gҩ7~vOrDx (Sl}izKg:$D)ahJb9gP6hhVD +$D땐h<Y<*6JBx M'X>bAlwĶ%Y9!e} n/*XO߯i;BPKARB <org/gradle/cli/CommandLineParser$MissingOptionArgState.class]O`6d 2 c&/1N4`W5S/ ^H#2-sKƖ&9O״ 1UQ4%ի*gE)D15u KD10ؓp :@^3\ٲŪ+^1UqQ)B&@)ʊ! B@n!fU(AiW|ۤᲵMnR# yd8!>] ZEGO*%UoulGQ/\TMn˹-02#/hAlu@t%-q*2Nw e&BgsQ\gݧ-F͌& nX{~ϐ2zfZC R6A Ӱἆ%GAa3@Lv0Lc8~csC7]Saf=/b !i!WiXN$!9!:">i!z&|'d&w`8Ig9NOXFiХ uD5x Rl( 38qhKCO p:mO.6p'%0o %rC?7o@Ӹ<=@QO b쫇|CMa"!˰߰d^7Uܰ^t^EY3G5Dh8%h+aVE3D*PKAs=org/gradle/cli/CommandLineParser$OptionAwareParserState.classUNQ=]v~b 5ȧ 1$ 4Z֏G MD$>e[ZҚE̙swy' :iB&1!CŴ3*f5$0b^E!YsMKc2\(4ff[\xf}f!?\t]($zr,.épx| luj6%;fOxզPɭq…% v2^l"P+ܧ8ش癄[2۟kVMa*W`sQ(r Q.$uv\ y;>ަVv[fnYx2=Y#X13:ncTŢ%,Xѱ5Pg),V5H3 K; 3'<<JڵSΐ!/g&W;;^ݸ v,ly/x-/TQZ]܉\E/˕=lk<RwgrŁtZ.=EPhh+"J6mL+"F~OA-, f1B 娆E$5"y  3@AU'2P*= QE3D5@}C2Pw'3y=RHI8[-vtn5@ $}_+:)+C EoPKA$ER7org/gradle/cli/CommandLineParser$OptionComparator.classUkOP~NQ: SV`PDpFd?̒%mG$({'<弗ՏJx+#91y 2D,ʤ[,aEF)JX%lH6- uA,&PrV>6IUngq>gg1(1mI5gڦV-M9a>oz>(7(-BK=VȧۦK %kVqbh4ZYNCu)t(zwKԊMe)-—E] 5A׭Zek_-r-ܖW7Y]k +x- #4[%l+`؎ Y6|[ubvǞA(@a*jc(us ZG?ԺAh j@.M94TBjdXh%j wqY4:i=#BvQ1_CjQd%\;C'vw]BڝmHq8!A>6'1"Нu%Ew6 ivΝ=w _|dŦmTҜ1d{mC);BNDfQkikɄaXq2ЯܵRB7"$(XAy/A7A#>;:|(F ͹=U&KcEZP|yq- S+·\U|HQS$dú݄31TMZ@wťK) sAChU0^]<. uʓGa(M"%z.업cg"W=d1fIٔd}F6,O).a9Ţ,se,~Bٿŋ#Ho8ª;UZ-6d7_8'_AyKBkxy>Eӂ4Ui 1 z]A@Z Moz~]{*Ka:r摎, b^hxc+ ݥNC:`U L1r) ;{2̜?3zE9nB;;esy-w'i< MI^NN֤ʍЛh8Yq|9wvc 91Ȩ(` (OaʓbmDm>X1#Ka ~B>FJg!6l|;4 kaCԍx0r/iA';`CG G¡c#'30@ÌnmaF⺙igM1*[Ĥ=\E7i PKAmU=org/gradle/cli/CommandLineParser$OptionStringComparator.classTOAfeZ`XR@)!&&M1m(2dKvN<؋H41o5vޏ{߼y_} u [b"obE<6I&MmX`TC0RS|jH8p3ZviG*`HsB_<=!Yu]Wp5o _:Ү;&lS ݜ`#kT؎pvYEJ}j{y~H8r^Oj}W*.7 A3$TaK}OZr=6_ϔtO%DCIb}9VD n @T>]q& Ry3l,0&h.6£p2=5=L\7al42/\ c)HOvqt#F~`xFl{WkȞ0V7ڟ`b =о*Uuv imS!Udpe=ҌvoNB E.0:Rڞ&$ɻc7.\1fi?>G (`iH mPD;PKA-h2org/gradle/cli/CommandLineParser$ParserState.classSoPN) sT4706|3M$m f{@ú[s-?&>GmdiҞ|~pF62΢jv,-4WׄTqB0;%v=^(>{J` aŴ9,D!PKAb'n?org/gradle/cli/CommandLineParser$UnknownOptionParserState.classURA==$a2@(""UZ`$SdL__\H*~7= B%q}q[A :0A0iDq5k rCw˺& ezabq [vJ%nWL[F ! ɕgY6rkk X>O 6E=}9A{gN)Ҧ>曖.^]S`{jHO~噎-pe}^^rJI=R7٠cSޒ{2iOcinی1^#r޲t1b7?4yvo=)L^)aFZ֜[c9*z'̼n$T BAa F&00 mGd6a4j3 C'/n{._VƵձ+1DX UR%H}4+.L_c=:"L32}F7>L`g%.dJ\)+QLgq.:K^dI׵{""¾j4e8 ț#u`@Ts䥓?|pP{fr mEhQr!"XOٯUVH/.\ 鋘6?9R?SͩUÕ@FcH:U1-NcjXc4Uy<blfq=PKAg*&org/gradle/cli/CommandLineParser.classY xTյ^+8$<"cB2DT0xHNLQ*Q`}WVEIQbm}>l[{o{zk[L&={wD4OE>-NS:`ҩYti]:^4i:TNDOnvsS\stZ5^qNxNHkD KuPft0\s|\'V9OfhH9_>|!E_Gx5n*kDK|(j|NWu\ ⊀_&Zy6xXr:͢ Ѣŭ>nvLdh]ȌŬӘU`$muN3j#Q1M3c֊p ǂN+)^n;ӸKÑ]a{rYƸ4Aŧ`*p3c1gkmj {m[Q{4qi9?B3ˬH׸3e]fʐ<~tk1j4/߲v40Wn3wU]`*jYVqh^4*b+;cTu `[،wE噳!3Ve3X2mj-!9Jme0l.\f(Um$5/XQT(ڲ*5>0X1MB)[C,] K\%uK6 :ZѵBQ+#fh Ġ;S>aogD|IJJ"/?{)KlllQK@pk lw7[ɀ旤Q&3 1exp 2^Њ$W,fHB9iȰTA/L gnqי1:m^W y^937=&3; Y9lays@ 㴒A^WV![Nlf4 wCVMAs6l|r%%,X2ѭъc9#_z`Υ1 N{ng߈d HŊ&oɑ4Vck i cQn2yU@C8dq~r4#3rafmA𝖵Ӑ-tچ1zǓ:H$m}]$ǝB<8&Hj60S+e1+UxvO L͈V)2dcKw$ԧT*fsM;s&A@+srPђd@X%~R,>AŅoBBotEQ ĢRtmAӠ/mAw2O^æa8nˠüzP>6xk&q;( winҠ1⡎A_e+ ?#<ZY} x ˎ-o ?71U )4Y7x `mEo;h%Kc ŕe.Cm5/2Kiy͕eH48MVˣHâY HB3u{>COicOYS}q1y)V\, >!i85~Q迁"%Fڶ!]8S8j/{3fcf]܂" lT/i$jp8/6[Z#ikM 'GmX܌Ƌw3*fTt ݪ¯jdLɱ8u7Gq3fۉfX )TeF'|_7 QdBɦ{6u^U߽S-J ~S%\hC8KpiG OF1wxp:bp4}GR/ŭA qp:UY?z-g= hະtp?"@4t44Ə(J}T0qefl/*_8xnqV?F:fVGdQCYA'vtXA] X1jM4XX"JV8#$Ӷa fdBYs@ֳ X]-pޚ J%+2'|Zy3+vܚ]yZ?IKUfQ S4p[])  7DI+6~6v=N Uҹp#ڢA[ !q7cODh$J5煰PcC`Ff"mXs4|'6ۭneX# kA:ŋ>+ Az /)2{ <dMÌOkfP;#\2fVZwE}Q*Fn2ld/-t\7*błQ%u„֌pUOHzYKxF/GnEտqW-{>Q8+(ڎk1hY?y^=܋h)s> 6GN,\ӛz)5ݽ4BH&Z t0f !Gӗ1gkrʏTW*UnU.=Bb w?7SASٳB^:FSQSY{iLCyw8C|Rz~oOCg' =4Ӥ&Y:ߋ:)=44tEP/6e(fB- T-:KѺ= R{)G0ԇ)TŋXlDMHTѻ 5* fB<Αvj9Kn-U?D+}ͨ3Y z܆|Gh)tIW71e|<$\*Rߍ46Tx-]NZKWi+uS3]M-tYHVDY=K黴~(DR0NH;\JqG]H;y#m͝t%參Gi/?h?O񫴟ߦO7V+9=HndWkIqzGl2A3Þ.6iS$C?jAnnWJӠӠևzk9z=;1Iw2{EnGN_Q='1FQ=6Ľ z1:s1"!llJJ.-\Y bq/Xb%MDU)BJyfxܪv/}[ł0Zre19& D[ /7w5 q`{EG$\hT@'5x^b@̥8$sӇ IyyCMm*[|V5z鼆r·.qW(k8)A6 QH(b0Yl.uK@. R~}NG`v&LO<}Q8ZؾJya `}tKg" h=v^ $%ӊB|L'Y|zw\].J?W:OKb8)`E%ds!1z 5HCo#<ߧi'1;Ԋ(K5dC ~ORiq >5$i\~~D?FY 4plFnJ3I9hm=U[/0fx#=./ˉh]OWf{Ik-\&4=Qx&9ZRV&"s$Яy@( oh9}Pn+G?8*uzö?#܅ϝ頯ֆG med''EѮ@{vG!'Q-KAT8ے۳RH,C H6~7~Z$g/uT*^@ /4?h3 COm>o("N{0w-r 3=81EIBhG1 hAv|v7*8 D O$Wr"'߇LXV<#v?UgWaKwX.ҝQr$a1C2ԭZ?vPX>Bn3TV;Ng(!j<$}E]bݍ;[7+.C{=hg^uuv?ϢEh?7sσ&7no?PKA[xn&org/gradle/cli/ParsedCommandLine.classWWFDUǎ&U8e9$x&M)qIpZ k"Ogȉ[JKЅ}34 &?7oW;3Kָ1/39;|g??0O 3Qq)ģ"Nݩ&I\x<#YKF\e'Nj2/%Ўd|9nq|E_~Uh&N~5X~]|Cq@ěxK|K<-ofMrdyֶtȝ:XZ2 ծZ_GeI5.ȔiZ(i%}qժh sqQ5 Sr,9UfVfSƝ-KZJiWlKEsgJ6nI)5K=_FHLMxf5 MBzʜWKgTKko&BM !}:ciRvt*݌x&H;b/&|f $t4Ұ\Qq8@ecKgEY+l_PKUƍ¤ t2VeWvjKj:p HWVZ9٠+tt˺,Z=6kћիn 7k'u;aJڼ÷8d%gG\M Dբz5膹UOd״IBA<&\K:@CfiĬYch&i6 ]'ZY`>p+o(8Y:nY XGQWwe|W}?%d735V?6FeTs q83x[S/W&SwE>F)L+5VeF'1Gy݌`RͫW5OޞJo0LA E.ȸ:V61'!unQúwTK7rXr̐[1E0w$DϞCόm  kV%8py'ЃZT1жO~0͢˚QX|l%>ow l (t((TyN,x0^*QpKWj'DG \! ׇ6!$>!À40d)w[@bbu1NbkP$LװMUWlV Gn"{$ G^FHbp,>"]vqvοVw΄1(|v3A?w2{10]/~ ^&IC9$Egd$}uvNrg5&<gSF2#SӤXLBi ,A|;"AmȜ2BXiL8 G:m$x0$aOIkZPKAWgradle-cli-classpath.properties+(JM.)**+MPKAbuild-receipt.properties+K-*+MJ-533PKSPKA META-INF/PKAE?cDAV)META-INF/MANIFEST.MFPKAorg/PKA org/gradle/PKAorg/gradle/wrapper/PKAzZ -org/gradle/wrapper/BootstrapMainStarter.classPKAhdf#Xorg/gradle/wrapper/Download$1.classPKAo:s@4<org/gradle/wrapper/Download$ProxyAuthenticator.classPKA&! org/gradle/wrapper/Download.classPKAN/ݡ1org/gradle/wrapper/DownloadProgressListener.classPKA~0^ 3org/gradle/wrapper/ExclusiveFileAccessManager.classPKAz\Q-org/gradle/wrapper/GradleUserHomeLookup.classPKAj 4*1"org/gradle/wrapper/GradleWrapperMain.classPKAXs"-org/gradle/wrapper/IDownload.classPKA6" .org/gradle/wrapper/Install$1.classPKA @$) 5org/gradle/wrapper/Install.classPKAy0VHorg/gradle/wrapper/Logger.classPKAj jV8Korg/gradle/wrapper/PathAssembler$LocalDistribution.classPKAbi+&Morg/gradle/wrapper/PathAssembler.classPKAdzߒ= 0HTorg/gradle/wrapper/SystemPropertiesHandler.classPKA ^F-(Yorg/gradle/wrapper/WrapperConfiguration.classPKA(\org/gradle/wrapper/WrapperExecutor.classPKA_#egradle-wrapper-classpath.propertiesPKApeorg/gradle/cli/PKA<S1eorg/gradle/cli/AbstractCommandLineConverter.classPKAy0L ;*horg/gradle/cli/AbstractPropertiesCommandLineConverter.classPKA# GK1lorg/gradle/cli/CommandLineArgumentException.classPKA'H g)enorg/gradle/cli/CommandLineConverter.classPKA n &oorg/gradle/cli/CommandLineOption.classPKA2_e(uorg/gradle/cli/CommandLineParser$1.classPKAF= ;vorg/gradle/cli/CommandLineParser$AfterFirstSubCommand.classPKA_>ң)30zorg/gradle/cli/CommandLineParser$AfterOptions.classPKAC| <$}org/gradle/cli/CommandLineParser$BeforeFirstSubCommand.classPKA4:Fhorg/gradle/cli/CommandLineParser$CaseInsensitiveStringComparator.classPKA?h=org/gradle/cli/CommandLineParser$KnownOptionParserState.classPKARB <(org/gradle/cli/CommandLineParser$MissingOptionArgState.classPKAs=*org/gradle/cli/CommandLineParser$OptionAwareParserState.classPKA$ER7org/gradle/cli/CommandLineParser$OptionComparator.classPKAl\ϧ8+org/gradle/cli/CommandLineParser$OptionParserState.classPKAGf3(org/gradle/cli/CommandLineParser$OptionString.classPKAmU= org/gradle/cli/CommandLineParser$OptionStringComparator.classPKA-h2org/gradle/cli/CommandLineParser$ParserState.classPKAb'n?<org/gradle/cli/CommandLineParser$UnknownOptionParserState.classPKAg*&zorg/gradle/cli/CommandLineParser.classPKA[xn&%org/gradle/cli/ParsedCommandLine.classPKA ,org/gradle/cli/ParsedCommandLineOption.classPKAA5l| :org/gradle/cli/ProjectPropertiesCommandLineConverter.classPKA;|9org/gradle/cli/SystemPropertiesCommandLineConverter.classPKAWgradle-cli-classpath.propertiesPKASbuild-receipt.propertiesPK22_Vauto-value-gson-1.3.1/gradle/wrapper/gradle-wrapper.properties000066400000000000000000000003131375747333100245240ustar00rootroot00000000000000distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists distributionUrl=https\://services.gradle.org/distributions/gradle-4.10.2-all.zip auto-value-gson-1.3.1/gradlew000077500000000000000000000122601375747333100161330ustar00rootroot00000000000000#!/usr/bin/env sh ############################################################################## ## ## Gradle start up script for UN*X ## ############################################################################## # Attempt to set APP_HOME # Resolve links: $0 may be a link PRG="$0" # Need this for relative symlinks. while [ -h "$PRG" ] ; do ls=`ls -ld "$PRG"` link=`expr "$ls" : '.*-> \(.*\)$'` if expr "$link" : '/.*' > /dev/null; then PRG="$link" else PRG=`dirname "$PRG"`"/$link" fi done SAVED="`pwd`" cd "`dirname \"$PRG\"`/" >/dev/null APP_HOME="`pwd -P`" cd "$SAVED" >/dev/null APP_NAME="Gradle" APP_BASE_NAME=`basename "$0"` # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. DEFAULT_JVM_OPTS="" # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD="maximum" warn () { echo "$*" } die () { echo echo "$*" echo exit 1 } # OS specific support (must be 'true' or 'false'). cygwin=false msys=false darwin=false nonstop=false case "`uname`" in CYGWIN* ) cygwin=true ;; Darwin* ) darwin=true ;; MINGW* ) msys=true ;; NONSTOP* ) nonstop=true ;; esac CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar # Determine the Java command to use to start the JVM. if [ -n "$JAVA_HOME" ] ; then if [ -x "$JAVA_HOME/jre/sh/java" ] ; then # IBM's JDK on AIX uses strange locations for the executables JAVACMD="$JAVA_HOME/jre/sh/java" else JAVACMD="$JAVA_HOME/bin/java" fi if [ ! -x "$JAVACMD" ] ; then die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME Please set the JAVA_HOME variable in your environment to match the location of your Java installation." fi else JAVACMD="java" which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. Please set the JAVA_HOME variable in your environment to match the location of your Java installation." fi # Increase the maximum file descriptors if we can. if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then MAX_FD_LIMIT=`ulimit -H -n` if [ $? -eq 0 ] ; then if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then MAX_FD="$MAX_FD_LIMIT" fi ulimit -n $MAX_FD if [ $? -ne 0 ] ; then warn "Could not set maximum file descriptor limit: $MAX_FD" fi else warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" fi fi # For Darwin, add options to specify how the application appears in the dock if $darwin; then GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" fi # For Cygwin, switch paths to Windows format before running java if $cygwin ; then APP_HOME=`cygpath --path --mixed "$APP_HOME"` CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` JAVACMD=`cygpath --unix "$JAVACMD"` # We build the pattern for arguments to be converted via cygpath ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` SEP="" for dir in $ROOTDIRSRAW ; do ROOTDIRS="$ROOTDIRS$SEP$dir" SEP="|" done OURCYGPATTERN="(^($ROOTDIRS))" # Add a user-defined pattern to the cygpath arguments if [ "$GRADLE_CYGPATTERN" != "" ] ; then OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" fi # Now convert the arguments - kludge to limit ourselves to /bin/sh i=0 for arg in "$@" ; do CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` else eval `echo args$i`="\"$arg\"" fi i=$((i+1)) done case $i in (0) set -- ;; (1) set -- "$args0" ;; (2) set -- "$args0" "$args1" ;; (3) set -- "$args0" "$args1" "$args2" ;; (4) set -- "$args0" "$args1" "$args2" "$args3" ;; (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; esac fi # Escape application args save () { for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done echo " " } APP_ARGS=$(save "$@") # Collect all arguments for the java command, following the shell quoting and substitution rules eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" # by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then cd "$(dirname "$0")" fi exec "$JAVACMD" "$@" auto-value-gson-1.3.1/gradlew.bat000066400000000000000000000043241375747333100166770ustar00rootroot00000000000000@if "%DEBUG%" == "" @echo off @rem ########################################################################## @rem @rem Gradle startup script for Windows @rem @rem ########################################################################## @rem Set local scope for the variables with windows NT shell if "%OS%"=="Windows_NT" setlocal set DIRNAME=%~dp0 if "%DIRNAME%" == "" set DIRNAME=. set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. set DEFAULT_JVM_OPTS= @rem Find java.exe if defined JAVA_HOME goto findJavaFromJavaHome set JAVA_EXE=java.exe %JAVA_EXE% -version >NUL 2>&1 if "%ERRORLEVEL%" == "0" goto init echo. echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. echo. echo Please set the JAVA_HOME variable in your environment to match the echo location of your Java installation. goto fail :findJavaFromJavaHome set JAVA_HOME=%JAVA_HOME:"=% set JAVA_EXE=%JAVA_HOME%/bin/java.exe if exist "%JAVA_EXE%" goto init echo. echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% echo. echo Please set the JAVA_HOME variable in your environment to match the echo location of your Java installation. goto fail :init @rem Get command-line arguments, handling Windows variants if not "%OS%" == "Windows_NT" goto win9xME_args :win9xME_args @rem Slurp the command line arguments. set CMD_LINE_ARGS= set _SKIP=2 :win9xME_args_slurp if "x%~1" == "x" goto execute set CMD_LINE_ARGS=%* :execute @rem Setup the command line set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar @rem Execute Gradle "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% :end @rem End local scope for the variables with windows NT shell if "%ERRORLEVEL%"=="0" goto mainEnd :fail rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of rem the _cmd.exe /c_ return code! if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 exit /b 1 :mainEnd if "%OS%"=="Windows_NT" endlocal :omega auto-value-gson-1.3.1/settings.gradle000066400000000000000000000002321375747333100175740ustar00rootroot00000000000000include ':example' include ':auto-value-gson' include ':auto-value-gson-extension' include ':auto-value-gson-factory' include ':auto-value-gson-runtime'