org.neo4j.codegen.source.SourceCode Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-codegen Show documentation
Show all versions of neo4j-codegen Show documentation
Simple library for generating code.
The newest version!
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [https://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.codegen.source;
import static java.util.Objects.requireNonNull;
import static org.neo4j.codegen.CompilationFailureException.format;
import static org.neo4j.codegen.source.ClasspathHelper.fullClasspathStringFor;
import java.io.IOException;
import java.io.PrintStream;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Collections;
import java.util.List;
import javax.annotation.processing.Processor;
import javax.tools.Diagnostic;
import javax.tools.JavaFileObject;
import org.neo4j.codegen.CodeGenerationStrategy;
import org.neo4j.codegen.CodeGenerationStrategyNotSupportedException;
import org.neo4j.codegen.CodeGenerator;
import org.neo4j.codegen.CodeGeneratorOption;
import org.neo4j.codegen.TypeReference;
public enum SourceCode implements CodeGeneratorOption {
SIMPLIFY_TRY_WITH_RESOURCE;
public static final CodeGenerationStrategy> SOURCECODE = new CodeGenerationStrategy() {
@Override
protected Configuration createConfigurator(ClassLoader loader) {
return new Configuration().withOptions("-classpath", fullClasspathStringFor(loader));
}
@Override
protected CodeGenerator createCodeGenerator(ClassLoader loader, Configuration configuration)
throws CodeGenerationStrategyNotSupportedException {
return new JavaSourceGenerator(loader, configuration, configuration.sourceCompilerFor(this));
}
@Override
protected String name() {
return "SOURCECODE";
}
};
public static final CodeGeneratorOption PRINT_SOURCE = new SourceVisitor() {
@Override
protected void visitSource(TypeReference reference, CharSequence sourceCode) {
String[] lines = sourceCode.toString().split("\\n");
System.out.println("=== Generated class " + reference.fullName() + " ===\n");
for (int i = 0; i < lines.length; i++) {
System.out.print(i + 1);
System.out.print('\t');
System.out.println(lines[i]);
}
}
@Override
public String toString() {
return "PRINT_SOURCE";
}
};
static final CodeGeneratorOption USE_JDK_JAVA_COMPILER = JdkCompiler.FACTORY;
public static final CodeGeneratorOption PRINT_WARNINGS = printWarningsTo(System.err);
private static CodeGeneratorOption printWarningsTo(PrintStream err) {
return new PrintWarningsOption(err);
}
public static CodeGeneratorOption annotationProcessor(Processor processor) {
return new AnnotationProcessorOption(requireNonNull(processor));
}
public static CodeGeneratorOption sourceLocation(Path path) {
return new SourceLocationOption(requireNonNull(path));
}
public static CodeGeneratorOption temporarySourceCodeLocation() {
try {
return new SourceLocationOption(Files.createTempDirectory(null));
} catch (IOException e) {
throw new RuntimeException("Failed to create temporary directory.", e);
}
}
@Override
public void applyTo(Object target) {
if (target instanceof Configuration) {
((Configuration) target).withFlag(this);
}
}
private static class AnnotationProcessorOption implements CodeGeneratorOption {
private final Processor processor;
AnnotationProcessorOption(Processor processor) {
this.processor = processor;
}
@Override
public void applyTo(Object target) {
if (target instanceof Configuration) {
((Configuration) target).withAnnotationProcessor(processor);
}
}
@Override
public String toString() {
return "annotationProcessor( " + processor + " )";
}
}
private static class SourceLocationOption extends SourceVisitor {
private final Path path;
SourceLocationOption(Path path) {
this.path = path;
}
@Override
public String toString() {
return "sourceLocation( " + path + " )";
}
@Override
protected void visitSource(TypeReference reference, CharSequence sourceCode) {
try {
Path location = path(reference);
Files.createDirectories(location.getParent());
Files.write(location, Collections.singletonList(sourceCode), Charset.defaultCharset());
} catch (IOException e) {
throw new RuntimeException("Failed to write source code", e);
}
}
private Path path(TypeReference reference) {
return path.resolve(reference.packageName().replace('.', '/') + "/" + reference.name() + ".java");
}
}
private static class PrintWarningsOption implements CodeGeneratorOption, WarningsHandler {
private final PrintStream target;
PrintWarningsOption(PrintStream target) {
this.target = target;
}
@Override
public void applyTo(Object target) {
if (target instanceof Configuration) {
((Configuration) target).withWarningsHandler(this);
}
}
@Override
public String toString() {
return "PRINT_WARNINGS";
}
@Override
public void handle(List> diagnostics) {
for (Diagnostic extends JavaFileObject> diagnostic : diagnostics) {
format(target, diagnostic);
}
}
}
}