Experimental feature for testing WKDCli against wkd-test-suite

This commit is contained in:
Paul Schaub 2022-03-02 18:01:12 +01:00
parent 049c14b779
commit cd3f36e26a
Signed by: vanitasvitae
GPG Key ID: 62BEE9264BF17311
9 changed files with 159 additions and 11 deletions

View File

@ -27,6 +27,8 @@ dependencies {
// Logging
testImplementation "ch.qos.logback:logback-classic:$logbackVersion"
implementation 'org.slf4j:slf4j-nop:1.7.36'
testImplementation project(":wkd-test-suite")
}
test {

View File

@ -0,0 +1,18 @@
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
//
// SPDX-License-Identifier: Apache-2.0
package pgp.wkd.cli;
public class CertNotFetchableException extends RuntimeException {
public static final int ERROR_CODE = 3;
public CertNotFetchableException(String message) {
super(message);
}
public CertNotFetchableException(String message, Throwable e) {
super(message, e);
}
}

View File

@ -8,9 +8,11 @@ package pgp.wkd.cli;
* Exception that gets thrown when an OpenPGP certificate is not carrying a User-ID binding for the email address
* that was used to look the certificate up via WKD.
*/
public class MissingUserIdException extends Exception {
public class MissingUserIdException extends RuntimeException {
public MissingUserIdException() {
super();
public static final int ERROR_CODE = 7;
public MissingUserIdException(String message) {
super(message);
}
}

View File

@ -24,6 +24,19 @@ public class WKDCLI {
public static int execute(String[] args) {
return new CommandLine(WKDCLI.class)
.setExitCodeExceptionMapper(new CommandLine.IExitCodeExceptionMapper() {
@Override
public int getExitCode(Throwable exception) {
if (exception instanceof MissingUserIdException) {
return MissingUserIdException.ERROR_CODE;
} else if (exception instanceof CertNotFetchableException) {
return CertNotFetchableException.ERROR_CODE;
}
// Others get mapped to 1
return 1;
}
})
.setCommandName("wkdcli")
.execute(args);
}

View File

@ -12,6 +12,7 @@ import pgp.wkd.AbstractWKDFetcher;
import pgp.wkd.HttpUrlConnectionWKDFetcher;
import pgp.wkd.WKDAddress;
import pgp.wkd.WKDAddressHelper;
import pgp.wkd.cli.CertNotFetchableException;
import pgp.wkd.cli.MissingUserIdException;
import picocli.CommandLine;
@ -40,7 +41,7 @@ public class Fetch implements Runnable {
)
boolean armor = false;
AbstractWKDFetcher fetcher = new HttpUrlConnectionWKDFetcher();
public static AbstractWKDFetcher fetcher = new HttpUrlConnectionWKDFetcher();
@Override
public void run() {
@ -65,7 +66,7 @@ public class Fetch implements Runnable {
}
}
if (!containsEmail) {
throw new MissingUserIdException();
throw new MissingUserIdException("Fetched certificate does not contain email address " + email);
}
if (armor) {
@ -77,12 +78,7 @@ public class Fetch implements Runnable {
}
} catch (IOException e) {
System.err.println("Could not fetch certificate.");
e.printStackTrace();
System.exit(1);
} catch (MissingUserIdException e) {
System.err.println("Fetched certificate does not contain email address " + email);
System.exit(1);
throw new CertNotFetchableException("Certificate cannot be fetched.", e);
}
}
}

View File

@ -0,0 +1,32 @@
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
//
// SPDX-License-Identifier: Apache-2.0
package pgp.wkd.cli.test_suite;
import pgp.wkd.AbstractWKDFetcher;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.nio.file.Path;
public class DirectoryBasedWkdFetcher extends AbstractWKDFetcher {
// The directory containing the .well-known subdirectory
private final Path rootPath;
public DirectoryBasedWkdFetcher(Path rootPath) {
this.rootPath = rootPath;
}
@Override
protected InputStream fetchUri(URI uri) throws IOException {
String path = uri.getPath();
File file = rootPath.resolve(path.substring(1)).toFile(); // get rid of leading slash at start of path
FileInputStream fileIn = new FileInputStream(file);
return fileIn;
}
}

View File

@ -0,0 +1,52 @@
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
//
// SPDX-License-Identifier: Apache-2.0
package pgp.wkd.cli.test_suite;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import pgp.wkd.cli.WKDCLI;
import pgp.wkd.cli.command.Fetch;
import pgp.wkd.test_suite.TestCase;
import pgp.wkd.test_suite.TestSuite;
import pgp.wkd.test_suite.TestSuiteGenerator;
import java.nio.file.Files;
import java.nio.file.Path;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotEquals;
public class TestSuiteTestRunner {
private static TestSuite suite;
@BeforeAll
public static void setup() throws Exception {
Path tempDir = Files.createTempDirectory("wkd-test");
tempDir.toFile().deleteOnExit();
Fetch.fetcher = new DirectoryBasedWkdFetcher(tempDir);
String domain = "example.com";
TestSuiteGenerator generator = new TestSuiteGenerator(domain);
suite = generator.generateTestSuiteInDirectory(tempDir.toFile(), TestSuiteGenerator.Method.direct);
}
@Test
public void runTestsAgainstTestSuite() {
for (TestCase testCase : suite.getTestCases()) {
System.out.println("Executing test " + testCase.getTestTitle());
int exitCode = WKDCLI.execute(new String[] {
"fetch", "--armor", testCase.getLookupMailAddress()
});
if (testCase.isExpectSuccess()) {
assertEquals(0, exitCode);
} else {
assertNotEquals(0, exitCode);
}
}
}
}

View File

@ -24,4 +24,28 @@ public class TestCase {
this.certificatePath = certificatePath.toString();
this.lookupUri = lookupUri;
}
public boolean isExpectSuccess() {
return expectSuccess;
}
public String getTestTitle() {
return testTitle;
}
public String getTestDescription() {
return testDescription;
}
public String getLookupMailAddress() {
return lookupMailAddress;
}
public String getCertificatePath() {
return certificatePath;
}
public URI getLookupUri() {
return lookupUri;
}
}

View File

@ -4,6 +4,7 @@
package pgp.wkd.test_suite;
import java.util.ArrayList;
import java.util.List;
public class TestSuite {
@ -15,4 +16,12 @@ public class TestSuite {
this.version = version;
this.testCases = testCases;
}
public List<TestCase> getTestCases() {
return new ArrayList<>(testCases);
}
public String getVersion() {
return version;
}
}