Skip to content

Commit 181af19

Browse files
committedApr 6, 2022
Add vks-java-cli Command Line Frontend
1 parent 352f6d3 commit 181af19

File tree

7 files changed

+313
-1
lines changed

7 files changed

+313
-1
lines changed
 

‎settings.gradle

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,5 @@
44

55
rootProject.name = 'VKS-Java'
66

7-
include 'vks-java'
7+
include 'vks-java',
8+
'vks-java-cli'

‎vks-java-cli/build.gradle

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
5+
plugins {
6+
id 'application'
7+
}
8+
9+
group 'org.pgpainless'
10+
11+
repositories {
12+
mavenCentral()
13+
}
14+
15+
dependencies {
16+
testImplementation "org.junit.jupiter:junit-jupiter-api:$junitVersion"
17+
testRuntimeOnly "org.junit.jupiter:junit-jupiter-engine:$junitVersion"
18+
19+
// VKS-Java
20+
implementation project(":vks-java")
21+
22+
// CLI
23+
implementation "info.picocli:picocli:4.6.3"
24+
}
25+
26+
application {
27+
mainClass = 'pgp.vks.client.cli.VKSCLI'
28+
}
29+
30+
test {
31+
useJUnitPlatform()
32+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
5+
package pgp.vks.client.cli;
6+
7+
import pgp.vks.client.Get;
8+
import pgp.vks.client.VKS;
9+
import picocli.CommandLine;
10+
11+
import java.io.IOException;
12+
import java.io.InputStream;
13+
import java.net.MalformedURLException;
14+
15+
@CommandLine.Command(name = "get", description = "Retrieve an OpenPGP certificate from the key server")
16+
public class GetCmd implements Runnable {
17+
18+
@CommandLine.Mixin
19+
VKSCLI.KeyServerMixin keyServerMixin;
20+
21+
@CommandLine.ArgGroup(exclusive = true, multiplicity = "1")
22+
Exclusive by;
23+
24+
static class Exclusive {
25+
@CommandLine.Option(names = {"-f", "--by-fingerprint"}, description = "Retrieve a key by its fingerprint (NOT prefixed with '0x')")
26+
String fingerprint;
27+
28+
@CommandLine.Option(names = {"-i", "--by-keyid"}, description = "Retrieve a key by its decimal key ID or that of one of its subkeys.")
29+
Long keyId;
30+
31+
@CommandLine.Option(names = {"-e", "--by-email"}, description = "Retrieve a key by email address.")
32+
String email;
33+
}
34+
35+
public void run() {
36+
VKS vks;
37+
try {
38+
vks = keyServerMixin.parent.getApi();
39+
} catch (MalformedURLException e) {
40+
throw new AssertionError(e);
41+
}
42+
43+
Get get = vks.get();
44+
InputStream inputStream = null;
45+
try {
46+
if (by.fingerprint != null) {
47+
inputStream = get.byFingerprint(by.fingerprint);
48+
} else if (by.keyId != null) {
49+
inputStream = get.byKeyId(by.keyId);
50+
} else if (by.email != null) {
51+
inputStream = get.byEmail(by.email);
52+
} else {
53+
throw new IllegalArgumentException("Missing --by-* option.");
54+
}
55+
56+
int read;
57+
byte[] buf = new byte[4096];
58+
while ((read = inputStream.read(buf)) != -1) {
59+
System.out.write(buf, 0, read);
60+
}
61+
62+
} catch (IOException e) {
63+
throw new AssertionError(e);
64+
} finally {
65+
if (inputStream != null) {
66+
try {
67+
inputStream.close();
68+
} catch (IOException e) {
69+
throw new AssertionError(e);
70+
}
71+
}
72+
}
73+
}
74+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
5+
package pgp.vks.client.cli;
6+
7+
import pgp.vks.client.RequestVerify;
8+
import pgp.vks.client.VKS;
9+
import picocli.CommandLine;
10+
11+
import java.io.IOException;
12+
import java.net.MalformedURLException;
13+
import java.util.Arrays;
14+
import java.util.List;
15+
16+
@CommandLine.Command(name = "request-verification", description = "Request verification for unverified user-ids")
17+
public class RequestVerificationCmd implements Runnable {
18+
19+
@CommandLine.Mixin
20+
VKSCLI.KeyServerMixin keyServerMixin;
21+
22+
@CommandLine.Option(names = {"-t", "--token"}, description = "Access token. Can be retrieved by uploading the certificate.",
23+
required = true, arity = "1", paramLabel = "TOKEN")
24+
String token;
25+
26+
@CommandLine.Option(names = {"-l", "--locale"}, description = "Locale for the verification mail")
27+
List<String> locale = Arrays.asList("en_US", "en_GB");
28+
29+
@CommandLine.Option(names = {"-e", "--email"}, description = "Email addresses to request a verification mail for", required = true, arity = "1..*")
30+
String[] addresses = new String[0];
31+
32+
33+
@Override
34+
public void run() {
35+
VKS vks;
36+
try {
37+
vks = keyServerMixin.parent.getApi();
38+
} catch (MalformedURLException e) {
39+
throw new AssertionError(e);
40+
}
41+
42+
RequestVerify requestVerify = vks.requestVerification();
43+
try {
44+
RequestVerify.Response response = requestVerify
45+
.forEmailAddresses(addresses)
46+
.execute(token, locale);
47+
48+
System.out.println("Verification E-Mails for key " + response.getKeyFingerprint() + " have been sent.");
49+
System.out.println("Token: " + response.getToken());
50+
System.out.println("Status:");
51+
for (String address : response.getStatus().keySet()) {
52+
System.out.println("\t" + address + "\t" + response.getStatus().get(address));
53+
}
54+
} catch (IOException e) {
55+
throw new AssertionError(e);
56+
}
57+
}
58+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
5+
package pgp.vks.client.cli;
6+
7+
import pgp.vks.client.RequestVerify;
8+
import pgp.vks.client.Status;
9+
import pgp.vks.client.Upload;
10+
import pgp.vks.client.VKS;
11+
import pgp.vks.client.exception.CertCannotBePublishedException;
12+
import picocli.CommandLine;
13+
14+
import java.io.IOException;
15+
import java.net.MalformedURLException;
16+
import java.util.ArrayList;
17+
import java.util.List;
18+
19+
@CommandLine.Command(name = "upload", description = "Upload an OpenPGP certificate to the key server")
20+
public class UploadCmd implements Runnable {
21+
22+
@CommandLine.Mixin
23+
VKSCLI.KeyServerMixin keyServerMixin;
24+
25+
@CommandLine.Option(names = {"-r", "--request-verification"},
26+
description = "Request verification mails for unpublished email addresses")
27+
boolean requestVerification;
28+
29+
public void run() {
30+
VKS vks;
31+
try {
32+
vks = keyServerMixin.parent.getApi();
33+
} catch (MalformedURLException e) {
34+
throw new AssertionError(e);
35+
}
36+
37+
Upload upload = vks.upload();
38+
try {
39+
Upload.Response response = upload.cert(System.in);
40+
41+
// Unpublished mail addresses
42+
List<String> unpublished = new ArrayList<>();
43+
int maxMailLen = 0;
44+
for (String address : response.getStatus().keySet()) {
45+
Status status = response.getStatus().get(address);
46+
if (address.length() > maxMailLen) {
47+
maxMailLen = address.length();
48+
}
49+
if (status != Status.published && status != Status.revoked) {
50+
unpublished.add(address);
51+
}
52+
}
53+
54+
System.out.println("Uploaded key " + response.getKeyFingerprint());
55+
System.out.println("Token: " + response.getToken());
56+
57+
if (!requestVerification || unpublished.isEmpty()) {
58+
System.out.println("Status:");
59+
for (String address : response.getStatus().keySet()) {
60+
Status status = response.getStatus().get(address);
61+
System.out.format("%-" + maxMailLen + "s %s\n", address, status);
62+
}
63+
return;
64+
}
65+
66+
RequestVerify.Response verifyResponse = vks.requestVerification().forEmailAddresses(unpublished.toArray(new String[0]))
67+
.execute(response.getToken());
68+
System.out.println("Status:");
69+
for (String address : verifyResponse.getStatus().keySet()) {
70+
Status status = response.getStatus().get(address);
71+
System.out.format("%-" + maxMailLen + "s %s\n", address, status);
72+
}
73+
} catch (CertCannotBePublishedException e) {
74+
throw new AssertionError(e.getMessage());
75+
} catch (IOException e) {
76+
throw new AssertionError(e);
77+
}
78+
}
79+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
5+
package pgp.vks.client.cli;
6+
7+
import pgp.vks.client.VKS;
8+
import pgp.vks.client.VKSImpl;
9+
import picocli.CommandLine;
10+
11+
import java.net.MalformedURLException;
12+
13+
@CommandLine.Command(
14+
subcommands = {
15+
CommandLine.HelpCommand.class,
16+
GetCmd.class,
17+
UploadCmd.class,
18+
RequestVerificationCmd.class
19+
}
20+
)
21+
public class VKSCLI {
22+
23+
String keyServer;
24+
25+
public static void main(String[] args) {
26+
int exitCode = execute(args);
27+
if (exitCode != 0) {
28+
System.exit(exitCode);
29+
}
30+
}
31+
32+
public static int execute(String[] args) {
33+
return new CommandLine(VKSCLI.class)
34+
.setExitCodeExceptionMapper(new CommandLine.IExitCodeExceptionMapper() {
35+
@Override
36+
public int getExitCode(Throwable exception) {
37+
return 1;
38+
}
39+
})
40+
.setCommandName("vkscli")
41+
.execute(args);
42+
}
43+
44+
public VKS getApi() throws MalformedURLException {
45+
return new VKSImpl(keyServer);
46+
}
47+
48+
public static class KeyServerMixin {
49+
50+
@CommandLine.ParentCommand
51+
VKSCLI parent;
52+
53+
@CommandLine.Option(names = "--key-server",
54+
description = "Address of the Verifying Key Server.\nDefaults to 'https://keys.openpgp.org'",
55+
paramLabel = "KEYSERVER")
56+
public void setKeyServer(String keyServer) {
57+
parent.keyServer = keyServer;
58+
}
59+
}
60+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
// SPDX-FileCopyrightText: 2022 Paul Schaub <vanitasvitae@fsfe.org>
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
5+
/**
6+
* Command Line Interface for VKS-Java.
7+
*/
8+
package pgp.vks.client.cli;

0 commit comments

Comments
 (0)
Please sign in to comment.