top.marchand.maven.gaulois.compiler.GCMojo Maven / Gradle / Ivy
/**
* Copyright © 2017, Christophe Marchand
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of the nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package top.marchand.maven.gaulois.compiler;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.nio.channels.Channels;
import java.nio.channels.FileChannel;
import java.nio.channels.ReadableByteChannel;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParserFactory;
import javax.xml.transform.Source;
import javax.xml.transform.SourceLocator;
import javax.xml.transform.TransformerException;
import javax.xml.transform.sax.SAXSource;
import javax.xml.transform.stream.StreamSource;
import net.sf.saxon.ma.map.MapItem;
import net.sf.saxon.s9api.Axis;
import net.sf.saxon.s9api.MessageListener;
import net.sf.saxon.s9api.QName;
import net.sf.saxon.s9api.SaxonApiException;
import net.sf.saxon.s9api.Serializer;
import net.sf.saxon.s9api.XdmAtomicValue;
import net.sf.saxon.s9api.XdmDestination;
import net.sf.saxon.s9api.XdmMap;
import net.sf.saxon.s9api.XdmNode;
import net.sf.saxon.s9api.XdmSequenceIterator;
import net.sf.saxon.s9api.XdmValue;
import net.sf.saxon.s9api.XsltExecutable;
import net.sf.saxon.s9api.XsltTransformer;
import net.sf.saxon.trans.XPathException;
import org.apache.maven.artifact.DependencyResolutionRequiredException;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugin.logging.Log;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.LifecyclePhase;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.plugins.annotations.ResolutionScope;
import org.apache.maven.project.MavenProject;
import org.apache.maven.shared.dependency.graph.DependencyGraphBuilder;
import org.xml.sax.Attributes;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.XMLFilter;
import org.xml.sax.XMLReader;
import org.xml.sax.ext.EntityResolver2;
import org.xml.sax.helpers.ParserAdapter;
import org.xml.sax.helpers.XMLFilterImpl;
import top.marchand.maven.gaulois.compiler.utils.GauloisConfigScanner;
import top.marchand.maven.gaulois.compiler.utils.GauloisSet;
import top.marchand.maven.gaulois.compiler.utils.GauloisXsl;
import top.marchand.maven.saxon.utils.SaxonOptions;
import top.marchand.xml.maven.plugin.xsl.AbstractCompiler;
@Mojo(name="gaulois-compiler", defaultPhase = LifecyclePhase.COMPILE, requiresDependencyResolution = ResolutionScope.COMPILE)
public class GCMojo extends AbstractCompiler {
@Parameter( defaultValue = "${project}", readonly = true, required = true )
private MavenProject project;
@Override
public MavenProject getProject() { return project; }
@Component( hint = "default" )
private DependencyGraphBuilder dependencyGraphBuilder;
@Override
public DependencyGraphBuilder getGraphBuilder() { return dependencyGraphBuilder; }
/**
* The directory containing generated classes of the project being tested.
* This will be included after the test classes in the test classpath.
*/
@Parameter( defaultValue = "${project.build.outputDirectory}" )
private File classesDirectory;
/**
* List of gauloisPipeFileset
*/
@Parameter
List gauloisPipeFilesets;
/**
* The catalog file to use to compile
*/
@Parameter
private File catalog;
@Parameter(defaultValue = "${project.basedir}")
private File projectBaseDir;
/**
* The directory where imported schemas will be copied to. Be aware that
* if your schema structure uses relatives parent (../xxx) location, no
* file could be copied outside of ${project.build.outputDirectory}
*/
@Parameter (defaultValue = "${project.build.outputDirectory}/gc/schemas")
private File schemasDestination;
/**
* A XSL to post-compile the gaulois-pipe config file, if required
*/
@Parameter
private File postCompiler;
/**
* Saxon options, to configure Saxon.
* See {@linkplain https://github.com/cmarchand/saxonOptions-mvn-plug-utils/wiki}
*/
@Parameter
SaxonOptions saxonOptions;
private XsltExecutable postCompilerXsl;
private XsltExecutable gauloisCompilerXsl;
private XsltExecutable xutScanner;
private XsltExecutable xutFilter;
/**
* The list of directories where XSL sources are located in
*/
@Parameter
List xslSourceDirs;
// inner working variables
private Set gauloisSets;
private Map foundXsls;
public static final SAXParserFactory PARSER_FACTORY = SAXParserFactory.newInstance();
private ArrayList classpaths;
private static final String XUT_NS = "https://github.com/mricaud/xml-utilities";
private static final QName QN_DEP_TYPE = new QName("dependency-type");
private static final QName QN_URI = new QName("uri");
private static final QName QN_ABS_URI = new QName("abs-uri");
private static final QName QN_NAME = new QName("name");
private static final QName QN_PARAM_SCHEMAS = new QName("schemas");
private static final QName QN_PARAM_XSLMAP = new QName("xslMap");
private static final QName QN_TARGET_PATH = new QName("targetPath");
@Override
public void execute() throws MojoExecutionException, MojoFailureException {
if(gauloisPipeFilesets==null) {
getLog().error(LOG_PREFIX+"\n"+ERROR_MESSAGE);
throw new MojoExecutionException(ERROR_MESSAGE);
}
if(xslSourceDirs==null) xslSourceDirs = new ArrayList<>();
if(xslSourceDirs.isEmpty()) {
xslSourceDirs.add(new File(projectBaseDir, "src/main/xsl"));
}
Log log = getLog();
try {
initSaxon();
} catch(XPathException ex) {
getLog().error("while configuring saxon:",ex);
}
loadClasspath();
gauloisSets = new TreeSet<>();
foundXsls = new HashMap<>();
ThreadLocal th = new ThreadLocal<>();
th.set(getEntityResolver());
getLog().warn(LOG_PREFIX+getXsltCompiler().getProcessor().getUnderlyingConfiguration().getSourceParserClass());
try {
URL url = getClass().getResource("/org/mricaud/xml-utilities/get-xml-file-static-dependency-tree.xsl");
StreamSource ssource = new StreamSource(url.openStream());
ssource.setSystemId(url.toExternalForm());
xutScanner = getXsltCompiler().compile(ssource);
xutFilter = getXsltCompiler().compile(new StreamSource(getClass().getResource("/top/marchand/maven/gaulois/compiler/schema-filter.xsl").openStream()));
} catch(SaxonApiException | IOException ex) {
throw new MojoFailureException("while compiling xut xsl", ex);
}
Path targetDir = classesDirectory.toPath();
boolean hasError = false;
getLog().debug(LOG_PREFIX+" looking for gaulois-pipe config files");
for(FileSet fs: gauloisPipeFilesets) {
if(fs.getUri()!=null && !fs.getUri().isEmpty()) {
try {
Source source = compiler.getURIResolver().resolve(fs.getUri(), null);
String sPath = fs.getUriPath();
getLog().debug(LOG_PREFIX+" sPath="+sPath);
Path targetPath = targetDir.resolve(sPath).getParent();
getLog().debug(LOG_PREFIX+" targetPath="+targetPath.toString());
String sourceFileName = sPath.substring(sPath.lastIndexOf("/")+1);
if(sourceFileName.contains("?")) {
sourceFileName = sourceFileName.substring(0, sourceFileName.indexOf("?")-1);
}
// we keep the same extension for gaulois config files
File targetFile = targetPath.resolve(sourceFileName).toFile();
getLog().debug(LOG_PREFIX+" targetFile="+targetFile.getAbsolutePath());
hasError |= scanGauloisFile(source, targetFile, targetDir);
} catch(TransformerException | URISyntaxException ex) {
hasError = true;
getLog().error("while parsing "+fs.getUri(), ex);
}
} else {
List pathes = fs.getFiles(projectBaseDir, log);
// this must be call after the call to fs.getFiles, as fs.dir is modified by fs.getFiles
Path basedir = new File(fs.getDir()).toPath();
getLog().debug(LOG_PREFIX+"looking in "+basedir.toString());
for(Path p: pathes) {
getLog().debug(LOG_PREFIX+"found "+p.toString());
File sourceFile = basedir.resolve(p).toFile();
Path targetPath = p.getParent()==null ? targetDir : targetDir.resolve(p.getParent());
String sourceFileName = sourceFile.getName();
// we keep the same extension for gaulois config files
File targetFile = targetPath.resolve(sourceFileName).toFile();
try {
hasError |= scanGauloisFile(sourceFile, targetFile, targetDir);
} catch(FileNotFoundException | URISyntaxException ex) {
// it can not be thrown but we are required to catch it
hasError = true;
getLog().error(LOG_PREFIX+"while parsing "+p.toString(), ex);
}
}
}
}
StringBuilder sb = new StringBuilder();
for(GauloisXsl gx: foundXsls.values()) sb.append(gx.getXslSystemId()).append("->").append(gx.getTargetFile().getAbsolutePath()).append("\n");
getLog().debug("Found XSL: "+sb.toString());
if(!hasError) {
for(String xslSystemId: foundXsls.keySet()) {
try {
getLog().debug(LOG_PREFIX+" compiling "+xslSystemId);
Source xslSource = new StreamSource(xslSystemId);
File targetFile = foundXsls.get(xslSystemId).getTargetFile();
compileFile(xslSource, targetFile);
} catch (FileNotFoundException | SaxonApiException ex) {
getLog().warn(LOG_PREFIX+" while compiling "+xslSystemId, ex);
}
}
Source xsl = new StreamSource(this.getClass().getResourceAsStream("/top/marchand/maven/gaulois/compiler/gaulois-compiler.xsl"));
try {
gauloisCompilerXsl = getXsltCompiler().compile(xsl);
// we need to construct a map target path>
XdmMap xslMap = buildXslMap(foundXsls);
for(GauloisSet gs: gauloisSets) {
getLog().debug(LOG_PREFIX+" compiling "+gs.getGauloisConfigSystemId());
// passer ici les schemas à déclarer
compileGaulois(new StreamSource(gs.getGauloisConfigSystemId()), gs.getTargetFile(), gs.getAllSchemas(), xslMap);
}
} catch(SaxonApiException ex) {
getLog().error(ex);
}
} else {
getLog().warn(LOG_PREFIX+" Errors occured");
}
}
private XdmMap buildXslMap(Map xsls) {
Map tempMap = new HashMap<>();
for(GauloisXsl gx: xsls.values()) {
tempMap.put(gx.getOriginalSystemId(), gx.getTargetFile().getAbsolutePath());
}
return XdmMap.makeMap(tempMap);
}
private static final String LOG_PREFIX = "[gaulois-compiler]";
private static final String ERROR_MESSAGE = "\n\t\n\t\tsrc/main/xsl... \n\t \n \n is required in gaulois-compiler-maven-plugin configuration";
@Override
public File getCatalogFile() {
return catalog;
}
/**
* Scans a gaulois config file to extract all xslt files, and store them into xslToCompile
* xslt/@href MUST be an absolute URI, in cp:/ protocol.
* Else, the whole gaulois-pipe config file is ignored.
* @param sourceFile The file to scan. It MUST be a gaulois config file.
* @param targetFile The target file where scanned file will be stored.
* @param targetDir The build dir
* @return false if an error occured
* @throws java.io.FileNotFoundException If a file is not found. Should never be thrown.
* @throws java.net.URISyntaxException Maybe... or not...
*/
protected boolean scanGauloisFile(File sourceFile, File targetFile, Path targetDir) throws FileNotFoundException, URISyntaxException {
String systemId = sourceFile.toURI().toString();
getLog().debug("scanGauloisFile("+systemId+",File, Path);");
InputSource is = new InputSource(new FileInputStream(sourceFile));
is.setSystemId(systemId);
SAXSource source = new SAXSource(is);
source.setSystemId(systemId);
return scanGauloisFile(source, targetFile, targetDir);
}
protected boolean scanGauloisFile(Source source, File targetFile, Path targetDir) throws URISyntaxException {
assert(source.getSystemId()!=null);
try {
final XMLReader reader = new ParserAdapter(PARSER_FACTORY.newSAXParser().getParser());
final GauloisConfigScanner scanner = new GauloisConfigScanner(xslSourceDirs, classesDirectory, getUriResolver(), getLog(), classpaths);
XMLFilter filter = new XMLFilterImpl(reader) {
@Override
public void startElement(String uri, String localName, String qName, Attributes atts) throws SAXException {
super.startElement(uri, localName, qName, atts);
scanner.startElement(uri, localName, qName, atts);
}
};
// use systemId to create a new InputSource, and to keep the Source not consumed
filter.parse(source.getSystemId());
if(scanner.hasErrors()) {
for(String errorMsg: scanner.getErrorMessages()) {
getLog().error(errorMsg);
}
} else {
GauloisSet set = new GauloisSet(source.getSystemId(), targetFile);
if(!gauloisSets.contains(set)) {
gauloisSets.add(set);
for(Source xslSource: scanner.getXslToCompile().keySet()) {
GauloisXsl xsl = foundXsls.get(xslSource.getSystemId());
if(xsl==null) {
GauloisConfigScanner.FileInfo fileInfo = scanner.getXslToCompile().get(xslSource);
xsl = new GauloisXsl(xslSource.getSystemId(), fileInfo.getFile(), fileInfo.getOriginalSystemId());
foundXsls.put(xslSource.getSystemId(), xsl);
scanForSchemas(xsl);
}
set.getXsls().add(xsl);
}
}
}
return scanner.hasErrors();
} catch(ParserConfigurationException | SAXException | SaxonApiException | IOException ex) {
getLog().error("while scanning "+source.getSystemId(), ex);
return true;
}
}
protected void compileGaulois(Source source, File target, Set schemas, XdmMap xslMap) throws SaxonApiException {
XsltTransformer tr = gauloisCompilerXsl.load();
tr.setURIResolver(getUriResolver());
ArrayList values = new ArrayList<>();
for(String schema:schemas) {
getLog().info(LOG_PREFIX+target.getName()+" has schema: "+schema);
values.add(new XdmAtomicValue(schema));
}
XdmValue sequence = new XdmValue(values);
tr.setMessageListener(new MessageListener() {
@Override
public void message(XdmNode xn, boolean bln, SourceLocator sl) {
getLog().debug(xn.toString());
}
});
tr.setParameter(QN_PARAM_SCHEMAS, sequence);
tr.setParameter(QN_PARAM_XSLMAP, xslMap);
tr.setParameter(QN_TARGET_PATH, XdmValue.makeValue(classesDirectory.getAbsolutePath()));
XsltTransformer first = tr;
// post compiler ?
XsltTransformer pc = getPostCompiler();
if(pc!=null) {
tr.setDestination(pc);
tr = pc;
}
Serializer ser = getProcessor().newSerializer(target);
tr.setDestination(ser);
XdmNode sourceNode = getBuilder().build(source);
first.setInitialContextNode(sourceNode);
first.transform();
first.close();
}
protected XsltTransformer getPostCompiler() {
if(postCompilerXsl==null && postCompiler!=null && postCompiler.exists() && postCompiler.isFile()) {
try {
postCompilerXsl = getXsltCompiler().compile(new StreamSource(new FileInputStream(postCompiler)));
} catch(SaxonApiException | FileNotFoundException ex) {
getLog().error("while compiling post-compiler "+postCompiler.getAbsolutePath(), ex);
}
}
return postCompilerXsl==null ? null : postCompilerXsl.load();
}
protected void scanForSchemas(GauloisXsl xsl) throws SaxonApiException, URISyntaxException, IOException {
getLog().debug(LOG_PREFIX+" scanning for schema "+xsl.getXslSystemId());
XsltTransformer xut = xutScanner.load();
xut.setMessageListener(new NullMessageListener());
XsltTransformer filter = xutFilter.load();
xut.setDestination(filter);
XdmDestination dest = new XdmDestination();
filter.setDestination(dest);
xut.setMessageListener(new MessageListener() {
@Override
public void message(XdmNode xn, boolean bln, SourceLocator sl) { }
});
XdmNode xslDocument = getBuilder().build(new StreamSource(xsl.getXslSystemId()));
xut.setInitialContextNode(xslDocument);
xut.setParameter(new QName(XUT_NS, "xut:get-xml-file-static-dependency-tree.filterDuplicatedDependencies"), new XdmAtomicValue(true));
xut.transform();
XdmNode dependencies = dest.getXdmNode();
// now, walk through dependencies
// all first-level childs are imported schemas
XdmNode file = (XdmNode)(dependencies.axisIterator(Axis.CHILD).next());
XdmSequenceIterator xsi = file.axisIterator(Axis.CHILD);
while(xsi.hasNext()) {
exploreFile(xsl, (XdmNode)(xsi.next()));
}
}
private void exploreFile(GauloisXsl xsl, XdmNode node) throws URISyntaxException, IOException {
String dependencyType = node.getAttributeValue(QN_DEP_TYPE);
String absUri = node.getAttributeValue(QN_ABS_URI);
getLog().debug(LOG_PREFIX+"\texploreFile <"+node.getNodeName()+" "+QN_DEP_TYPE.toString()+"="+dependencyType+" absUri="+absUri);
if(dependencyType.equals("xsl:import-schema")) { // always true, but for documentation
String uri = node.getAttributeValue(QN_URI);
String name = node.getAttributeValue(QN_NAME);
SchemaTarget targetSchema = getTargetSchemaFile(name, absUri);
xsl.getSchemas().add(targetSchema.getAccessUri());
getLog().debug(LOG_PREFIX+"\turi is "+absUri);
copyUriToFile(absUri, targetSchema.getFileLocation());
XdmSequenceIterator it = node.axisIterator(Axis.CHILD);
while(it.hasNext()) {
XdmNode schemaNode = (XdmNode)it.next();
copySubSchema(targetSchema.getFileLocation(), schemaNode);
}
}
}
private void copySubSchema(File parent, XdmNode schemaNode) throws URISyntaxException, IOException {
String uri = schemaNode.getAttributeValue(QN_URI);
String absUri = schemaNode.getAttributeValue(QN_ABS_URI);
File schemaFile = parent.toPath().resolve(uri).toFile();
copyFile(new File(new URI(absUri)), schemaFile);
XdmSequenceIterator it = schemaNode.axisIterator(Axis.CHILD);
while(it.hasNext()) {
XdmNode subSchemaNode = (XdmNode)it.next();
copySubSchema(schemaFile, subSchemaNode);
}
}
private SchemaTarget getTargetSchemaFile(String name, String absUri) {
File destSchema = new File(getSchemasDestination(), name);
Path p = classesDirectory.toPath().relativize(destSchema.toPath());
String accessUri = "cp:/"+p.toString();
return new SchemaTarget(accessUri, destSchema);
}
private void copyFile(File source, File dest) throws IOException {
dest.getParentFile().mkdirs();
try (
FileChannel in = new FileInputStream(source).getChannel();
FileChannel out = new FileOutputStream(dest).getChannel()) {
in.transferTo (0, in.size(), out);
}
}
private void copyUriToFile(String uri, File dest) throws IOException, URISyntaxException {
dest.getParentFile().mkdirs();
URL url = new URI(uri).toURL();
InputStream is = url.openStream();
try (
ReadableByteChannel in = Channels.newChannel(is);
FileChannel out = new FileOutputStream(dest).getChannel()) {
final long size = 5*1024;
long offset = 0;
long vol = out.transferFrom(in, 0, size);
while(vol==size) {
offset+=vol;
vol = out.transferFrom(in, offset, size);
}
}
}
private void loadClasspath() {
try {
classpaths = new ArrayList<>(project.getCompileClasspathElements().size());
for(Object i:project.getCompileClasspathElements()) {
File f = new File(i.toString());
classpaths.add(f.toURI().toString());
}
getLog().debug(LOG_PREFIX+"classpaths="+classpaths);
} catch(DependencyResolutionRequiredException ex) {
getLog().error(LOG_PREFIX+ex.getMessage(),ex);
}
}
/**
* Returns the schemas destination
* @return The schemas destination directory
*/
public File getSchemasDestination() {
return schemasDestination;
}
/**
* A class to store a schema location.
*/
private class SchemaTarget {
private final String accessUri;
private final File fileLocation;
public SchemaTarget(final String accessUri, final File fileLocation) {
super();
this.accessUri = accessUri;
this.fileLocation = fileLocation;
}
public String getAccessUri() {
return accessUri;
}
public File getFileLocation() {
return fileLocation;
}
}
@Override
public SaxonOptions getSaxonOptions() {
return saxonOptions;
}
private class NullMessageListener implements MessageListener {
@Override
public void message(XdmNode xn, boolean bln, SourceLocator sl) {}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy