com.igormaznitsa.jbbp.plugin.mvn.JBBPGenerateMojo Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jbbp-maven-plugin Show documentation
Show all versions of jbbp-maven-plugin Show documentation
Generator of sources from JBBP scripts
The newest version!
/*
* Copyright 2017 Igor Maznitsa.
*
* 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.igormaznitsa.jbbp.plugin.mvn;
import static com.igormaznitsa.jbbp.plugin.common.utils.CommonUtils.ensureEncodingName;
import static com.igormaznitsa.jbbp.utils.JBBPUtils.ARRAY_STRING_EMPTY;
import com.igormaznitsa.jbbp.JBBPCustomFieldTypeProcessor;
import com.igormaznitsa.jbbp.compiler.JBBPNamedFieldInfo;
import com.igormaznitsa.jbbp.compiler.tokenizer.JBBPFieldTypeParameterContainer;
import com.igormaznitsa.jbbp.io.JBBPArraySizeLimiter;
import com.igormaznitsa.jbbp.io.JBBPBitInputStream;
import com.igormaznitsa.jbbp.io.JBBPBitOrder;
import com.igormaznitsa.jbbp.mapper.Bin;
import com.igormaznitsa.jbbp.model.JBBPAbstractField;
import com.igormaznitsa.jbbp.plugin.common.converters.JBBPScriptTranslator;
import com.igormaznitsa.jbbp.plugin.common.converters.ParserFlags;
import com.igormaznitsa.jbbp.plugin.common.converters.Target;
import com.igormaznitsa.meta.annotation.MustNotContainNull;
import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.apache.commons.io.FileUtils;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugins.annotations.LifecyclePhase;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter;
/**
* The Mojo looks for all JBBP scripts in source and generate sources.
*
* @author Igor Maznitsa
*/
@Mojo(name = "generate", defaultPhase = LifecyclePhase.GENERATE_SOURCES, threadSafe = true)
public class JBBPGenerateMojo extends AbstractJBBPMojo {
/**
* Interfaces for section 'implements' in generated classes.
*/
@Parameter(alias = "interfaces")
private final Set interfaces = new HashSet();
/**
* List of parser flags.
*
* - SKIP_REMAINING_FIELDS_IF_EOF
*
*/
@Parameter(alias = "parserFlags")
private final Set parserFlags = new HashSet();
/**
* List of names of allowed custom value types.
*/
@Parameter(alias = "customTypes")
private final Set customTypes = new HashSet();
/**
* Interfaces for structures, structure will be implementing the interface and getter will return interface instead of structure type.
*/
@Parameter(alias = "mapStructToInterfaces")
private final Map mapStructToInterfaces = new HashMap();
/**
* Superclasses for structures, structure will be extending mapped superclass.
*
* @since 1.4.0
*/
@Parameter(alias = "mapStructToSuperclasses")
private final Map mapStructToSuperclasses = new HashMap();
/**
* Do not make generated inner classes as static ones.
*
* @since 1.4.0
*/
@Parameter(alias = "doInnerClassesNonStatic")
private boolean doInnerClassesNonStatic;
/**
* Specify output file encoding; defaults to source encoding.
*/
@Parameter(alias = "outputEncoding", defaultValue = "${project.build.sourceEncoding}")
private String outputEncoding;
/**
* Specify grammar file encoding; defaults to source encoding.
*/
@Parameter(alias = "inputEncoding", defaultValue = "${project.build.sourceEncoding}")
private String inputEncoding;
/**
* File contains text of cap comment for each generated class file. The Cap
* text will be placed before package name and usually it can be used to
* provide license information.
*/
@Parameter(alias = "headCommentFile")
private File headCommentFile;
/**
* Plain text of cap comment for each generated class file. The Cap text will
* be placed before package name and usually it can be used to provide license
* information.
*/
@Parameter(alias = "headComment")
private String headComment;
/**
* File contains text of custom section to be added into class body.
*/
@Parameter(alias = "customTextFile")
private File customTextFile;
/**
* Plain text of custom section to be added into class body.
*/
@Parameter(alias = "customText")
private String customText;
/**
* Disable generate fields in result class.
*/
@Parameter(alias = "disableGenerateFields", defaultValue = "false")
private boolean disableGenerateFields;
/**
* Generate getters and setters for class fields (class fields will be private
* ones).
*/
@Parameter(alias = "addGettersSetters")
private boolean addGettersSetters;
/**
* Super class for generated classes.
*/
@Parameter(alias = "superClass")
private String superClass;
/**
* Force abstract modifier for generated classes even if they don't have abstract methods.
*/
@Parameter(alias = "doAbstract")
private boolean doAbstract;
/**
* Add generated output folder to list of source folders.
*
* @since 1.4.0
*/
@Parameter(alias = "addToSourceFolders")
private boolean addToSourceFolders;
/**
* Add generated output folder to list of test source folders.
*
* @since 1.4.0
*/
@Parameter(alias = "addToTestSourceFolders")
private boolean addToTestSourceFolders;
/**
* Turn on generate of newInstance methods inside generated classes.
*
* @since 2.0.0
*/
@Parameter(alias = "addNewInstanceMethods", defaultValue = "false")
private boolean addNewInstanceMethods;
/**
* Add Bin annotations
*
* @see Bin
* @since 2.0.0
*/
@Parameter(alias = "addBinAnnotations", defaultValue = "false")
private boolean addBinAnnotations;
public boolean isAddToSourceFolders() {
return this.addToSourceFolders;
}
public void setAddToSourceFolders(final boolean value) {
this.addToSourceFolders = value;
}
public boolean isAddToTestSourceFolders() {
return this.addToTestSourceFolders;
}
public void setAddToTestSourceFolders(final boolean value) {
this.addToTestSourceFolders = value;
}
@Nullable
public String getSuperClass() {
return this.superClass;
}
public boolean isDoAbstract() {
return this.doAbstract;
}
public void setDoAbstract(final boolean value) {
this.doAbstract = value;
}
@Nonnull
public Map getMapStructToInterfaces() {
return this.mapStructToInterfaces;
}
@Nonnull
public Map getMapStructToSuperclasses() {
return this.mapStructToSuperclasses;
}
@Nonnull
@MustNotContainNull
public Set getInterfaces() {
return this.interfaces;
}
@Nonnull
@MustNotContainNull
public Set getCustomTypes() {
return this.customTypes;
}
public boolean getAddGettersSetters() {
return this.addGettersSetters;
}
@Nullable
public File getCustomTextFile() {
return this.customTextFile;
}
@Nullable
public String getCustomText() {
return this.customText;
}
@Nullable
public File getHeadCommentFile() {
return this.headCommentFile;
}
@Nullable
public String getHeadComment() {
return this.headComment;
}
@Nullable
public String getInputEncoding() {
return this.inputEncoding;
}
@Nullable
public String getOutputEncoding() {
return this.outputEncoding;
}
@MustNotContainNull
@Nonnull
public Set getParserFlags() {
return this.parserFlags;
}
public boolean isDisableGenerateFields() {
return this.disableGenerateFields;
}
public void setDisableGenerateFields(final boolean value) {
this.disableGenerateFields = value;
}
public boolean isDoInnerClassesNonStatic() {
return this.doInnerClassesNonStatic;
}
public void setDoInnerClassesNonStatic(final boolean value) {
this.doInnerClassesNonStatic = value;
}
@Nullable
private String makeCapText(@Nonnull final String inEncoding) throws IOException {
String result = null;
if (this.headComment != null) {
result = this.headComment;
} else if (this.headCommentFile != null) {
getLog().debug("Provided CAP comment file: " + this.headCommentFile.getPath());
result = FileUtils.readFileToString(this.headCommentFile, inEncoding);
}
return result;
}
@Nullable
private String makeCustomText(@Nonnull final String inEncoding) throws IOException {
String result = null;
if (this.customText != null) {
result = this.customText;
} else if (this.customTextFile != null) {
getLog().debug("Provided custom text file: " + this.customTextFile.getPath());
result = FileUtils.readFileToString(this.customTextFile, inEncoding);
}
return result;
}
@Override
protected void executeMojo() throws MojoExecutionException, MojoFailureException {
final String inEncoding = ensureEncodingName(this.inputEncoding);
final String outEncoding = ensureEncodingName(this.outputEncoding);
getLog().debug("Encoding In: " + inEncoding);
getLog().debug("Encoding Out: " + outEncoding);
final String capText;
try {
capText = makeCapText(inEncoding);
} catch (IOException ex) {
throw new MojoExecutionException("Can't read cap comment file", ex);
}
final String customTextForClass;
try {
customTextForClass = makeCustomText(inEncoding);
} catch (IOException ex) {
throw new MojoExecutionException("Can't read custom text file", ex);
}
if (this.includes.isEmpty()) {
this.includes.add("**/*.jbbp");
}
final Set normalizedCustomTypeNames = new HashSet();
for (final String s : this.customTypes) {
final String trimmed = s.trim();
final String normalized = trimmed.toLowerCase(Locale.ENGLISH);
if (!normalized.equals(trimmed)) {
getLog().warn(String
.format("Custom type name '%s' in JBBP normal form is '%s' ", trimmed, normalized));
}
normalizedCustomTypeNames.add(normalized);
}
getLog().debug("Defined normalized custom types : " + normalizedCustomTypeNames);
final String[] customTypesArray = normalizedCustomTypeNames.toArray(ARRAY_STRING_EMPTY);
final JBBPCustomFieldTypeProcessor customFieldProcessor = new JBBPCustomFieldTypeProcessor() {
@Override
@Nonnull
@MustNotContainNull
public String[] getCustomFieldTypes() {
return customTypesArray;
}
@Override
public boolean isAllowed(@Nonnull final JBBPFieldTypeParameterContainer fieldType,
@Nullable final String fieldName, final int extraData,
final boolean isArray) {
final boolean result = normalizedCustomTypeNames.contains(fieldType.getTypeName());
if (!result) {
getLog().warn("Detected not allowed custom type name : " + fieldType.getTypeName());
}
return result;
}
@Override
@Nonnull
public JBBPAbstractField readCustomFieldType(@Nonnull final JBBPBitInputStream in,
@Nonnull final JBBPBitOrder bitOrder,
final int parserFlags, @Nonnull
final JBBPFieldTypeParameterContainer customTypeFieldInfo,
@Nullable final JBBPNamedFieldInfo fieldName,
final int extraData,
final boolean readWholeStream,
final int arrayLength,
@Nonnull
final JBBPArraySizeLimiter arraySizeLimiter)
throws IOException {
throw new Error("Must not be called");
}
};
final Set foundJBBPScripts = findSources(this.source);
if (checkSetNonEmptyWithLogging(foundJBBPScripts)) {
final Target theTarget = findTarget();
final JBBPScriptTranslator.Parameters parameters = new JBBPScriptTranslator.Parameters();
parameters
.setPackageName(this.packageName)
.setParserFlags(ParserFlags.makeFromSet(this.parserFlags))
.setOutputDir(this.output)
.setHeadComment(capText)
.setCustomText(customTextForClass)
.setEncodingIn(inEncoding)
.setEncodingOut(outEncoding)
.setCustomFieldTypeProcessor(customFieldProcessor)
.setSuperClass(this.getSuperClass())
.setClassImplements(this.getInterfaces())
.setSubClassInterfaces(this.getMapStructToInterfaces())
.setDoInternalClassesNonStatic(this.isDoInnerClassesNonStatic())
.setSubClassSuperclasses(this.getMapStructToSuperclasses())
.setAddGettersSetters(this.getAddGettersSetters())
.setDoAbstract(this.isDoAbstract())
.setDisableGenerateFields(this.isDisableGenerateFields())
.setAddBinAnnotations(this.isAddBinAnnotations())
.setAddNewInstanceMethods(this.isAddNewInstanceMethods());
for (final File aScript : foundJBBPScripts) {
parameters.setScriptFile(aScript).assertAllOk();
getLog().debug("Processing JBBP script file : " + aScript);
try {
final Set files = theTarget.getTranslator().translate(parameters, false);
getLog().debug("Converted " + aScript + " into " + files);
for (final File f : files) {
logInfo(String
.format("JBBP script '%s' has been converted into '%s'", aScript.getName(),
f.getName()), false);
}
} catch (IOException ex) {
throw new MojoExecutionException(
"Error during JBBP script translation : " + aScript.getAbsolutePath(), ex);
}
}
if (this.isAddToSourceFolders()) {
getLog().info("Add folder to compile source root: " + this.getOutput().getAbsolutePath());
this.project.addCompileSourceRoot(this.getOutput().getAbsolutePath());
}
if (this.isAddToTestSourceFolders()) {
getLog()
.info("Add folder to test compile source root: " + this.getOutput().getAbsolutePath());
this.project.addTestCompileSourceRoot(this.getOutput().getAbsolutePath());
}
}
registerSourceRoot(this.output);
}
public boolean isAddNewInstanceMethods() {
return this.addNewInstanceMethods;
}
public void setAddNewInstanceMethods(final boolean addNewInstanceMethods) {
this.addNewInstanceMethods = addNewInstanceMethods;
}
public boolean isAddBinAnnotations() {
return addBinAnnotations;
}
public void setAddBinAnnotations(boolean addBinAnnotations) {
this.addBinAnnotations = addBinAnnotations;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy