Move CommandLineHelpGenerator to initializr-web

Closes gh-807
This commit is contained in:
Stephane Nicoll
2019-02-04 17:21:12 +01:00
parent f5395c07e1
commit ce2d210d75
7 changed files with 4 additions and 4 deletions

View File

@@ -29,7 +29,6 @@ import javax.servlet.http.HttpServletRequest;
import com.samskivert.mustache.Mustache;
import io.spring.initializr.generator.BasicProjectRequest;
import io.spring.initializr.generator.CommandLineHelpGenerator;
import io.spring.initializr.generator.ProjectGenerator;
import io.spring.initializr.generator.ProjectRequest;
import io.spring.initializr.metadata.DependencyMetadata;
@@ -45,6 +44,7 @@ import io.spring.initializr.web.mapper.InitializrMetadataJsonMapper;
import io.spring.initializr.web.mapper.InitializrMetadataV21JsonMapper;
import io.spring.initializr.web.mapper.InitializrMetadataV2JsonMapper;
import io.spring.initializr.web.mapper.InitializrMetadataVersion;
import io.spring.initializr.web.support.CommandLineHelpGenerator;
import org.apache.tools.ant.Project;
import org.apache.tools.ant.taskdefs.Tar;
import org.apache.tools.ant.taskdefs.Zip;

View File

@@ -0,0 +1,326 @@
/*
* Copyright 2012-2019 the original author or authors.
*
* 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 io.spring.initializr.web.support;
import java.beans.PropertyDescriptor;
import java.util.Comparator;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.stream.Collectors;
import io.spring.initializr.metadata.Dependency;
import io.spring.initializr.metadata.InitializrMetadata;
import io.spring.initializr.metadata.MetadataElement;
import io.spring.initializr.metadata.Type;
import io.spring.initializr.util.TemplateRenderer;
import org.springframework.beans.BeanWrapperImpl;
/**
* Generate help pages for command-line clients.
*
* @author Stephane Nicoll
*/
public class CommandLineHelpGenerator {
private static final String LOGO = " . ____ _ __ _ _\n"
+ " /\\\\ / ___'_ __ _ _(_)_ __ __ _ \\ \\ \\ \\\n"
+ "( ( )\\___ | '_ | '_| | '_ \\/ _` | \\ \\ \\ \\\n"
+ " \\\\/ ___)| |_)| | | | | || (_| | ) ) ) )\n"
+ " ' |____| .__|_| |_|_| |_\\__, | / / / /\n"
+ " =========|_|==============|___/=/_/_/_/";
private final TemplateRenderer template;
public CommandLineHelpGenerator(TemplateRenderer template) {
this.template = template;
}
/**
* Generate the capabilities of the service as a generic plain text document. Used
* when no particular agent was detected.
* @param metadata the initializr metadata
* @param serviceUrl the service URL
* @return the generic capabilities text document
*/
public String generateGenericCapabilities(InitializrMetadata metadata,
String serviceUrl) {
Map<String, Object> model = initializeCommandLineModel(metadata, serviceUrl);
model.put("hasExamples", false);
return this.template.process("cli-capabilities.txt", model);
}
/**
* Generate the capabilities of the service using "curl" as a plain text document.
* @param metadata the initializr metadata
* @param serviceUrl the service URL
* @return the generic capabilities text document
*/
public String generateCurlCapabilities(InitializrMetadata metadata,
String serviceUrl) {
Map<String, Object> model = initializeCommandLineModel(metadata, serviceUrl);
model.put("examples", this.template.process("curl-examples.txt", model));
model.put("hasExamples", true);
return this.template.process("cli-capabilities.txt", model);
}
/**
* Generate the capabilities of the service using "HTTPie" as a plain text document.
* @param metadata the initializr metadata
* @param serviceUrl the service URL
* @return the generic capabilities text document
*/
public String generateHttpieCapabilities(InitializrMetadata metadata,
String serviceUrl) {
Map<String, Object> model = initializeCommandLineModel(metadata, serviceUrl);
model.put("examples", this.template.process("httpie-examples.txt", model));
model.put("hasExamples", true);
return this.template.process("cli-capabilities.txt", model);
}
/**
* Generate the capabilities of the service using Spring Boot CLI as a plain text
* document.
* @param metadata the initializr metadata
* @param serviceUrl the service URL
* @return the generic capabilities text document
*/
public String generateSpringBootCliCapabilities(InitializrMetadata metadata,
String serviceUrl) {
Map<String, Object> model = initializeSpringBootCliModel(metadata, serviceUrl);
model.put("hasExamples", false);
return this.template.process("boot-cli-capabilities.txt", model);
}
protected Map<String, Object> initializeCommandLineModel(InitializrMetadata metadata,
String serviceUrl) {
Map<String, Object> model = new LinkedHashMap<>();
model.put("logo", LOGO);
model.put("serviceUrl", serviceUrl);
model.put("dependencies", generateDependencyTable(metadata));
model.put("types", generateTypeTable(metadata, "Rel", false));
Map<String, Object> defaults = metadata.defaults();
defaults.put("applicationName", metadata.getConfiguration()
.generateApplicationName(metadata.getName().getContent()));
defaults.put("baseDir", "no base dir");
defaults.put("dependencies", "none");
Map<String, Object> parametersDescription = buildParametersDescription(metadata);
String[][] parameterTable = new String[defaults.size() + 1][];
parameterTable[0] = new String[] { "Parameter", "Description", "Default value" };
int i = 1;
for (String id : defaults.keySet().stream().sorted()
.collect(Collectors.toList())) {
String[] data = new String[3];
data[0] = id;
data[1] = (String) parametersDescription.get(id);
data[2] = (String) defaults.get(id);
parameterTable[i++] = data;
}
model.put("parameters", TableGenerator.generate(parameterTable));
return model;
}
protected Map<String, Object> initializeSpringBootCliModel(
InitializrMetadata metadata, String serviceUrl) {
Map<String, Object> model = new LinkedHashMap<>();
model.put("logo", LOGO);
model.put("serviceUrl", serviceUrl);
model.put("dependencies", generateDependencyTable(metadata));
model.put("types", generateTypeTable(metadata, "Id", true));
Map<String, Object> defaults = metadata.defaults();
Map<String, Object> parametersDescription = buildParametersDescription(metadata);
String[][] parameterTable = new String[defaults.size() + 1][];
parameterTable[0] = new String[] { "Id", "Description", "Default value" };
int i = 1;
for (String id : defaults.keySet().stream().sorted()
.collect(Collectors.toList())) {
String[] data = new String[3];
data[0] = id;
data[1] = (String) parametersDescription.get(id);
data[2] = (String) defaults.get(id);
parameterTable[i++] = data;
}
model.put("parameters", TableGenerator.generate(parameterTable));
return model;
}
protected String generateDependencyTable(InitializrMetadata metadata) {
String[][] dependencyTable = new String[metadata.getDependencies().getAll().size()
+ 1][];
dependencyTable[0] = new String[] { "Id", "Description", "Required version" };
int i = 1;
for (Dependency dep : metadata.getDependencies().getAll().stream()
.sorted(Comparator.comparing(MetadataElement::getId))
.collect(Collectors.toList())) {
String[] data = new String[3];
data[0] = dep.getId();
data[1] = (dep.getDescription() != null) ? dep.getDescription()
: dep.getName();
data[2] = dep.getVersionRequirement();
dependencyTable[i++] = data;
}
return TableGenerator.generate(dependencyTable);
}
protected String generateTypeTable(InitializrMetadata metadata, String linkHeader,
boolean addTags) {
String[][] typeTable = new String[metadata.getTypes().getContent().size() + 1][];
if (addTags) {
typeTable[0] = new String[] { linkHeader, "Description", "Tags" };
}
else {
typeTable[0] = new String[] { linkHeader, "Description" };
}
int i = 1;
for (Type type : metadata.getTypes().getContent().stream()
.sorted(Comparator.comparing(MetadataElement::getId))
.collect(Collectors.toList())) {
String[] data = new String[typeTable[0].length];
data[0] = (type.isDefault() ? type.getId() + " *" : type.getId());
data[1] = (type.getDescription() != null) ? type.getDescription()
: type.getName();
if (addTags) {
data[2] = buildTagRepresentation(type);
}
typeTable[i++] = data;
}
return TableGenerator.generate(typeTable);
}
protected Map<String, Object> buildParametersDescription(
InitializrMetadata metadata) {
Map<String, Object> result = new LinkedHashMap<>();
BeanWrapperImpl wrapper = new BeanWrapperImpl(metadata);
for (PropertyDescriptor descriptor : wrapper.getPropertyDescriptors()) {
Object value = wrapper.getPropertyValue(descriptor.getName());
BeanWrapperImpl nested = new BeanWrapperImpl(value);
if (nested.isReadableProperty("description")
&& nested.isReadableProperty("id")) {
result.put((String) nested.getPropertyValue("id"),
nested.getPropertyValue("description"));
}
}
result.put("applicationName", "application name");
result.put("baseDir", "base directory to create in the archive");
return result;
}
private static String buildTagRepresentation(Type type) {
if (type.getTags().isEmpty()) {
return "";
}
return String.join(",",
type.getTags().entrySet().stream()
.map((entry) -> entry.getKey() + ":" + entry.getValue())
.toArray(String[]::new));
}
/**
* Utility to generate a text table.
*/
private static class TableGenerator {
static final String NEW_LINE = System.getProperty("line.separator");
/**
* Generate a table description for the specified {@code content}.
* <p>
* The {@code content} is a two-dimensional array holding the rows of the table.
* The first entry holds the header of the table.
* @param content the table content
* @return the generated table
*/
public static String generate(String[][] content) {
StringBuilder sb = new StringBuilder();
int[] columnsLength = computeColumnsLength(content);
appendTableSeparation(sb, columnsLength);
appendRow(sb, content, columnsLength, 0); // Headers
appendTableSeparation(sb, columnsLength);
for (int i = 1; i < content.length; i++) {
appendRow(sb, content, columnsLength, i);
}
appendTableSeparation(sb, columnsLength);
return sb.toString();
}
private static void appendRow(StringBuilder sb, String[][] content,
int[] columnsLength, int rowIndex) {
String[] row = content[rowIndex];
if (row != null) {
for (int i = 0; i < row.length; i++) {
sb.append("| ").append(fill(row[i], columnsLength[i])).append(" ");
}
}
sb.append("|");
sb.append(NEW_LINE);
}
private static void appendTableSeparation(StringBuilder sb, int[] headersLength) {
for (int headerLength : headersLength) {
sb.append("+").append(multiply("-", headerLength + 2));
}
sb.append("+");
sb.append(NEW_LINE);
}
private static String fill(String data, int columnSize) {
if (data == null) {
return multiply(" ", columnSize);
}
else {
int i = columnSize - data.length();
return data + multiply(" ", i);
}
}
private static String multiply(String value, int size) {
StringBuilder s = new StringBuilder();
for (int i = 0; i < size; i++) {
s.append(value);
}
return s.toString();
}
private static int[] computeColumnsLength(String[][] content) {
int count = content[0].length;
int[] result = new int[count];
for (int i = 0; i < count; i++) {
result[i] = largest(content, i);
}
return result;
}
private static int largest(String[][] content, int column) {
int max = 0;
for (String[] rows : content) {
if (rows != null) {
String s = rows[column];
if (s != null && s.length() > max) {
max = s.length();
}
}
}
return max;
}
}
}

View File

@@ -0,0 +1,11 @@
{{{logo}}}
:: Service capabilities :: {{serviceUrl}}
Supported dependencies
{{{dependencies}}}
Project types (* denotes the default)
{{{types}}}
Parameters
{{{parameters}}}

View File

@@ -0,0 +1,24 @@
{{{logo}}}
:: Spring Initializr :: {{serviceUrl}}
This service generates quickstart projects that can be easily customized.
Possible customizations include a project's dependencies, Java version, and
build system or build structure. See below for further details.
The services uses a HAL based hypermedia format to expose a set of resources
to interact with. If you access this root resource requesting application/json
as media type the response will contain the following links:
{{{types}}}
The URI templates take a set of parameters to customize the result of a request
to the linked resource.
{{{parameters}}}
The following section has a list of supported identifiers for the comma-separated
list of "dependencies".
{{{dependencies}}}
{{#hasExamples}}
Examples:
{{{examples}}}
{{/hasExamples}}

View File

@@ -0,0 +1,13 @@
To create a default demo.zip:
$ curl {{serviceUrl}}/starter.zip -o demo.zip
To create a web project using Java 11:
$ curl {{serviceUrl}}/starter.zip -d dependencies=web \\
-d javaVersion=11 -o demo.zip
To create a web/data-jpa gradle project unpacked:
$ curl {{serviceUrl}}/starter.tgz -d dependencies=web,data-jpa \\
-d type=gradle-project -d baseDir=my-dir | tar -xzvf -
To generate a Maven POM with war packaging:
$ curl {{serviceUrl}}/pom.xml -d packaging=war -o pom.xml

View File

@@ -0,0 +1,13 @@
To create a default project:
$ http {{serviceUrl}}/starter.zip -d
To create a web project using Java 11:
$ http {{serviceUrl}}/starter.zip dependencies==web \\
javaVersion==11 -d
To create a web/data-jpa gradle project unpacked:
$ http {{serviceUrl}}/starter.tgz dependencies==web,data-jpa \\
type==gradle-project baseDir==my-dir | tar -xzvf -
To generate a Maven POM with war packaging:
$ http {{serviceUrl}}/pom.xml packaging==war -o pom.xml

View File

@@ -0,0 +1,179 @@
/*
* Copyright 2012-2019 the original author or authors.
*
* 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 io.spring.initializr.web.support;
import java.util.Arrays;
import io.spring.initializr.metadata.Dependency;
import io.spring.initializr.metadata.InitializrMetadata;
import io.spring.initializr.metadata.Type;
import io.spring.initializr.test.metadata.InitializrMetadataTestBuilder;
import io.spring.initializr.util.TemplateRenderer;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import static org.assertj.core.api.Assertions.assertThat;
/**
* @author Stephane Nicoll
*/
class CommandLineHelpGeneratorTests {
private CommandLineHelpGenerator generator;
@BeforeEach
public void init() {
this.generator = new CommandLineHelpGenerator(new TemplateRenderer());
}
@Test
void generateGenericCapabilities() {
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addDependencyGroup("test", createDependency("id-b", "depB"),
createDependency("id-a", "depA", "and some description"))
.build();
String content = this.generator.generateGenericCapabilities(metadata,
"https://fake-service");
assertCommandLineCapabilities(content);
assertThat(content).contains("id-a | and some description |");
assertThat(content).contains("id-b | depB");
assertThat(content).contains("https://fake-service");
assertThat(content).doesNotContain("Examples:");
assertThat(content).doesNotContain("curl");
}
@Test
void generateCapabilitiesWithTypeDescription() {
Type type = new Type();
type.setId("foo");
type.setName("foo-name");
type.setDescription("foo-desc");
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addType(type).build();
String content = this.generator.generateGenericCapabilities(metadata,
"https://fake-service");
assertCommandLineCapabilities(content);
assertThat(content).contains("| foo");
assertThat(content).contains("| foo-desc");
}
@Test
void generateCapabilitiesWithAlias() {
Dependency dependency = createDependency("dep", "some description");
dependency.setAliases(Arrays.asList("legacy", "another"));
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addDependencyGroup("test", dependency).build();
String content = this.generator.generateGenericCapabilities(metadata,
"https://fake-service");
assertCommandLineCapabilities(content);
assertThat(content).contains("dep | some description |");
assertThat(content).doesNotContain("legacy").doesNotContain("another");
}
@Test
void generateCurlCapabilities() {
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addDependencyGroup("test", createDependency("id-b", "depB"),
createDependency("id-a", "depA", "and some description"))
.build();
String content = this.generator.generateCurlCapabilities(metadata,
"https://fake-service");
assertCommandLineCapabilities(content);
assertThat(content).contains("id-a | and some description |");
assertThat(content).contains("id-b | depB");
assertThat(content).contains("https://fake-service");
assertThat(content).contains("Examples:");
assertThat(content).contains("curl https://fake-service");
}
@Test
void generateHttpCapabilities() {
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addDependencyGroup("test", createDependency("id-b", "depB"),
createDependency("id-a", "depA", "and some description"))
.build();
String content = this.generator.generateHttpieCapabilities(metadata,
"https://fake-service");
assertCommandLineCapabilities(content);
assertThat(content).contains("id-a | and some description |");
assertThat(content).contains("id-b | depB");
assertThat(content).contains("https://fake-service");
assertThat(content).contains("Examples:");
assertThat(content).doesNotContain("curl");
assertThat(content).contains("http https://fake-service");
}
@Test
void generateSpringBootCliCapabilities() {
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addDependencyGroup("test", createDependency("id-b", "depB"),
createDependency("id-a", "depA", "and some description"))
.build();
String content = this.generator.generateSpringBootCliCapabilities(metadata,
"https://fake-service");
assertThat(content).contains("| Id");
assertThat(content).contains("| Tags");
assertThat(content).contains("id-a | and some description |");
assertThat(content).contains("id-b | depB");
assertThat(content).contains("https://fake-service");
assertThat(content).doesNotContain("Examples:");
assertThat(content).doesNotContain("curl");
assertThat(content).doesNotContain("| Rel");
assertThat(content).doesNotContain("| dependencies");
assertThat(content).doesNotContain("| applicationName");
assertThat(content).doesNotContain("| baseDir");
}
@Test
void generateCapabilitiesWithVersionRange() {
Dependency first = Dependency.withId("first");
first.setDescription("first desc");
first.setVersionRange("1.2.0.RELEASE");
Dependency second = Dependency.withId("second");
second.setDescription("second desc");
second.setVersionRange(" [1.2.0.RELEASE,1.3.0.M1) ");
InitializrMetadata metadata = InitializrMetadataTestBuilder.withDefaults()
.addDependencyGroup("test", first, second).build();
String content = this.generator.generateSpringBootCliCapabilities(metadata,
"https://fake-service");
assertThat(content)
.contains("| first | first desc | >=1.2.0.RELEASE |");
assertThat(content)
.contains("| second | second desc | >=1.2.0.RELEASE and <1.3.0.M1 |");
}
private static void assertCommandLineCapabilities(String content) {
assertThat(content).contains("| Rel");
assertThat(content).contains("| dependencies");
assertThat(content).contains("| applicationName");
assertThat(content).contains("| baseDir");
assertThat(content).doesNotContain("| Tags");
}
private static Dependency createDependency(String id, String name) {
return createDependency(id, name, null);
}
private static Dependency createDependency(String id, String name,
String description) {
Dependency dependency = Dependency.withId(id);
dependency.setDescription(description);
dependency.setName(name);
return dependency;
}
}