2016-05-02 16:17:39 -07:00
|
|
|
/*
|
2017-04-12 11:42:50 -07:00
|
|
|
* Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
|
2016-05-02 16:17:39 -07:00
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This code 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
|
|
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
|
|
* accompanied this code).
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License version
|
|
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
|
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
|
|
* questions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package toolbox;
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.Path;
|
2016-06-17 17:40:01 -07:00
|
|
|
import java.nio.file.Paths;
|
2016-05-02 16:17:39 -07:00
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.Arrays;
|
2016-06-17 17:40:01 -07:00
|
|
|
import java.util.LinkedHashSet;
|
2016-05-02 16:17:39 -07:00
|
|
|
import java.util.List;
|
2016-06-17 17:40:01 -07:00
|
|
|
import java.util.Set;
|
2016-05-02 16:17:39 -07:00
|
|
|
import java.util.stream.Collectors;
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Builder for module declarations.
|
|
|
|
*/
|
2016-05-02 16:17:39 -07:00
|
|
|
public class ModuleBuilder {
|
|
|
|
|
|
|
|
private final ToolBox tb;
|
|
|
|
private final String name;
|
2016-06-17 17:40:01 -07:00
|
|
|
private String comment = "";
|
2017-12-13 17:27:43 -08:00
|
|
|
private boolean open;
|
2016-06-17 17:40:01 -07:00
|
|
|
private List<String> requires = new ArrayList<>();
|
|
|
|
private List<String> exports = new ArrayList<>();
|
2017-04-18 06:29:53 -07:00
|
|
|
private List<String> opens = new ArrayList<>();
|
2016-06-17 17:40:01 -07:00
|
|
|
private List<String> uses = new ArrayList<>();
|
|
|
|
private List<String> provides = new ArrayList<>();
|
2016-05-02 16:17:39 -07:00
|
|
|
private List<String> content = new ArrayList<>();
|
2016-06-17 17:40:01 -07:00
|
|
|
private Set<Path> modulePath = new LinkedHashSet<>();
|
2016-05-02 16:17:39 -07:00
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Creates a builder for a module.
|
2017-12-13 17:27:43 -08:00
|
|
|
* @param tb a Toolbox that can be used to compile the module declaration
|
2016-06-17 17:40:01 -07:00
|
|
|
* @param name the name of the module to be built
|
|
|
|
*/
|
2016-05-02 16:17:39 -07:00
|
|
|
public ModuleBuilder(ToolBox tb, String name) {
|
2017-12-13 17:27:43 -08:00
|
|
|
this(tb, false, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a builder for a module.
|
|
|
|
* @param tb a Toolbox that can be used to compile the module declaration
|
|
|
|
* @param open whether or not this is an open module
|
|
|
|
* @param name the name of the module to be built
|
|
|
|
*/
|
|
|
|
public ModuleBuilder(ToolBox tb, boolean open, String name) {
|
2016-05-02 16:17:39 -07:00
|
|
|
this.tb = tb;
|
2017-12-13 17:27:43 -08:00
|
|
|
this.open = open;
|
2016-05-02 16:17:39 -07:00
|
|
|
this.name = name;
|
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Sets the doc comment for the declaration.
|
|
|
|
* @param comment the content of the comment, excluding the initial
|
|
|
|
* '/**', leading whitespace and asterisks, and the final trailing 'a;/'.
|
|
|
|
* @return this builder
|
|
|
|
*/
|
|
|
|
public ModuleBuilder comment(String comment) {
|
|
|
|
this.comment = comment;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-12-01 09:02:42 +00:00
|
|
|
* Adds a "requires" directive to the declaration.
|
|
|
|
* @param module the name of the module that is required
|
|
|
|
* @param modulePath a path in while to locate the modules
|
|
|
|
* if the declaration is compiled
|
|
|
|
* @return this builder
|
|
|
|
*/
|
|
|
|
public ModuleBuilder requires(String module, Path... modulePath) {
|
|
|
|
addDirective(requires, "requires " + module + ";");
|
|
|
|
this.modulePath.addAll(Arrays.asList(modulePath));
|
|
|
|
return this;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a "requires static" directive to the declaration.
|
|
|
|
* @param module the name of the module that is required
|
2016-06-17 17:40:01 -07:00
|
|
|
* @param modulePath a path in which to locate the modules
|
|
|
|
* if the declaration is compiled
|
|
|
|
* @return this builder
|
|
|
|
*/
|
2016-12-01 09:02:42 +00:00
|
|
|
public ModuleBuilder requiresStatic(String module, Path... modulePath) {
|
|
|
|
addDirective(requires, "requires static " + module + ";");
|
2016-06-17 17:40:01 -07:00
|
|
|
this.modulePath.addAll(Arrays.asList(modulePath));
|
|
|
|
return this;
|
2016-05-02 16:17:39 -07:00
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
2016-12-01 09:02:42 +00:00
|
|
|
* Adds a "requires transitive" directive to the declaration.
|
|
|
|
* @param module the name of the module that is required
|
|
|
|
* @param modulePath a path in which to locate the modules
|
2016-06-17 17:40:01 -07:00
|
|
|
* if the declaration is compiled
|
|
|
|
* @return this builder
|
|
|
|
*/
|
2016-12-01 09:02:42 +00:00
|
|
|
public ModuleBuilder requiresTransitive(String module, Path... modulePath) {
|
|
|
|
addDirective(requires, "requires transitive " + module + ";");
|
2016-06-17 17:40:01 -07:00
|
|
|
this.modulePath.addAll(Arrays.asList(modulePath));
|
2016-05-02 16:17:39 -07:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2016-12-01 09:02:42 +00:00
|
|
|
/**
|
|
|
|
* Adds a "requires static transitive" directive to the declaration.
|
|
|
|
* @param module the name of the module that is required
|
|
|
|
* @param modulePath a path in which to locate the modules
|
|
|
|
* if the declaration is compiled
|
|
|
|
* @return this builder
|
|
|
|
*/
|
|
|
|
public ModuleBuilder requiresStaticTransitive(String module, Path... modulePath) {
|
|
|
|
addDirective(requires, "requires static transitive " + module + ";");
|
|
|
|
this.modulePath.addAll(Arrays.asList(modulePath));
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds an unqualified "exports" directive to the declaration.
|
|
|
|
* @param pkg the name of the package to be exported
|
|
|
|
* @return this builder
|
|
|
|
*/
|
|
|
|
public ModuleBuilder exports(String pkg) {
|
|
|
|
return addDirective(exports, "exports " + pkg + ";");
|
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Adds a qualified "exports" directive to the declaration.
|
|
|
|
* @param pkg the name of the package to be exported
|
|
|
|
* @param module the name of the module to which it is to be exported
|
|
|
|
* @return this builder
|
|
|
|
*/
|
2016-05-02 16:17:39 -07:00
|
|
|
public ModuleBuilder exportsTo(String pkg, String module) {
|
2016-12-01 09:02:42 +00:00
|
|
|
return addDirective(exports, "exports " + pkg + " to " + module + ";");
|
2016-05-02 16:17:39 -07:00
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
2017-04-18 06:29:53 -07:00
|
|
|
* Adds an unqualified "opens" directive to the declaration.
|
|
|
|
* @param pkg the name of the package to be opened
|
2016-12-01 09:02:42 +00:00
|
|
|
* @return this builder
|
|
|
|
*/
|
2017-04-18 06:29:53 -07:00
|
|
|
public ModuleBuilder opens(String pkg) {
|
|
|
|
return addDirective(opens, "opens " + pkg + ";");
|
2016-12-01 09:02:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-04-18 06:29:53 -07:00
|
|
|
* Adds a qualified "opens" directive to the declaration.
|
|
|
|
* @param pkg the name of the package to be opened
|
|
|
|
* @param module the name of the module to which it is to be opened
|
2016-12-01 09:02:42 +00:00
|
|
|
* @return this builder
|
|
|
|
*/
|
2017-04-18 06:29:53 -07:00
|
|
|
public ModuleBuilder opensTo(String pkg, String module) {
|
|
|
|
return addDirective(opens, "opens " + pkg + " to " + module + ";");
|
2016-05-02 16:17:39 -07:00
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Adds a "uses" directive to the declaration.
|
|
|
|
* @param service the name of the service type
|
|
|
|
* @return this builder
|
|
|
|
*/
|
|
|
|
public ModuleBuilder uses(String service) {
|
2016-12-01 09:02:42 +00:00
|
|
|
return addDirective(uses, "uses " + service + ";");
|
2016-05-02 16:17:39 -07:00
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Adds a "provides" directive to the declaration.
|
|
|
|
* @param service the name of the service type
|
|
|
|
* @param implementation the name of the implementation type
|
|
|
|
* @return this builder
|
|
|
|
*/
|
2016-05-02 16:17:39 -07:00
|
|
|
public ModuleBuilder provides(String service, String implementation) {
|
2016-12-01 09:02:42 +00:00
|
|
|
return addDirective(provides, "provides " + service + " with " + implementation + ";");
|
|
|
|
}
|
|
|
|
|
|
|
|
private ModuleBuilder addDirective(List<String> directives, String directive) {
|
|
|
|
directives.add(directive);
|
2016-05-02 16:17:39 -07:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Adds type definitions to the module.
|
|
|
|
* @param content a series of strings, each representing the content of
|
|
|
|
* a compilation unit to be included with the module
|
|
|
|
* @return this builder
|
|
|
|
*/
|
2016-05-02 16:17:39 -07:00
|
|
|
public ModuleBuilder classes(String... content) {
|
|
|
|
this.content.addAll(Arrays.asList(content));
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Writes the module declaration and associated additional compilation
|
|
|
|
* units to a module directory within a given directory.
|
|
|
|
* @param srcDir the directory in which a directory will be created
|
|
|
|
* to contain the source files for the module
|
|
|
|
* @return the directory containing the source files for the module
|
|
|
|
*/
|
|
|
|
public Path write(Path srcDir) throws IOException {
|
|
|
|
Files.createDirectories(srcDir);
|
2016-05-02 16:17:39 -07:00
|
|
|
List<String> sources = new ArrayList<>();
|
2016-06-17 17:40:01 -07:00
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
if (!comment.isEmpty()) {
|
2017-04-12 11:42:50 -07:00
|
|
|
sb.append("/**\n * ")
|
|
|
|
.append(comment.replace("\n", "\n * "))
|
|
|
|
.append("\n */\n");
|
2016-06-17 17:40:01 -07:00
|
|
|
}
|
2017-12-13 17:27:43 -08:00
|
|
|
if (open) {
|
|
|
|
sb.append("open ");
|
|
|
|
}
|
2016-08-15 18:00:36 -07:00
|
|
|
sb.append("module ").append(name).append(" {\n");
|
2016-06-17 17:40:01 -07:00
|
|
|
requires.forEach(r -> sb.append(" " + r + "\n"));
|
|
|
|
exports.forEach(e -> sb.append(" " + e + "\n"));
|
2017-04-18 06:29:53 -07:00
|
|
|
opens.forEach(o -> sb.append(" " + o + "\n"));
|
2016-06-17 17:40:01 -07:00
|
|
|
uses.forEach(u -> sb.append(" " + u + "\n"));
|
|
|
|
provides.forEach(p -> sb.append(" " + p + "\n"));
|
|
|
|
sb.append("}");
|
|
|
|
sources.add(sb.toString());
|
2016-05-02 16:17:39 -07:00
|
|
|
sources.addAll(content);
|
2016-06-17 17:40:01 -07:00
|
|
|
Path moduleSrc = srcDir.resolve(name);
|
2016-05-02 16:17:39 -07:00
|
|
|
tb.writeJavaFiles(moduleSrc, sources.toArray(new String[]{}));
|
|
|
|
return moduleSrc;
|
|
|
|
}
|
|
|
|
|
2016-06-17 17:40:01 -07:00
|
|
|
/**
|
|
|
|
* Writes the source files for the module to an interim directory,
|
|
|
|
* and then compiles them to a given directory.
|
|
|
|
* @param modules the directory in which a directory will be created
|
|
|
|
* to contain the compiled class files for the module
|
|
|
|
* @throws IOException if an error occurs while compiling the files
|
|
|
|
*/
|
|
|
|
public void build(Path modules) throws IOException {
|
|
|
|
build(Paths.get(modules + "Src"), modules);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Writes the source files for the module to a specified directory,
|
|
|
|
* and then compiles them to a given directory.
|
|
|
|
* @param srcDir the directory in which a directory will be created
|
|
|
|
* to contain the source files for the module
|
|
|
|
* @param modules the directory in which a directory will be created
|
|
|
|
* to contain the compiled class files for the module
|
|
|
|
* @throws IOException if an error occurs while compiling the files
|
|
|
|
*/
|
|
|
|
public void build(Path src, Path modules) throws IOException {
|
|
|
|
Path moduleSrc = write(src);
|
|
|
|
String mp = modulePath.stream()
|
|
|
|
.map(Path::toString)
|
|
|
|
.collect(Collectors.joining(File.pathSeparator));
|
2016-05-02 16:17:39 -07:00
|
|
|
new JavacTask(tb)
|
2016-06-17 17:40:01 -07:00
|
|
|
.outdir(Files.createDirectories(modules.resolve(name)))
|
2016-08-30 20:49:41 -07:00
|
|
|
.options("--module-path", mp)
|
2016-05-02 16:17:39 -07:00
|
|
|
.files(tb.findJavaFiles(moduleSrc))
|
|
|
|
.run()
|
|
|
|
.writeAll();
|
|
|
|
}
|
|
|
|
}
|