com.adobe.acs.commons.mcp.form.DialogProviderAnnotationProcessor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of acs-aem-commons-bundle Show documentation
Show all versions of acs-aem-commons-bundle Show documentation
Main ACS AEM Commons OSGi Bundle. Includes commons utilities.
The newest version!
/*
* ACS AEM Commons
*
* Copyright (C) 2013 - 2023 Adobe
*
* 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.adobe.acs.commons.mcp.form;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Collections;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.annotation.processing.AbstractProcessor;
import javax.annotation.processing.RoundEnvironment;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.TypeElement;
import javax.tools.JavaFileObject;
import org.apache.commons.lang3.StringUtils;
import org.apache.sling.models.annotations.Model;
/**
* Processes the DialogProvider annotation, producing a corresponding OSGi
* service to provide the generated sling dialog resources. This annotation
* processor will skip any classes which do not identify their corresponding
* sling model either as part of the model annotation or by a property or getter
* method.
*
* This annotation processor needs to be registered explicitly via the processor option of javac
* in order to prevent it from being automatically active whenever this bundle is on the classpath.
* For usage with Maven this can be achieved with the following code:
*
{@code
*
* maven-compiler-plugin
*
*
* com.adobe.acs.commons.mcp.form.DialogProviderAnnotationProcessor
*
*
*
* }
*
* @see maven-compiler-plugin
* @see javax.annotation.processing.Processor
*/
public class DialogProviderAnnotationProcessor extends AbstractProcessor {
private static final Logger LOG = Logger.getLogger(DialogProviderAnnotationProcessor.class.getName());
@Override
public boolean process(Set extends TypeElement> annotations, RoundEnvironment roundEnv) {
for (Element annotatedElement : roundEnv.getElementsAnnotatedWith(DialogProvider.class)) {
try {
processDialogProviderAnnotation(annotatedElement);
} catch (IOException ex) {
LOG.log(Level.SEVERE, null, ex);
return false;
}
}
return true;
}
@Override
public Set getSupportedAnnotationTypes() {
return Collections.singleton(DialogProvider.class.getCanonicalName());
}
@Override
public SourceVersion getSupportedSourceVersion() {
return SourceVersion.latestSupported();
}
private void processDialogProviderAnnotation(Element element) throws IOException {
TypeElement t = (TypeElement) element;
String className = t.getQualifiedName().toString();
String serviceClassName = getServiceClassName(className);
if (providesResourceType(t)) {
if (LOG.isLoggable(Level.INFO)) {
LOG.log(Level.INFO, String.format("Generated resource provider service for class %s => %s", className, serviceClassName));
}
JavaFileObject builderFile = processingEnv.getFiler().createSourceFile(serviceClassName);
writeServiceStub(builderFile, serviceClassName, className);
} else {
if (LOG.isLoggable(Level.WARNING)) {
LOG.log(Level.WARNING, String.format("Class %s declares or inherits the DialogProvider annotation but does not declare a resource type -- no resource provider generated.", className));
}
}
}
private void writeServiceStub(JavaFileObject builderFile, String serviceClass, String targetClass) throws IOException {
try (PrintWriter out = new PrintWriter(builderFile.openWriter())) {
String packageName = StringUtils.substringBeforeLast(serviceClass, ".");
String className = StringUtils.substringAfterLast(serviceClass, ".");
String osgiService = DialogResourceProvider.class.getCanonicalName();
out.println(String.format("package %s;", packageName));
out.println();
out.println("import javax.annotation.Generated;");
out.println("import org.osgi.annotation.versioning.ConsumerType;");
out.println("import org.osgi.framework.BundleContext;");
out.println("import org.osgi.service.component.annotations.*;");
out.println();
out.println("@Generated(\"Created by the ACS Commons DialogProviderAnnotationProcessor\")");
out.println("@ConsumerType");
out.printf("@Component(service = %s.class, immediate = true)%n", osgiService);
out.printf("public class %s implements %s {%n", className, osgiService);
out.println();
out.println(" @Override");
out.println(" public Class getTargetClass() {");
out.printf(" return %s.class;%n", targetClass);
out.println(" }");
out.println();
out.println(" @Activate");
out.println(" public void activate(BundleContext context) throws InstantiationException, IllegalAccessException, ReflectiveOperationException {");
out.println(" this.doActivate(context);");
out.println(" }");
out.println();
out.println(" @Deactivate");
out.println(" public void deactivate(BundleContext context) {");
out.println(" this.doDeactivate();");
out.println(" }");
out.println("}");
out.flush();
}
}
private boolean providesResourceType(TypeElement t) {
Model model = t.getAnnotation(Model.class);
if (model != null && model.resourceType() != null && model.resourceType().length > 0) {
return true;
} else {
return t.getEnclosedElements().stream().anyMatch(this::elementProvidesResourceType);
}
}
private boolean elementProvidesResourceType(Element t) {
switch (t.getKind()) {
case LOCAL_VARIABLE:
case FIELD:
return t.getSimpleName().contentEquals("resourceType");
case METHOD:
return t.getSimpleName().contentEquals("getResourceType");
default:
return false;
}
}
private static String getServiceClassName(String modelClass) {
String[] parts = StringUtils.split(modelClass, '.');
StringBuilder name = new StringBuilder();
String separator = ".";
for (String part : parts) {
char firstChar = part.charAt(0);
String newSeparator = separator;
if (firstChar >= 'A' && firstChar <= 'Z' && separator.equals(".")) {
newSeparator = "$";
name.append(".impl");
}
if (name.length() > 0) {
name.append(separator);
}
name.append(part);
separator = newSeparator;
}
return name + "_dialogResourceProvider";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy