Browse Source
Use `maven-failsafe-plugin` to run CLI integration tests as part of the `spring-boot-cli` project, removing the need for `spring-boot-cli-integration-tests`.pull/244/head
9 changed files with 266 additions and 254 deletions
@ -1,51 +0,0 @@ |
|||||||
<?xml version="1.0" encoding="UTF-8"?> |
|
||||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" |
|
||||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> |
|
||||||
<modelVersion>4.0.0</modelVersion> |
|
||||||
<parent> |
|
||||||
<groupId>org.springframework.boot</groupId> |
|
||||||
<artifactId>spring-boot-parent</artifactId> |
|
||||||
<version>0.5.0.BUILD-SNAPSHOT</version> |
|
||||||
<relativePath>../spring-boot-parent</relativePath> |
|
||||||
</parent> |
|
||||||
<artifactId>spring-boot-cli-integration-tests</artifactId> |
|
||||||
<properties> |
|
||||||
<main.basedir>${basedir}/..</main.basedir> |
|
||||||
</properties> |
|
||||||
<dependencies> |
|
||||||
<dependency> |
|
||||||
<groupId>junit</groupId> |
|
||||||
<artifactId>junit</artifactId> |
|
||||||
<scope>test</scope> |
|
||||||
</dependency> |
|
||||||
</dependencies> |
|
||||||
<build> |
|
||||||
<plugins> |
|
||||||
<plugin> |
|
||||||
<groupId>org.apache.maven.plugins</groupId> |
|
||||||
<artifactId>maven-dependency-plugin</artifactId> |
|
||||||
<executions> |
|
||||||
<execution> |
|
||||||
<id>unpack</id> |
|
||||||
<phase>generate-resources</phase> |
|
||||||
<goals> |
|
||||||
<goal>unpack</goal> |
|
||||||
</goals> |
|
||||||
<configuration> |
|
||||||
<artifactItems> |
|
||||||
<artifactItem> |
|
||||||
<groupId>${project.groupId}</groupId> |
|
||||||
<artifactId>spring-boot-cli</artifactId> |
|
||||||
<version>${project.version}</version> |
|
||||||
<classifier>bin</classifier> |
|
||||||
<type>zip</type> |
|
||||||
<overWrite>true</overWrite> |
|
||||||
</artifactItem> |
|
||||||
</artifactItems> |
|
||||||
</configuration> |
|
||||||
</execution> |
|
||||||
</executions> |
|
||||||
</plugin> |
|
||||||
</plugins> |
|
||||||
</build> |
|
||||||
</project> |
|
||||||
@ -1,70 +0,0 @@ |
|||||||
package org.springframework.boot.cli.it; |
|
||||||
|
|
||||||
import java.io.IOException; |
|
||||||
|
|
||||||
import org.junit.Test; |
|
||||||
import org.springframework.boot.cli.it.infrastructure.Cli; |
|
||||||
import org.springframework.boot.cli.it.infrastructure.CliInvocation; |
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals; |
|
||||||
import static org.junit.Assert.assertTrue; |
|
||||||
|
|
||||||
/** |
|
||||||
* @author Andy Wilkinson |
|
||||||
*/ |
|
||||||
public class UsabilityTests { |
|
||||||
|
|
||||||
private final Cli cli = new Cli(); |
|
||||||
|
|
||||||
@Test |
|
||||||
public void hintProducesListOfValidCommands() throws IOException, |
|
||||||
InterruptedException { |
|
||||||
CliInvocation cli = this.cli.invoke("hint"); |
|
||||||
|
|
||||||
assertEquals(0, cli.await()); |
|
||||||
assertEquals(0, cli.getErrorOutput().length()); |
|
||||||
assertEquals(6, cli.getStandardOutputLines().size()); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void invokingWithNoArgumentsDisplaysHelp() throws IOException, |
|
||||||
InterruptedException { |
|
||||||
CliInvocation cli = this.cli.invoke(); |
|
||||||
|
|
||||||
assertEquals(1, cli.await()); // TODO Should this be 0? Probably not.
|
|
||||||
assertEquals(0, cli.getErrorOutput().length()); |
|
||||||
assertTrue(cli.getStandardOutput().startsWith("usage:")); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void unrecognizedCommandsAreHandledGracefully() throws IOException, |
|
||||||
InterruptedException { |
|
||||||
CliInvocation cli = this.cli.invoke("not-a-real-command"); |
|
||||||
|
|
||||||
assertEquals(1, cli.await()); |
|
||||||
assertTrue( |
|
||||||
cli.getErrorOutput(), |
|
||||||
cli.getErrorOutput().contains( |
|
||||||
"'not-a-real-command' is not a valid command")); |
|
||||||
assertEquals(0, cli.getStandardOutput().length()); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void version() throws IOException, InterruptedException { |
|
||||||
CliInvocation cli = this.cli.invoke("version"); |
|
||||||
|
|
||||||
assertEquals(0, cli.await()); |
|
||||||
assertEquals(0, cli.getErrorOutput().length()); |
|
||||||
assertTrue(cli.getStandardOutput(), |
|
||||||
cli.getStandardOutput().startsWith("Spring CLI v")); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void help() throws IOException, InterruptedException { |
|
||||||
CliInvocation cli = this.cli.invoke("help"); |
|
||||||
|
|
||||||
assertEquals(1, cli.await()); // TODO Should this be 0? Perhaps.
|
|
||||||
assertEquals(0, cli.getErrorOutput().length()); |
|
||||||
assertTrue(cli.getStandardOutput().startsWith("usage:")); |
|
||||||
} |
|
||||||
} |
|
||||||
@ -1,48 +0,0 @@ |
|||||||
package org.springframework.boot.cli.it.infrastructure; |
|
||||||
|
|
||||||
import java.io.File; |
|
||||||
import java.io.IOException; |
|
||||||
import java.util.ArrayList; |
|
||||||
import java.util.Arrays; |
|
||||||
import java.util.List; |
|
||||||
|
|
||||||
/** |
|
||||||
* @author Andy Wilkinson |
|
||||||
*/ |
|
||||||
public final class Cli { |
|
||||||
|
|
||||||
public CliInvocation invoke(String... args) throws IOException { |
|
||||||
return new CliInvocation(launchCli(args)); |
|
||||||
} |
|
||||||
|
|
||||||
private Process launchCli(String... args) throws IOException { |
|
||||||
List<String> arguments = Arrays.asList(args); |
|
||||||
|
|
||||||
List<String> command = new ArrayList<String>(); |
|
||||||
command.add(findLaunchScript().getAbsolutePath()); |
|
||||||
command.addAll(arguments); |
|
||||||
|
|
||||||
return new ProcessBuilder(command).start(); |
|
||||||
} |
|
||||||
|
|
||||||
private File findLaunchScript() { |
|
||||||
File dependencyDirectory = new File("target/dependency"); |
|
||||||
if (dependencyDirectory.isDirectory()) { |
|
||||||
File[] files = dependencyDirectory.listFiles(); |
|
||||||
if (files.length == 1) { |
|
||||||
File binDirectory = new File(files[0], "bin"); |
|
||||||
if (isWindows()) { |
|
||||||
return new File(binDirectory, "spring.bat"); |
|
||||||
} |
|
||||||
else { |
|
||||||
return new File(binDirectory, "spring"); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
throw new IllegalStateException("Could not find CLI launch script"); |
|
||||||
} |
|
||||||
|
|
||||||
private boolean isWindows() { |
|
||||||
return File.separatorChar == '\\'; |
|
||||||
} |
|
||||||
} |
|
||||||
@ -1,84 +0,0 @@ |
|||||||
package org.springframework.boot.cli.it.infrastructure; |
|
||||||
|
|
||||||
import java.io.BufferedReader; |
|
||||||
import java.io.IOException; |
|
||||||
import java.io.InputStream; |
|
||||||
import java.io.StringReader; |
|
||||||
import java.util.ArrayList; |
|
||||||
import java.util.List; |
|
||||||
|
|
||||||
/** |
|
||||||
* @author Andy Wilkinson |
|
||||||
*/ |
|
||||||
public final class CliInvocation { |
|
||||||
|
|
||||||
private final StringBuffer errorOutput = new StringBuffer(); |
|
||||||
|
|
||||||
private final StringBuffer standardOutput = new StringBuffer(); |
|
||||||
|
|
||||||
private final Process process; |
|
||||||
|
|
||||||
CliInvocation(Process process) { |
|
||||||
this.process = process; |
|
||||||
|
|
||||||
new Thread(new StreamReadingRunnable(this.process.getErrorStream(), |
|
||||||
this.errorOutput)).start(); |
|
||||||
new Thread(new StreamReadingRunnable(this.process.getInputStream(), |
|
||||||
this.standardOutput)).start(); |
|
||||||
} |
|
||||||
|
|
||||||
public String getErrorOutput() { |
|
||||||
return this.errorOutput.toString(); |
|
||||||
} |
|
||||||
|
|
||||||
public String getStandardOutput() { |
|
||||||
return this.standardOutput.toString(); |
|
||||||
} |
|
||||||
|
|
||||||
public List<String> getStandardOutputLines() { |
|
||||||
BufferedReader reader = new BufferedReader(new StringReader( |
|
||||||
this.standardOutput.toString())); |
|
||||||
String line; |
|
||||||
List<String> lines = new ArrayList<String>(); |
|
||||||
try { |
|
||||||
while ((line = reader.readLine()) != null) { |
|
||||||
lines.add(line); |
|
||||||
} |
|
||||||
} |
|
||||||
catch (IOException e) { |
|
||||||
throw new RuntimeException("Failed to read standard output"); |
|
||||||
} |
|
||||||
return lines; |
|
||||||
} |
|
||||||
|
|
||||||
public int await() throws InterruptedException { |
|
||||||
return this.process.waitFor(); |
|
||||||
} |
|
||||||
|
|
||||||
private final class StreamReadingRunnable implements Runnable { |
|
||||||
|
|
||||||
private final InputStream stream; |
|
||||||
|
|
||||||
private final StringBuffer output; |
|
||||||
|
|
||||||
private final byte[] buffer = new byte[4096]; |
|
||||||
|
|
||||||
private StreamReadingRunnable(InputStream stream, StringBuffer buffer) { |
|
||||||
this.stream = stream; |
|
||||||
this.output = buffer; |
|
||||||
} |
|
||||||
|
|
||||||
public void run() { |
|
||||||
int read; |
|
||||||
try { |
|
||||||
while ((read = this.stream.read(this.buffer)) > 0) { |
|
||||||
this.output.append(new String(this.buffer, 0, read)); |
|
||||||
} |
|
||||||
} |
|
||||||
catch (IOException e) { |
|
||||||
// Allow thread to die
|
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
@ -0,0 +1,84 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2012-2014 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 org.springframework.boot.cli; |
||||||
|
|
||||||
|
import java.io.IOException; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
import org.springframework.boot.cli.infrastructure.CommandLineInvoker; |
||||||
|
import org.springframework.boot.cli.infrastructure.CommandLineInvoker.Invocation; |
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.containsString; |
||||||
|
import static org.hamcrest.Matchers.equalTo; |
||||||
|
import static org.hamcrest.Matchers.startsWith; |
||||||
|
import static org.junit.Assert.assertThat; |
||||||
|
|
||||||
|
/** |
||||||
|
* Integration Tests for the command line application. |
||||||
|
* |
||||||
|
* @author Andy Wilkinson |
||||||
|
* @author Phillip Webb |
||||||
|
*/ |
||||||
|
public class CommandLineIT { |
||||||
|
|
||||||
|
private final CommandLineInvoker cli = new CommandLineInvoker(); |
||||||
|
|
||||||
|
@Test |
||||||
|
public void hintProducesListOfValidCommands() throws IOException, |
||||||
|
InterruptedException { |
||||||
|
Invocation cli = this.cli.invoke("hint"); |
||||||
|
assertThat(cli.await(), equalTo(0)); |
||||||
|
assertThat(cli.getErrorOutput().length(), equalTo(0)); |
||||||
|
assertThat(cli.getStandardOutputLines().size(), equalTo(6)); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void invokingWithNoArgumentsDisplaysHelp() throws IOException, |
||||||
|
InterruptedException { |
||||||
|
Invocation cli = this.cli.invoke(); |
||||||
|
assertThat(cli.await(), equalTo(1)); |
||||||
|
assertThat(cli.getErrorOutput().length(), equalTo(0)); |
||||||
|
assertThat(cli.getStandardOutput(), startsWith("usage:")); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void unrecognizedCommandsAreHandledGracefully() throws IOException, |
||||||
|
InterruptedException { |
||||||
|
Invocation cli = this.cli.invoke("not-a-real-command"); |
||||||
|
assertThat(cli.await(), equalTo(1)); |
||||||
|
assertThat(cli.getErrorOutput(), |
||||||
|
containsString("'not-a-real-command' is not a valid command")); |
||||||
|
assertThat(cli.getStandardOutput().length(), equalTo(0)); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void version() throws IOException, InterruptedException { |
||||||
|
Invocation cli = this.cli.invoke("version"); |
||||||
|
assertThat(cli.await(), equalTo(0)); |
||||||
|
assertThat(cli.getErrorOutput().length(), equalTo(0)); |
||||||
|
assertThat(cli.getStandardOutput(), startsWith("Spring CLI v")); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void help() throws IOException, InterruptedException { |
||||||
|
Invocation cli = this.cli.invoke("help"); |
||||||
|
assertThat(cli.await(), equalTo(1)); |
||||||
|
assertThat(cli.getErrorOutput().length(), equalTo(0)); |
||||||
|
assertThat(cli.getStandardOutput(), startsWith("usage:")); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
@ -0,0 +1,150 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2012-2014 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 org.springframework.boot.cli.infrastructure; |
||||||
|
|
||||||
|
import java.io.BufferedReader; |
||||||
|
import java.io.File; |
||||||
|
import java.io.FileFilter; |
||||||
|
import java.io.IOException; |
||||||
|
import java.io.InputStream; |
||||||
|
import java.io.StringReader; |
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.Arrays; |
||||||
|
import java.util.List; |
||||||
|
|
||||||
|
import org.springframework.util.Assert; |
||||||
|
|
||||||
|
/** |
||||||
|
* Utility to invoke the command line in the same way as a user would, i.e. via the shell |
||||||
|
* script in the package's bin directory. |
||||||
|
* |
||||||
|
* @author Andy Wilkinson |
||||||
|
* @author Phillip Webb |
||||||
|
*/ |
||||||
|
public final class CommandLineInvoker { |
||||||
|
|
||||||
|
public Invocation invoke(String... args) throws IOException { |
||||||
|
return new Invocation(runCliProcess(args)); |
||||||
|
} |
||||||
|
|
||||||
|
private Process runCliProcess(String... args) throws IOException { |
||||||
|
List<String> command = new ArrayList<String>(); |
||||||
|
command.add(findLaunchScript().getAbsolutePath()); |
||||||
|
command.addAll(Arrays.asList(args)); |
||||||
|
return new ProcessBuilder(command).start(); |
||||||
|
} |
||||||
|
|
||||||
|
private File findLaunchScript() { |
||||||
|
File dir = new File("target"); |
||||||
|
dir = dir.listFiles(new FileFilter() { |
||||||
|
@Override |
||||||
|
public boolean accept(File pathname) { |
||||||
|
return pathname.isDirectory() && pathname.getName().contains("-bin"); |
||||||
|
} |
||||||
|
})[0]; |
||||||
|
dir = new File(dir, dir.getName().replace("-bin", "") |
||||||
|
.replace("spring-boot-cli", "spring")); |
||||||
|
dir = new File(dir, "bin"); |
||||||
|
File launchScript = new File(dir, isWindows() ? "spring.bat" : "spring"); |
||||||
|
Assert.state(launchScript.exists() && launchScript.isFile(), |
||||||
|
"Could not find CLI launch script " + launchScript.getAbsolutePath()); |
||||||
|
return launchScript; |
||||||
|
} |
||||||
|
|
||||||
|
private boolean isWindows() { |
||||||
|
return File.separatorChar == '\\'; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* An ongoing CLI invocation. |
||||||
|
*/ |
||||||
|
public final class Invocation { |
||||||
|
|
||||||
|
private final StringBuffer err = new StringBuffer(); |
||||||
|
|
||||||
|
private final StringBuffer out = new StringBuffer(); |
||||||
|
|
||||||
|
private final Process process; |
||||||
|
|
||||||
|
Invocation(Process process) { |
||||||
|
this.process = process; |
||||||
|
new Thread(new StreamReadingRunnable(this.process.getErrorStream(), this.err)) |
||||||
|
.start(); |
||||||
|
new Thread(new StreamReadingRunnable(this.process.getInputStream(), this.out)) |
||||||
|
.start(); |
||||||
|
} |
||||||
|
|
||||||
|
public String getErrorOutput() { |
||||||
|
return this.err.toString(); |
||||||
|
} |
||||||
|
|
||||||
|
public String getStandardOutput() { |
||||||
|
return this.out.toString(); |
||||||
|
} |
||||||
|
|
||||||
|
public List<String> getStandardOutputLines() { |
||||||
|
BufferedReader reader = new BufferedReader(new StringReader( |
||||||
|
this.out.toString())); |
||||||
|
String line; |
||||||
|
List<String> lines = new ArrayList<String>(); |
||||||
|
try { |
||||||
|
while ((line = reader.readLine()) != null) { |
||||||
|
lines.add(line); |
||||||
|
} |
||||||
|
} |
||||||
|
catch (IOException e) { |
||||||
|
throw new RuntimeException("Failed to read standard output"); |
||||||
|
} |
||||||
|
return lines; |
||||||
|
} |
||||||
|
|
||||||
|
public int await() throws InterruptedException { |
||||||
|
return this.process.waitFor(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* {@link Runnable} to copy stream output. |
||||||
|
*/ |
||||||
|
private final class StreamReadingRunnable implements Runnable { |
||||||
|
|
||||||
|
private final InputStream stream; |
||||||
|
|
||||||
|
private final StringBuffer output; |
||||||
|
|
||||||
|
private final byte[] buffer = new byte[4096]; |
||||||
|
|
||||||
|
private StreamReadingRunnable(InputStream stream, StringBuffer buffer) { |
||||||
|
this.stream = stream; |
||||||
|
this.output = buffer; |
||||||
|
} |
||||||
|
|
||||||
|
public void run() { |
||||||
|
int read; |
||||||
|
try { |
||||||
|
while ((read = this.stream.read(this.buffer)) > 0) { |
||||||
|
this.output.append(new String(this.buffer, 0, read)); |
||||||
|
} |
||||||
|
} |
||||||
|
catch (IOException ex) { |
||||||
|
// Allow thread to die
|
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
Loading…
Reference in new issue