Guia de Integração CDI
O ArC, o contêiner CDI no Quarkus, é inicializado no momento da construção. Para integrar-se ao contêiner, podem ser usadas Extensões Compatíveis com o CDI Build , bem como uma API de extensão específica do Quarkus. As extensões portáteis CDI não são e não podem ser suportadas. Este guia foca na API de extensões específicas do Quarkus.
O contêiner é inicializado em várias fases. De uma perspectiva de alto nível, essas fases são as seguintes:
-
Inicialização
-
Descoberta do Bean
-
Registo de componentes sintéticos
-
Validação
Na fase de inicialização , o trabalho preparatório está sendo realizado e os contextos personalizados são registrados. A descoberta de beans é o processo em que o contêiner analisa todas as classes da aplicação, identifica os beans e os conecta com base nos metadados fornecidos. Posteriormente, as extensões podem registrar componentes sintéticos . Os atributos desses componentes são totalmente controlados pelas extensões, ou seja, não são derivados de uma classe existente. Por fim, a implementação é validada . Por exemplo, o contêiner valida cada ponto de injeção na aplicação e falha a construção se não houver nenhum bean que satisfaça o tipo e os qualificadores exigidos.
You can see more information about the bootstrap by enabling additional logging. Simply run the Maven build with -X or --debug and grep the lines that contain io.quarkus.arc . In dev mode, you can use quarkus.log.category."io.quarkus.arc.processor".level=DEBUG and two special endpoints are also registered automatically to provide some basic debug info in the JSON format.
|
As etapas de construção do Quarkus podem produzir e consumir vários itens de construção e se conectar a cada fase. Nas seções a seguir, descreveremos todos os itens de construção relevantes e cenários comuns.
1. Fontes de metadados
Classes and annotations are the primary source of bean-level metadata. The initial metadata are read from the bean archive index, an immutable Jandex index which is built from various sources during bean discovery. However, extensions can add, remove or transform the metadata at certain stages of the bootstrap. Moreover, extensions can also register synthetic components. This is an important aspect to realize when integrating CDI components in Quarkus.
Dessa forma, as extensões podem transformar classes, que de outra forma seriam ignoradas, em beans e vice-versa. Por exemplo, uma classe que declara um método @Scheduled
é sempre registrada como um bean, mesmo que não tenha sido anotada com uma anotação de definição de bean e, normalmente, seria ignorada.
2. Caso de uso - Minha classe não é reconhecida como um Bean
An UnsatisfiedResolutionException
indicates a problem during typesafe resolution.
Sometimes an injection point cannot be satisfied even if there is a class on the classpath that appears to be eligible for injection.
There are several reasons why a class is not recognized and also several ways to fix it.
In the first step we should identify the reason.
2.1. Razão 1: Classe Não Está Descoberta
Quarkus has a simplified discovery. It might happen that the class is not part of the application index. For example, classes from the runtime module of a Quarkus extension are not indexed automatically.
Solução : Use o AdditionalBeanBuildItem
. Esse item de construção pode ser usado para especificar uma ou mais classes adicionais a serem analisadas durante a descoberta. As classes de bean adicionais são adicionadas de forma transparente ao índice da aplicação processado pelo contêiner.
It is not possible to conditionally enable/disable additional beans via the @IfBuildProfile , @UnlessBuildProfile , @IfBuildProperty and @UnlessBuildProperty annotations as described in cdi-reference and cdi-reference. Extensions should inspect the configuration or the current profile and only produce an AdditionalBeanBuildItem if really needed.
|
AdditionalBeanBuildItem
@BuildStep
AdditionalBeanBuildItem additionalBeans() {
return new AdditionalBeanBuildItem(SmallRyeHealthReporter.class, HealthServlet.class)); (1)
}
1 | AdditionalBeanBuildItem.Builder pode ser utilizado para casos de uso mais complexos. |
Bean classes added via AdditionalBeanBuildItem
are removable by default.
If the container considers them unused, they are just ignored.
However, you can use AdditionalBeanBuildItem.Builder.setUnremovable()
method to instruct the container to never remove bean classes registered via this build item.
See also Removing Unused Beans and Motivo 3: A Classe Foi Descoberta e Tem uma Anotação que Define o Bean, mas Foi Removida for more details.
Também é possível definir o escopo padrão por meio de AdditionalBeanBuildItem.Builder#setDefaultScope()
. O escopo padrão só é usado se não houver um escopo declarado na classe do bean.
Se não for especificado um escopo padrão, é utilizado o pseudo escopo @Dependent .
|
2.2. Motivo 2: A Classe É Descoberta mas Não Tem uma Anotação de Definição de Bean
No Quarkus, a aplicação é representada por um único bean archive com o modo de descoberta de bean anotado
. Portanto, as classes de bean que não têm uma anotação de definição de bean são ignoradas. As anotações de definição de bean são declaradas no nível da classe e incluem escopos, estereótipos e @Interceptor
.
Solução 1: Utilizar o AutoAddScopeBuildItem
. Este item de construção pode ser utilizado para adicionar um escopo a uma classe que satisfaça determinadas condições.
AutoAddScopeBuildItem
@BuildStep
AutoAddScopeBuildItem autoAddScope() {
return AutoAddScopeBuildItem.builder().containsAnnotations(SCHEDULED_NAME, SCHEDULES_NAME) (1)
.defaultScope(BuiltinScope.SINGLETON) (2)
.build();
}
1 | Encontrar todas as classes anotadas com @Scheduled . |
2 | Adicione @Singleton como escopo padrão. As classes já anotadas com um escopo são ignoradas automaticamente. |
Solução 2: Se precisar processar classes anotadas com uma anotação específica, é possível estender o conjunto de anotações que definem beans através do BeanDefiningAnnotationBuildItem
.
BeanDefiningAnnotationBuildItem
@BuildStep
BeanDefiningAnnotationBuildItem additionalBeanDefiningAnnotation() {
return new BeanDefiningAnnotationBuildItem(Annotations.GRAPHQL_API); (1)
}
1 | Adicionar org.eclipse.microprofile.graphql.GraphQLApi ao conjunto de anotações de definição de bean. |
Bean classes added via BeanDefiningAnnotationBuildItem
are not removable by default, i.e. the resulting beans must not be removed even if they are considered unused.
However, you can change the default behavior.
See also Removing Unused Beans and Motivo 3: A Classe Foi Descoberta e Tem uma Anotação que Define o Bean, mas Foi Removida for more details.
Também é possível especificar o escopo padrão. O escopo padrão é usado somente se não houver escopo declarado na classe do bean.
Se não for especificado um escopo padrão, é utilizado o pseudo escopo @Dependent .
|
2.3. Motivo 3: A Classe Foi Descoberta e Tem uma Anotação que Define o Bean, mas Foi Removida
The container attempts to remove all unused beans during the build by default.
This optimization allows for framework-level dead code elimination.
In few special cases, it’s not possible to correctly identify an unused bean.
In particular, Quarkus is not able to detect the usage of the CDI.current()
static method yet.
Extensions can eliminate possible false positives by producing an UnremovableBeanBuildItem
.
UnremovableBeanBuildItem
@BuildStep
UnremovableBeanBuildItem unremovableBeans() {
return UnremovableBeanBuildItem.targetWithAnnotation(STARTUP_NAME); (1)
}
1 | Tornar todas as classes anotadas com @Startup não removíveis. |
3. Caso de uso - Minha Anotação não É Reconhecida Como Um Qualificador ou Uma Ligação de Interceptador
É provável que a classe de anotação não faça parte do índice da aplicação. Por exemplo, as classes do módulo de tempo de execução de uma extensão do Quarkus não são indexadas automaticamente.
Solution: Use the AdditionalBeanBuildItem
as described in Razão 1: Classe Não Está Descoberta.
4. Caso de Utilização - Preciso Transformar Metadados de Anotações
Em alguns casos, é útil poder modificar os metadados da anotação. O Quarkus oferece uma alternativa eficiente para jakarta.enterprise.inject.spi.ProcessAnnotatedType
and jakarta.enterprise.inject.build.compatible.spi.Enhancement
. Com um AnnotationsTransformerBuildItem
, é possível substituir as anotações existentes nas classes de bean.
Tenha em mente que os transformadores de anotação devem ser produzidos antes do início da descoberta de bean. |
Por exemplo, você pode querer adicionar uma vinculação de interceptador a uma classe de bean específica. Você pode usar uma API conveniente, semelhante a um builder, para criar uma instância de transformador:
Exemplo de Builder
@BuildStep
AnnotationsTransformerBuildItem transform() {
return new AnnotationsTransformerBuildItem(AnnotationsTransformer.appliedToClass() (1)
.whenClass(c -> c.name().toString().equals("org.acme.Bar")) (2)
.thenTransform(t -> t.add(MyInterceptorBinding.class))); (3)
}
1 | O transformador só é aplicado às classes. |
2 | Aplicar a transformação apenas se o nome da classe for igual a org.acme.Bar . |
3 | Adicionar a anotação @MyInterceptorBinding . |
O exemplo acima pode ser reescrito com uma classe anônima:
AnnotationsTransformerBuildItem
@BuildStep
AnnotationsTransformerBuildItem transform() {
return new AnnotationsTransformerBuildItem(new AnnotationsTransformer() {
public boolean appliesTo(org.jboss.jandex.AnnotationTarget.Kind kind) {
return kind == org.jboss.jandex.AnnotationTarget.Kind.CLASS; (1)
}
public void transform(TransformationContext context) {
if (context.getTarget().asClass().name().toString().equals("org.acme.Bar")) {
context.transform().add(MyInterceptorBinding.class).done(); (2)
}
}
});
}
1 | O transformador só é aplicado às classes. |
2 | Se o nome da classe for igual a org.acme.Bar , adicione @MyInterceptorBinding . Não se esqueça de invocar Transformation#done() . |
As etapas de construção podem consultar as anotações transformadas para um determinado alvo de anotação através do TransformedAnnotationsBuildItem
.
TransformedAnnotationsBuildItem
@BuildStep
void queryAnnotations(TransformedAnnotationsBuildItem transformedAnnotations, BuildProducer<MyBuildItem> myBuildItem) {
ClassInfo myClazz = ...;
if (transformedAnnotations.getAnnotations(myClazz).isEmpty()) { (1)
myBuildItem.produce(new MyBuildItem());
}
}
1 | TransformedAnnotationsBuildItem.getAnnotations() devolverá um conjunto de anotações possivelmente transformadas. |
There are other build items specialized in transformation: Caso de Uso - Ligações de Interceptadores Adicionais and Caso de Uso - Transformação de Pontos de Injeção. |
4.1. Como Ativar o Log de Trace para Transformadores de Anotações
Pode definir o nível de TRACE
para a categoria io.quarkus.arc.processor
e tentar analisar a saída do log posteriormente.
application.properties
quarkus.log.category."io.quarkus.arc.processor".min-level=TRACE (1)
quarkus.log.category."io.quarkus.arc.processor".level=TRACE
1 | Também é necessário ajustar o nível mínimo de log para a categoria relevante. |
5. Caso de Uso - Inspecionar Beans, Observadores e Pontos de Injeção
5.1. Solução 1: BeanDiscoveryFinishedBuildItem
Os consumidores do BeanDiscoveryFinishedBuildItem
podem inspecionar facilmente todos os beans, observadores e pontos de injeção baseados em classe registrados na aplicação. No entanto, os beans e observadores sintéticos não são incluídos porque esse item de construção é produzido antes que os componentes sintéticos sejam registrados.
Além disso, o resolvedor de beans retornado por BeanDiscoveryFinishedBuildItem#getBeanResolver()
pode ser utilizado para aplicar as regras de resolução de tipagem segura, por exemplo, para descobrir se existe um bean que satisfaça uma determinada combinação de tipo e qualificadores necessários.
BeanDiscoveryFinishedBuildItem
@BuildStep
void doSomethingWithNamedBeans(BeanDiscoveryFinishedBuildItem beanDiscovery, BuildProducer<NamedBeansBuildItem> namedBeans) {
List<BeanInfo> namedBeans = beanDiscovery.beanStream().withName().collect(toList())); (1)
namedBeans.produce(new NamedBeansBuildItem(namedBeans));
}
1 | A lista resultante não conterá beans sintéticos @Named . |
5.2. Solução 2: SynthesisFinishedBuildItem
Os consumidores de SynthesisFinishedBuildItem
podem inspecionar facilmente todos os beans, observadores e pontos de injeção registrados na aplicação. Os beans e observadores sintéticos estão incluídos porque este item de construção é produzido depois dos componentes sintéticos serem registrados.
Além disso, o resolvedor de beans retornado por SynthesisFinishedBuildItem#getBeanResolver()
pode ser utilizado para aplicar as regras de resolução de tipagem segura, por exemplo, para descobrir se existe um bean que satisfaça uma determinada combinação de tipo e qualificadores necessários.
SynthesisFinishedBuildItem
@BuildStep
void doSomethingWithNamedBeans(SynthesisFinishedBuildItem synthesisFinished, BuildProducer<NamedBeansBuildItem> namedBeans) {
List<BeanInfo> namedBeans = synthesisFinished.beanStream().withName().collect(toList())); (1)
namedBeans.produce(new NamedBeansBuildItem(namedBeans));
}
1 | A lista resultante conterá beans sintéticos @Named . |
6. Caso de Uso - A Necessidade de Beans Sintéticos
Às vezes, é prático poder registrar um bean sintético . Os atributos de bean de um bean sintético não são derivados de uma classe, método ou campo Java. Em vez disso, todos os atributos são definidos por uma extensão. No CDI regular, isso poderia ser feito usando os métodos AfterBeanDiscovery.addBean()
e SyntheticComponents.addBean()
.
Solução: Se for necessário registrar um bean sintético, utilize o SyntheticBeanBuildItem
.
SyntheticBeanBuildItem
@BuildStep
SyntheticBeanBuildItem syntheticBean() {
return SyntheticBeanBuildItem.configure(String.class)
.qualifiers(AnnotationInstance.builder(MyQualifier.class).build())
.creator(mc -> mc.returnValue(mc.load("foo"))) (1)
.done();
}
1 | Gerar o bytecode da implementação jakarta.enterprise.context.spi.Contextual#create(CreationalContext<T>) . |
A saída de um configurador de bean é gravada como bytecode. Portanto, há algumas limitações na forma como uma instância de bean sintético é criada em tempo de execução. Você pode:
-
Gerar o bytecode do método
Contextual#create(CreationalContext<T>)
diretamente através doExtendedBeanConfigurator.creator(Consumer<MethodCreator>)
. -
Passar uma subclasse de
io.quarkus.arc.BeanCreator
através doExtendedBeanConfigurator#creator(Class<? extends BeanCreator<U>>)
, e possivelmente especificar alguns parâmetros de tempo de construção através doExtendedBeanConfigurator#param()
e pontos de injeção sintéticos através doExtendedBeanConfigurator#addInjectionPoint()
. -
Produce the runtime instance through a proxy returned from a
@Recorder
method and set it viaExtendedBeanConfigurator#runtimeValue(RuntimeValue<?>)
,ExtendedBeanConfigurator#runtimeProxy(Object)
,ExtendedBeanConfigurator#supplier(Supplier<?>)
orExtendedBeanConfigurator#createWith(Function<SyntheticCreationalContext<?>, <?>)
.
SyntheticBeanBuildItem
@BuildStep
@Record(STATIC_INIT) (1)
SyntheticBeanBuildItem syntheticBean(TestRecorder recorder) {
return SyntheticBeanBuildItem.configure(Foo.class).scope(Singleton.class)
.runtimeValue(recorder.createFoo()) (2)
.done();
}
1 | Por padrão, um bean sintético é inicializado durante STATIC_INIT . |
2 | A instância do bean é fornecida por um valor retornado por um método de gravação. |
It is possible to mark a synthetic bean to be initialized during RUNTIME_INIT
.
See the Three Phases of Bootstrap and Quarkus Philosophy for more information about the difference between STATIC_INIT
and RUNTIME_INIT
.
RUNTIME_INIT
SyntheticBeanBuildItem
@BuildStep
@Record(RUNTIME_INIT) (1)
SyntheticBeanBuildItem syntheticBean(TestRecorder recorder) {
return SyntheticBeanBuildItem.configure(Foo.class).scope(Singleton.class)
.setRuntimeInit() (2)
.runtimeValue(recorder.createFoo())
.done();
}
1 | O gravador deve ser executado na fase ExecutionTime.RUNTIME_INIT . |
2 | A instância do bean é inicializada durante RUNTIME_INIT . |
Os beans sintéticos inicializados durante
|
Também é possível utilizar o BeanRegistrationPhaseBuildItem para registrar um bean sintético. No entanto, recomendamos aos autores de extensões que utilizem SyntheticBeanBuildItem , que é mais idiomático para o Quarkus.
|
6.1. Pontos de Injeção Sintéticos
A synthetic bean may register a synthetic injection point via the ExtendedBeanConfigurator#addInjectionPoint()
method.
This injection point is validated at build time and considered when detecting unused beans.
The injected reference is accessible through the SyntheticCreationalContext#getInjectedReference()
methods at runtime.
import org.jboss.jandex.ClassType;
import org.jboss.jandex.DotName;
@BuildStep
@Record(RUNTIME_INIT) (1)
SyntheticBeanBuildItem syntheticBean(TestRecorder recorder) {
return SyntheticBeanBuildItem.configure(Foo.class)
.scope(Singleton.class)
.addInjectionPoint(ClassType.create(DotName.createSimple(Bar.class))) (2)
.createWith(recorder.createFoo()) (3)
.done();
}
1 | A instância do bean é inicializada durante RUNTIME_INIT . |
2 | Foi adicionado um ponto de injeção sintético com o tipo obrigatório Bar ; este é equivalente a @Inject Bar . |
3 | A instância do bean é criada com uma função retornada de um método de gravação. |
@Recorder
public class TestRecorder {
public Function<SyntheticCreationalContext<Foo>, Foo> createFoo() {
return (context) -> {
return new Foo(context.getInjectedReference(Bar.class)); (1)
};
}
}
1 | Passa uma referência contextual de Bar ao construtor de Foo . |
7. Caso de Uso - Observadores Sintéticos
Similar to synthetic beans, the attributes of a synthetic observer method are not derived from a Java method. Instead, all the attributes are defined by an extension.
Solução: Se for necessário registrar um observador sintético, utilize o ObserverRegistrationPhaseBuildItem
.
Uma etapa de construção que consome o ObserverRegistrationPhaseBuildItem deve sempre produzir um ObserverConfiguratorBuildItem ou, pelo menos, injetar um BuildProducer para este item de construção, caso contrário, pode ser ignorado ou processado no momento errado (por exemplo, após a fase de bootstrap correta do CDI).
|
ObserverRegistrationPhaseBuildItem
@BuildStep
void syntheticObserver(ObserverRegistrationPhaseBuildItem observerRegistrationPhase,
BuildProducer<MyBuildItem> myBuildItem,
BuildProducer<ObserverConfiguratorBuildItem> observerConfigurationRegistry) {
observerConfigurationRegistry.produce(new ObserverConfiguratorBuildItem(observerRegistrationPhase.getContext()
.configure()
.beanClass(DotName.createSimple(MyBuildStep.class.getName()))
.observedType(String.class)
.notify(mc -> {
// do some gizmo bytecode generation...
})));
myBuildItem.produce(new MyBuildItem());
}
A saída de um ObserverConfigurator
é gravada como bytecode. Portanto, há algumas limitações na forma como um observador sintético é chamado no tempo de execução. Atualmente, você deve gerar o bytecode do corpo do método diretamente.
8. Caso de Uso - Tenho uma Classe de Bean Gerada
Sem problemas. Você pode gerar o bytecode de uma classe de bean manualmente e, então, tudo o que precisa fazer é produzir um GeneratedBeanBuildItem
em vez de GeneratedClassBuildItem
.
GeneratedBeanBuildItem
@BuildStep
void generatedBean(BuildProducer<GeneratedBeanBuildItem> generatedBeans) {
ClassOutput beansClassOutput = new GeneratedBeanGizmoAdaptor(generatedBeans); (1)
ClassCreator beanClassCreator = ClassCreator.builder().classOutput(beansClassOutput)
.className("org.acme.MyBean")
.build();
beanClassCreator.addAnnotation(Singleton.class);
beanClassCreator.close(); (2)
}
1 | io.quarkus.arc.deployment.GeneratedBeanGizmoAdaptor facilita a produção de GeneratedBeanBuildItem s a partir de constructos Gizmo. |
2 | A classe de bean resultante é algo como public class @Singleton MyBean { } . |
9. Caso de Uso - Preciso Validar a Implantação
Por vezes, as extensões precisam de inspecionar os beans, os observadores e os pontos de injeção e, em seguida, efetuar validações adicionais e falhar a construção se algo estiver errado.
Solução: Se uma extensão precisa validar a implantação, deve utilizar o ValidationPhaseBuildItem
.
Uma etapa de construção que consome o ValidationPhaseBuildItem deve sempre produzir um ValidationErrorBuildItem ou, pelo menos, injetar um BuildProducer para este item de construção, caso contrário, pode ser ignorado ou processado no momento errado (por exemplo, após a fase de bootstrap correta do CDI).
|
@BuildStep
void validate(ValidationPhaseBuildItem validationPhase,
BuildProducer<MyBuildItem> myBuildItem,
BuildProducer<ValidationErrorBuildItem> errors) {
if (someCondition) {
errors.produce(new ValidationErrorBuildItem(new IllegalStateException()));
myBuildItem.produce(new MyBuildItem());
}
}
Você pode filtrar facilmente todos os beans registrados através do prático BeanStream retornado pelo método ValidationPhaseBuildItem.getContext().beans() .
|
10. Caso de Uso - Registrar um Contexto CDI Personalizado
Às vezes, as extensões precisam estender o conjunto de contextos CDI integrados.
Solução: Se for necessário registrar um contexto personalizado, utilize o ContextRegistrationPhaseBuildItem
.
Uma etapa de construção que consome o ContextRegistrationPhaseBuildItem deve sempre produzir um ContextConfiguratorBuildItem ou, pelo menos, injetar um BuildProducer para este item de construção, caso contrário, pode ser ignorado ou processado no momento errado (por exemplo, após a fase de bootstrap correta do CDI).
|
Exemplo de ContextRegistrationPhaseBuildItem
@BuildStep
ContextConfiguratorBuildItem registerContext(ContextRegistrationPhaseBuildItem phase) {
return new ContextConfiguratorBuildItem(phase.getContext().configure(TransactionScoped.class).normal().contextClass(TransactionContext.class));
}
Além disso, cada extensão que registra um contexto CDI personalizado através do ContextRegistrationPhaseBuildItem
deve também produzir o CustomScopeBuildItem
de modo a contribuir com o nome da anotação de escopo personalizado para o conjunto de anotações que definem o bean.
Exemplo de CustomScopeBuildItem
@BuildStep
CustomScopeBuildItem customScope() {
return new CustomScopeBuildItem(DotName.createSimple(TransactionScoped.class.getName()));
}
11. Caso de Uso - Ligações de Interceptadores Adicionais
Em casos raros, pode ser útil registrar programaticamente uma anotação existente que não esteja anotada com @jakarta.interceptor.InterceptorBinding
como uma vinculação de interceptador. Isso é semelhante ao que o CDI consegue fazer por meio de BeforeBeanDiscovery#addInterceptorBinding()
. Vamos usar o InterceptorBindingRegistrarBuildItem
para fazer isso.
InterceptorBindingRegistrarBuildItem
@BuildStep
InterceptorBindingRegistrarBuildItem addInterceptorBindings() {
return new InterceptorBindingRegistrarBuildItem(new InterceptorBindingRegistrar() {
@Override
public List<InterceptorBinding> getAdditionalBindings() {
return List.of(InterceptorBinding.of(NotAnInterceptorBinding.class));
}
});
}
12. Caso de Uso - Qualificadores Adicionais
Às vezes, pode ser útil registrar uma anotação existente que não esteja anotada com @jakarta.inject.Qualifier
como um qualificador CDI. Isso é semelhante ao que a CDI faz por meio do BeforeBeanDiscovery#addQualifier()
. Vamos usar o QualifierRegistrarBuildItem
para fazer isso.
QualifierRegistrarBuildItem
@BuildStep
QualifierRegistrarBuildItem addQualifiers() {
return new QualifierRegistrarBuildItem(new QualifierRegistrar() {
@Override
public Map<DotName, Set<String>> getAdditionalQualifiers() {
return Collections.singletonMap(DotName.createSimple(NotAQualifier.class.getName()),
Collections.emptySet());
}
});
}
13. Caso de Uso - Estereótipos Adicionais
Às vezes, é útil registrar uma anotação existente que não esteja anotada com @jakarta.enterprise.inject.Stereotype
como um estereótipo CDI. Isso é semelhante ao que a CDI faz por meio do BeforeBeanDiscovery#addStereotype()
. Vamos usar o StereotypeRegistrarBuildItem
para fazer isso.
StereotypeRegistrarBuildItem
@BuildStep
StereotypeRegistrarBuildItem addStereotypes() {
return new StereotypeRegistrarBuildItem(new StereotypeRegistrar() {
@Override
public Set<DotName> getAdditionalStereotypes() {
return Collections.singleton(DotName.createSimple(NotAStereotype.class.getName()));
}
});
}
If the newly registered stereotype annotation doesn’t have the appropriate meta-annotations, such as scope or interceptor bindings, use an annotation transformation to add them.
14. Caso de Uso - Transformação de Pontos de Injeção
De vez em quando, é útil poder alterar os qualificadores de um ponto de injeção de forma programática. Você pode fazer exatamente isso com InjectionPointTransformerBuildItem
. O exemplo a seguir mostra como aplicar a transformação a pontos de injeção com o tipo Foo
que contém o qualificador MyQualifier
:
InjectionPointTransformerBuildItem
@BuildStep
InjectionPointTransformerBuildItem transformer() {
return new InjectionPointTransformerBuildItem(new InjectionPointsTransformer() {
public boolean appliesTo(Type requiredType) {
return requiredType.name().equals(DotName.createSimple(Foo.class.getName()));
}
public void transform(TransformationContext context) {
if (context.getQualifiers().stream()
.anyMatch(a -> a.name().equals(DotName.createSimple(MyQualifier.class.getName())))) {
context.transform()
.removeAll()
.add(DotName.createSimple(MyOtherQualifier.class.getName()))
.done();
}
}
});
}
In theory, you can use an AnnotationsTransformer to achieve the same goal. However, there are few differences that make InjectionPointsTransformer more suitable for this particular task: (1) annotation transformers are applied to all classes during bean discovery, whereas InjectionPointsTransformer is only applied to discovered injection points after bean discovery; (2) with InjectionPointsTransformer you don’t need to handle various types of injection points (field, parameters of initializer methods, etc.).
|
15. Caso de Uso - Anotações e Injeção de Recursos
O ResourceAnnotationBuildItem
pode ser usado para especificar anotações de recursos que possibilitam a resolução de pontos de injeção não CDI, como os recursos do Jakarta EE. Um integrador também deve fornecer uma implementação correspondente do provedor de serviços io.quarkus.arc.ResourceReferenceProvider
.
ResourceAnnotationBuildItem
@BuildStep
void setupResourceInjection(BuildProducer<ResourceAnnotationBuildItem> resourceAnnotations, BuildProducer<GeneratedResourceBuildItem> resources) {
resources.produce(new GeneratedResourceBuildItem("META-INF/services/io.quarkus.arc.ResourceReferenceProvider",
MyResourceReferenceProvider.class.getName().getBytes()));
resourceAnnotations.produce(new ResourceAnnotationBuildItem(DotName.createSimple(MyAnnotation.class.getName())));
}
16. Metadados de Tempo de Construção Disponíveis
Qualquer uma das extensões acima que opere com BuildExtension.BuildContext
pode aproveitar determinados metadados de tempo de construção gerados durante a construção. As chaves integradas, localizadas em io.quarkus.arc.processor.BuildExtension.Key
são:
- ANNOTATION_STORE
-
Contém um
AnnotationStore
que mantém informações sobre todas as anotaçõesAnnotationTarget
após a aplicação de transformadores de anotações - INJECTION_POINTS
-
Collection<InjectionPointInfo>
contendo todos os pontos de injeção - BEANS
-
Collection<BeanInfo>
contendo todos os beans - REMOVED_BEANS
-
Collection<BeanInfo>
containing all the removed beans; see Removing unused beans for more information - OBSERVERS
-
Collection<ObserverInfo>
contendo todos os observadores - SCOPES
-
Collection<ScopeInfo>
contendo todos os escopos, incluindo os personalizados - QUALIFIERS
-
Map<DotName, ClassInfo>
contendo todos os qualificadores - INTERCEPTOR_BINDINGS
-
Map<DotName, ClassInfo>
contendo todas os vínculos de interceptadores - STEREOTYPES
-
Map<DotName, StereotypeInfo>
contendo todos os estereótipos
Para obtê-los, basta consultar o objeto de contexto da extensão para uma determinada chave. Observe que esses metadados são disponibilizados à medida que a construção avança, o que significa que as extensões só podem aproveitar os metadados que foram criados antes das extensões serem invocadas. Se sua extensão tentar recuperar metadados que ainda não foram produzidos, será retornado null
. Aqui está um resumo de quais extensões podem acessar quais metadados:
- AnnotationsTransformer
-
Não deve depender de quaisquer metadados, uma vez que pode ser utilizado em qualquer momento de qualquer fase do bootstrap
- ContextRegistrar
-
Tem acesso a
ANNOTATION_STORE
,QUALIFIERS
,INTERCEPTOR_BINDINGS
,STEREOTYPES
- InjectionPointsTransformer
-
Tem acesso a
ANNOTATION_STORE
,QUALIFIERS
,INTERCEPTOR_BINDINGS
,STEREOTYPES
- ObserverTransformer
-
Tem acesso a
ANNOTATION_STORE
,QUALIFIERS
,INTERCEPTOR_BINDINGS
,STEREOTYPES
- BeanRegistrar
-
Tem acesso a
ANNOTATION_STORE
,QUALIFIERS
,INTERCEPTOR_BINDINGS
,STEREOTYPES
,BEANS
(apenas beans baseados em classes),OBSERVERS
(apenas observadores baseados em classes),INJECTION_POINTS
- ObserverRegistrar
-
Tem acesso a
ANNOTATION_STORE
,QUALIFIERS
,INTERCEPTOR_BINDINGS
,STEREOTYPES
,BEANS
,OBSERVERS
(apenas observadores baseados em classes),INJECTION_POINTS
- BeanDeploymentValidator
-
Tem acesso a todos os metadados de construção