blob: a5b15f2bd733381a28b3e7d49d8dfc4191fe672f [file] [log] [blame]
/*
* 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);
}
}