aboutsummaryrefslogtreecommitdiff
path: root/src/support/lombok/publish/PublishToBucket.java
diff options
context:
space:
mode:
authorReinier Zwitserloot <r.zwitserloot@projectlombok.org>2023-02-03 02:21:20 +0100
committerReinier Zwitserloot <r.zwitserloot@projectlombok.org>2023-02-03 02:21:20 +0100
commit756f0019c3f7df0438f1ad4eb0b9c13670fd3a24 (patch)
tree9fe60b4b4b6e2e3476f1e4912d5dff780ef33b15 /src/support/lombok/publish/PublishToBucket.java
parent930cc2410525f8605868fecc2373603ef3d1472b (diff)
downloadlombok-756f0019c3f7df0438f1ad4eb0b9c13670fd3a24.tar.gz
lombok-756f0019c3f7df0438f1ad4eb0b9c13670fd3a24.tar.bz2
lombok-756f0019c3f7df0438f1ad4eb0b9c13670fd3a24.zip
[build] Uploading edge repo to cloudflare R2 bucket straight from ant now possible
Diffstat (limited to 'src/support/lombok/publish/PublishToBucket.java')
-rw-r--r--src/support/lombok/publish/PublishToBucket.java159
1 files changed, 159 insertions, 0 deletions
diff --git a/src/support/lombok/publish/PublishToBucket.java b/src/support/lombok/publish/PublishToBucket.java
new file mode 100644
index 00000000..509a3e91
--- /dev/null
+++ b/src/support/lombok/publish/PublishToBucket.java
@@ -0,0 +1,159 @@
+package lombok.publish;
+
+import java.io.IOException;
+import java.net.URI;
+import java.nio.file.DirectoryStream;
+import java.nio.file.Files;
+import java.nio.file.NoSuchFileException;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+import java.util.ArrayList;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.stream.Collectors;
+
+import software.amazon.awssdk.auth.credentials.AwsBasicCredentials;
+import software.amazon.awssdk.regions.Region;
+import software.amazon.awssdk.services.s3.S3Client;
+import software.amazon.awssdk.services.s3.model.Delete;
+import software.amazon.awssdk.services.s3.model.DeleteObjectsRequest;
+import software.amazon.awssdk.services.s3.model.ListObjectsV2Request;
+import software.amazon.awssdk.services.s3.model.ListObjectsV2Response;
+import software.amazon.awssdk.services.s3.model.ObjectIdentifier;
+import software.amazon.awssdk.services.s3.model.PutObjectRequest;
+import software.amazon.awssdk.services.s3.model.S3Object;
+
+public class PublishToBucket {
+ private static final boolean DEBUG = false;
+
+ private static final class AppException extends Exception {
+ AppException(String msg) {
+ super(msg);
+ }
+ }
+
+ public static void main(String[] args) {
+ try {
+ if (args.length != 4) throw new AppException("4 args required: [path to creds file] [path to file root to upload] [target dir in bucket] [delete files to create a perfect copy or not]");
+ boolean delete;
+ if (args[3].equalsIgnoreCase("true")) delete = true;
+ else if (args[3].equalsIgnoreCase("false")) delete = false;
+ else throw new AppException("4th arg must be 'true' or 'false'");
+ new PublishToBucket().go(args[0], args[1], args[2], delete);
+ System.exit(0);
+ } catch (AppException e) {
+ System.err.println(e.getMessage());
+ System.exit(1);
+ }
+ }
+
+ private URI endpoint;
+ private String bucketName;
+ private AwsBasicCredentials creds;
+
+ /**
+ * @param credsPath path to the creds file; first line in that file is the access key, second line is the secret.
+ * @param rootPath path to a directory; this directory is replicated into the bucket.
+ */
+ private void go(String credsPath, String rootPath, String bucketDir, boolean delete) throws AppException {
+ readCreds(Paths.get(credsPath));
+
+ S3Client s3 = S3Client.builder()
+ .endpointOverride(endpoint)
+ .region(Region.of("auto"))
+ .credentialsProvider(() -> creds)
+ .build();
+
+ ListObjectsV2Response objList = s3.listObjectsV2(ListObjectsV2Request.builder()
+ .bucket(bucketName)
+ .prefix(bucketDir + "/")
+ .build());
+
+ Set<String> inBucketBeforeUpload = new HashSet<String>();
+ for (S3Object obj : objList.contents()) inBucketBeforeUpload.add(obj.key());
+
+ dbg("Already in bucket:\n" +
+ inBucketBeforeUpload.stream().map(x -> " " + x + "\n").collect(Collectors.joining()) +
+ (inBucketBeforeUpload.isEmpty() ? " (Nothing)\n" : ""));
+
+ try {
+ go0(s3, inBucketBeforeUpload, bucketDir + "/", Paths.get(rootPath));
+ } catch (IOException e) {
+ throw new AppException("I/O exception uploading: " + e.getClass() + ": " + e.getMessage());
+ }
+
+ if (delete) {
+ dbg("Uploads complete. Files to delete:\n" +
+ inBucketBeforeUpload.stream().map(x -> " " + x + "\n").collect(Collectors.joining()) +
+ (inBucketBeforeUpload.isEmpty() ? " (Nothing)\n" : ""));
+
+ if (!inBucketBeforeUpload.isEmpty()) {
+ List<ObjectIdentifier> objsToDelete = new ArrayList<ObjectIdentifier>();
+ for (String key : inBucketBeforeUpload) {
+ objsToDelete.add(ObjectIdentifier.builder().key(key).build());
+ }
+ s3.deleteObjects(DeleteObjectsRequest.builder()
+ .bucket(bucketName)
+ .delete(Delete.builder().objects(objsToDelete).build())
+ .build());
+ dbg("Deletion completed");
+ }
+ }
+ }
+
+ private static void dbg(String msg) {
+ if (DEBUG) System.out.println(msg);
+ }
+
+ private void go0(S3Client s3, Set<String> inBucketBeforeUpload, String prefix, Path tgt) throws IOException {
+ try (DirectoryStream<Path> ds = Files.newDirectoryStream(tgt)) {
+ for (Path child : ds) {
+ if (Files.isDirectory(child)) {
+ go0(s3, inBucketBeforeUpload, prefix + child.getFileName().toString() + "/", child);
+ continue;
+ }
+
+ String key = prefix + child.getFileName().toString();
+ PutObjectRequest req = PutObjectRequest.builder()
+ .bucket(bucketName)
+ .key(key)
+ .build();
+ s3.putObject(req, child);
+ boolean overwrote = inBucketBeforeUpload.remove(key);
+ dbg("Uploaded: " + key + (overwrote ? " (overwrote)" : ""));
+ }
+ }
+ }
+
+ private static final String LINE_DESCRIPTIONS = "accessKey/secretKey/endpoint/bucket";
+ private void readCreds(Path path) throws AppException {
+ try {
+ List<String> lines = Files.readAllLines(path);
+ String accessKey = null, secretKey = null, endPoint = null, bucketName = null;
+ for (String line : lines) {
+ int idx = line.indexOf('#');
+ if (idx != -1) line = line.substring(0, idx);
+ line = line.trim();
+ if (line.isEmpty()) continue;
+ if (accessKey == null) { accessKey = line; continue; }
+ if (secretKey == null) { secretKey = line; continue; }
+ if (endPoint == null) { endPoint = line; continue; }
+ if (bucketName == null) { bucketName = line; continue; }
+ throw new AppException("Too many lines in " + path.toAbsolutePath() + " - only 4 expected: " + LINE_DESCRIPTIONS);
+ }
+ if (bucketName == null) throw new AppException("Expected 3 lines in " + path.toAbsolutePath() + ": " + LINE_DESCRIPTIONS);
+ creds = AwsBasicCredentials.create(accessKey, secretKey);
+ endpoint = URI.create(endPoint);
+ this.bucketName = bucketName;
+ } catch (NoSuchFileException e) {
+ throw new AppException("File with bucket endpoint + credentials is not available. Make file " + path.toAbsolutePath() + "; it should contain something like: \n" +
+ "123456789abcdef0123456789abcdef0 # this is the access key\n" +
+ "123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0 # this is the secret\n" +
+ "https://12345.r2.cloudflarestorage.com # this is the endpoint\n" +
+ "lombok-data # this is the bucket name");
+ } catch (IOException e) {
+ throw new AppException("I/O issue reading creds file " + path.toAbsolutePath() + ": " + e.getClass() + ": " + e.getMessage());
+ }
+ }
+}