| /* |
| * Copyright (C) 2024 The Dagger Authors. |
| * |
| * 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 dagger.internal.codegen.writing; |
| |
| import static com.squareup.javapoet.TypeSpec.classBuilder; |
| import static dagger.internal.codegen.binding.MapKeys.KEEP_FIELD_TYPE_FIELD; |
| import static dagger.internal.codegen.binding.MapKeys.LAZY_CLASS_KEY_NAME_FIELD; |
| import static dagger.internal.codegen.binding.MapKeys.lazyClassKeyProxyClassName; |
| import static javax.lang.model.element.Modifier.FINAL; |
| import static javax.lang.model.element.Modifier.PUBLIC; |
| import static javax.lang.model.element.Modifier.STATIC; |
| |
| import androidx.room.compiler.processing.XElement; |
| import androidx.room.compiler.processing.XFiler; |
| import androidx.room.compiler.processing.XMethodElement; |
| import androidx.room.compiler.processing.XProcessingEnv; |
| import com.google.common.collect.ImmutableList; |
| import com.squareup.javapoet.ClassName; |
| import com.squareup.javapoet.FieldSpec; |
| import com.squareup.javapoet.TypeSpec; |
| import dagger.internal.codegen.base.SourceFileGenerator; |
| import dagger.internal.codegen.javapoet.TypeNames; |
| import javax.inject.Inject; |
| |
| /** |
| * Generate a class containing fields that works with proguard rules to support @LazyClassKey |
| * usages. |
| */ |
| public final class LazyMapKeyProxyGenerator extends SourceFileGenerator<XMethodElement> { |
| |
| @Inject |
| LazyMapKeyProxyGenerator(XFiler filer, XProcessingEnv processingEnv) { |
| super(filer, processingEnv); |
| } |
| |
| @Override |
| public XElement originatingElement(XMethodElement input) { |
| return input; |
| } |
| |
| @Override |
| public ImmutableList<TypeSpec.Builder> topLevelTypes(XMethodElement input) { |
| return ImmutableList.of(lazyClassKeyProxyTypeSpec(input).toBuilder()); |
| } |
| |
| private TypeSpec lazyClassKeyProxyTypeSpec(XMethodElement element) { |
| return classBuilder(lazyClassKeyProxyClassName(element)) |
| .addModifiers(PUBLIC, FINAL) |
| .addAnnotation(TypeNames.IDENTIFIER_NAME_STRING) |
| .addFields(lazyClassKeyFields(element)) |
| .build(); |
| } |
| |
| private static ImmutableList<FieldSpec> lazyClassKeyFields(XMethodElement element) { |
| ClassName lazyClassMapKeyClassName = |
| element |
| .getAnnotation(TypeNames.LAZY_CLASS_KEY) |
| .getAsType("value") |
| .getTypeElement() |
| .getClassName(); |
| // Generate a string referencing the map key class name, and dagger will apply |
| // identifierrnamestring rule to it to make sure it is correctly obfuscated. |
| FieldSpec lazyClassKeyField = |
| FieldSpec.builder(TypeNames.STRING, LAZY_CLASS_KEY_NAME_FIELD) |
| // TODO(b/217435141): Leave the field as non-final. We will apply |
| // @IdentifierNameString on the field, which doesn't work well with static final |
| // fields. |
| .addModifiers(STATIC, PUBLIC) |
| .initializer("$S", lazyClassMapKeyClassName.reflectionName()) |
| .build(); |
| // In proguard, we need to keep the classes referenced by @LazyClassKey, we do that by |
| // generating a field referencing the type, and then applying @KeepFieldType to the |
| // field. Here, we generate the field in the proxy class. For classes that are |
| // accessible from the dagger component, we generate fields in LazyClassKeyProvider. |
| // Note: the generated field should not be initialized to avoid class loading. |
| FieldSpec keepFieldTypeField = |
| FieldSpec.builder(lazyClassMapKeyClassName, KEEP_FIELD_TYPE_FIELD) |
| .addModifiers(STATIC) |
| .addAnnotation(TypeNames.KEEP_FIELD_TYPE) |
| .build(); |
| return ImmutableList.of(keepFieldTypeField, lazyClassKeyField); |
| } |
| } |