diff options
author | Amadornes <amadornes@gmail.com> | 2018-07-11 19:49:47 +0200 |
---|---|---|
committer | Amadornes <amadornes@gmail.com> | 2018-07-11 19:49:47 +0200 |
commit | 5bdc425b044246521849e902290ec8e65812f7be (patch) | |
tree | 1c9fe71a8b6610a5dd8e8ed0204d5f9c8ffc576b /src/shared/java/com | |
parent | e52ed01dd0992f3ebadc716af2687a70be9f325c (diff) | |
download | Artifactural-5bdc425b044246521849e902290ec8e65812f7be.tar.gz Artifactural-5bdc425b044246521849e902290ec8e65812f7be.tar.bz2 Artifactural-5bdc425b044246521849e902290ec8e65812f7be.zip |
Fixed gradle setup for better dependency handling
Diffstat (limited to 'src/shared/java/com')
10 files changed, 496 insertions, 0 deletions
diff --git a/src/shared/java/com/amadornes/artifactural/base/artifact/ArtifactBase.java b/src/shared/java/com/amadornes/artifactural/base/artifact/ArtifactBase.java new file mode 100644 index 0000000..4656285 --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/artifact/ArtifactBase.java @@ -0,0 +1,48 @@ +package com.amadornes.artifactural.base.artifact; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.artifact.ArtifactMetadata; +import com.amadornes.artifactural.api.artifact.ArtifactType; +import com.amadornes.artifactural.api.cache.ArtifactCache; +import com.amadornes.artifactural.api.artifact.ArtifactIdentifier; +import com.amadornes.artifactural.api.transform.ArtifactTransformer; + +public abstract class ArtifactBase implements Artifact { + + private final ArtifactIdentifier identifier; + private final ArtifactType type; + private final ArtifactMetadata metadata; + + ArtifactBase(ArtifactIdentifier identifier, ArtifactType type, ArtifactMetadata metadata) { + this.identifier = identifier; + this.type = type; + this.metadata = metadata; + } + + @Override + public ArtifactIdentifier getIdentifier() { + return identifier; + } + + @Override + public ArtifactType getType() { + return type; + } + + @Override + public ArtifactMetadata getMetadata() { + return metadata; + } + + @Override + public Artifact apply(ArtifactTransformer transformer) { + if (!transformer.appliesTo(this)) return this; + return transformer.transform(this); + } + + @Override + public Artifact cache(ArtifactCache cache, String specifier) { + return cache.store(this, specifier); + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/artifact/ArtifactIdentifierImpl.java b/src/shared/java/com/amadornes/artifactural/base/artifact/ArtifactIdentifierImpl.java new file mode 100644 index 0000000..26acd29 --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/artifact/ArtifactIdentifierImpl.java @@ -0,0 +1,42 @@ +package com.amadornes.artifactural.base.artifact; + +import com.amadornes.artifactural.api.artifact.ArtifactIdentifier; + +public class ArtifactIdentifierImpl implements ArtifactIdentifier { + + private final String group, name, version, classifier, extension; + + public ArtifactIdentifierImpl(String group, String name, String version, String classifier, String extension) { + this.group = group; + this.name = name; + this.version = version; + this.classifier = classifier; + this.extension = extension; + } + + @Override + public String getGroup() { + return group; + } + + @Override + public String getName() { + return name; + } + + @Override + public String getVersion() { + return version; + } + + @Override + public String getClassifier() { + return classifier; + } + + @Override + public String getExtension() { + return extension; + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/artifact/SimpleArtifactMetadata.java b/src/shared/java/com/amadornes/artifactural/base/artifact/SimpleArtifactMetadata.java new file mode 100644 index 0000000..e08cd7b --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/artifact/SimpleArtifactMetadata.java @@ -0,0 +1,57 @@ +package com.amadornes.artifactural.base.artifact; + +import com.amadornes.artifactural.api.artifact.ArtifactMetadata; + +import java.nio.charset.StandardCharsets; +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.util.Base64; +import java.util.LinkedList; +import java.util.stream.Collectors; + +public class SimpleArtifactMetadata implements ArtifactMetadata { + + private final LinkedList<Entry> entries = new LinkedList<>(); + + public SimpleArtifactMetadata() { + } + + private SimpleArtifactMetadata(SimpleArtifactMetadata parent, Entry entry) { + this.entries.addAll(parent.entries); + this.entries.add(entry); + } + + @Override + public ArtifactMetadata with(String key, String value) { + return new SimpleArtifactMetadata(this, new Entry(key, value)); + } + + @Override + public String getHash() { + try { + String str = entries.stream().map(Entry::toString).collect(Collectors.joining(";;")); + MessageDigest digest = MessageDigest.getInstance("SHA-256"); + byte[] hash = digest.digest(str.getBytes(StandardCharsets.UTF_8)); + return Base64.getEncoder().encodeToString(hash); + } catch (NoSuchAlgorithmException ex) { + throw new RuntimeException(ex); + } + } + + private static class Entry { + + private final String key, value; + + private Entry(String key, String value) { + this.key = key; + this.value = value; + } + + @Override + public String toString() { + return '[' + key + ',' + value + ']'; + } + + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/artifact/StreamableArtifact.java b/src/shared/java/com/amadornes/artifactural/base/artifact/StreamableArtifact.java new file mode 100644 index 0000000..4f3df7a --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/artifact/StreamableArtifact.java @@ -0,0 +1,56 @@ +package com.amadornes.artifactural.base.artifact; + +import com.amadornes.artifactural.api.artifact.*; + +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.net.URL; + +public class StreamableArtifact extends ArtifactBase { + + public static Artifact ofJar(ArtifactIdentifier identifier, ArtifactType type, File file) { + return ofStreamable(identifier, type, () -> new FileInputStream(file)); + } + + public static Artifact ofURL(ArtifactIdentifier identifier, ArtifactType type, URL url) { + return ofStreamable(identifier, type, url::openStream); + } + + public static Artifact ofStreamable(ArtifactIdentifier identifier, ArtifactType type, Streamable streamable) { + return new StreamableArtifact(identifier, type, streamable); + } + + private final Streamable streamable; + + private StreamableArtifact(ArtifactIdentifier identifier, ArtifactType type, Streamable streamable) { + this(identifier, type, ArtifactMetadata.empty(), streamable); + } + + private StreamableArtifact(ArtifactIdentifier identifier, ArtifactType type, ArtifactMetadata metadata, Streamable streamable) { + super(identifier, type, metadata); + this.streamable = streamable; + } + + @Override + public Artifact withMetadata(ArtifactMetadata metadata) { + return new StreamableArtifact(getIdentifier(), getType(), metadata, streamable); + } + + @Override + public boolean isPresent() { + try (InputStream is = openStream()) { + is.close(); + return true; + } catch (IOException ex) { + return false; + } + } + + @Override + public InputStream openStream() throws IOException { + return streamable.openStream(); + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/cache/ArtifactCacheBase.java b/src/shared/java/com/amadornes/artifactural/base/cache/ArtifactCacheBase.java new file mode 100644 index 0000000..5f1c240 --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/cache/ArtifactCacheBase.java @@ -0,0 +1,33 @@ +package com.amadornes.artifactural.base.cache; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.cache.ArtifactCache; +import com.amadornes.artifactural.base.artifact.StreamableArtifact; + +import java.io.*; + +abstract class ArtifactCacheBase implements ArtifactCache { + + Artifact doStore(File path, Artifact artifact) { + return StreamableArtifact.ofStreamable(artifact.getIdentifier(), artifact.getType(), () -> stream(path, artifact)) + .withMetadata(artifact.getMetadata()); + } + + private InputStream stream(File path, Artifact artifact) throws IOException { + if (!path.exists()) { + path.getParentFile().mkdirs(); + path.createNewFile(); + FileOutputStream fos = new FileOutputStream(path); + InputStream is = artifact.openStream(); + int read; + byte[] bytes = new byte[256]; + while ((read = is.read(bytes)) > 0) { + fos.write(bytes, 0, read); + } + fos.close(); + is.close(); + } + return new FileInputStream(path); + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/cache/LocatedArtifactCache.java b/src/shared/java/com/amadornes/artifactural/base/cache/LocatedArtifactCache.java new file mode 100644 index 0000000..caa5de7 --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/cache/LocatedArtifactCache.java @@ -0,0 +1,31 @@ +package com.amadornes.artifactural.base.cache; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.artifact.ArtifactIdentifier; + +import java.io.File; + +public class LocatedArtifactCache extends ArtifactCacheBase { + + private final File path; + + public LocatedArtifactCache(File path) { + this.path = path; + } + + @Override + public Artifact store(Artifact artifact, String specifier) { + ArtifactIdentifier identifier = artifact.getIdentifier(); + File cachePath = new File(path.getAbsolutePath() + .replace("${GROUP}", identifier.getGroup()) + .replace("${NAME}", identifier.getName()) + .replace("${VERSION}", identifier.getVersion()) + .replace("${CLASSIFIER}", identifier.getClassifier()) + .replace("${EXTENSION}", identifier.getExtension()) + .replace("${SPECIFIER}", specifier) + .replace("${META_HASH}", artifact.getMetadata().getHash()) + ); + return doStore(cachePath, artifact); + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/repository/ArtifactProviderBuilder.java b/src/shared/java/com/amadornes/artifactural/base/repository/ArtifactProviderBuilder.java new file mode 100644 index 0000000..1f756ec --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/repository/ArtifactProviderBuilder.java @@ -0,0 +1,81 @@ +package com.amadornes.artifactural.base.repository; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.repository.ArtifactProvider; + +import java.util.HashSet; +import java.util.Set; +import java.util.function.Function; +import java.util.function.Predicate; + +public class ArtifactProviderBuilder<S, I> implements ArtifactProvider.Builder<S, I> { + + public static <I> ArtifactProviderBuilder<I, I> begin(Class<I> type) { + return new ArtifactProviderBuilder<>(Function.identity()); + } + + private final Function<S, I> mapper; + private final Set<Predicate<I>> filters = new HashSet<>(); + + private ArtifactProviderBuilder(Function<S, I> mapper) { + this.mapper = mapper; + } + + @Override + public ArtifactProvider.Builder<S, I> filter(Predicate<I> filter) { + filters.add(filter); + return this; + } + + @Override + public <D> ArtifactProvider.Builder<S, D> mapInfo(Function<I, D> mapper) { + if (filters.isEmpty()) { + return new ArtifactProviderBuilder<>(this.mapper.andThen(mapper)); + } + return new ArtifactProviderBuilder<>((S info) -> { + I localInfo = this.mapper.apply(info); + if (localInfo == null) return null; + for (Predicate<I> filter : filters) { + if (!filter.test(localInfo)) { + return null; + } + } + return mapper.apply(localInfo); + }); + } + + @Override + public ArtifactProvider.Builder.Complete<S, I> provide(ArtifactProvider<I> provider) { + return new Complete<>(mapper).provide(provider); + } + + private static class Complete<S, I> implements ArtifactProvider.Builder.Complete<S, I> { + + private final Set<ArtifactProvider<I>> providers = new HashSet<>(); + private final Function<S, I> mapper; + + private Complete(Function<S, I> mapper) { + this.mapper = mapper; + } + + @Override + public Builder.Complete<S, I> provide(ArtifactProvider<I> provider) { + providers.add(provider); + return this; + } + + @Override + public Artifact getArtifact(S info) { + I localInfo = mapper.apply(info); + if (localInfo == null) return Artifact.none(); + + for (ArtifactProvider<I> provider : providers) { + Artifact artifact = provider.getArtifact(localInfo); + if (artifact.isPresent()) return artifact; + } + return Artifact.none(); + } + + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/repository/SimpleRepository.java b/src/shared/java/com/amadornes/artifactural/base/repository/SimpleRepository.java new file mode 100644 index 0000000..7f437f8 --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/repository/SimpleRepository.java @@ -0,0 +1,25 @@ +package com.amadornes.artifactural.base.repository; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.artifact.ArtifactIdentifier; +import com.amadornes.artifactural.api.repository.ArtifactProvider; +import com.amadornes.artifactural.api.repository.Repository; + +public class SimpleRepository implements Repository { + + public static Repository of(ArtifactProvider<ArtifactIdentifier> provider) { + return new SimpleRepository(provider); + } + + private final ArtifactProvider<ArtifactIdentifier> provider; + + private SimpleRepository(ArtifactProvider<ArtifactIdentifier> provider) { + this.provider = provider; + } + + @Override + public Artifact getArtifact(ArtifactIdentifier identifier) { + return provider.getArtifact(identifier); + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/transform/ExclusiveTransformer.java b/src/shared/java/com/amadornes/artifactural/base/transform/ExclusiveTransformer.java new file mode 100644 index 0000000..e16be1f --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/transform/ExclusiveTransformer.java @@ -0,0 +1,77 @@ +package com.amadornes.artifactural.base.transform; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.artifact.ArtifactMetadata; +import com.amadornes.artifactural.api.artifact.ArtifactType; +import com.amadornes.artifactural.api.transform.ArtifactTransformer; +import com.amadornes.artifactural.base.artifact.StreamableArtifact; + +import java.io.IOException; +import java.util.HashSet; +import java.util.Set; +import java.util.function.Predicate; +import java.util.regex.Pattern; +import java.util.stream.Collectors; +import java.util.zip.ZipEntry; +import java.util.zip.ZipInputStream; + +public class ExclusiveTransformer implements ArtifactTransformer { + + public static ExclusiveTransformer of(boolean whitelist, String... filters) { + return new ExclusiveTransformer(whitelist, filters); + } + + private final Set<Pattern> filters = new HashSet<>(); + private final boolean whitelist; + + private ExclusiveTransformer(boolean whitelist, String... filters) { + this.whitelist = whitelist; + for (String s : filters) { + String regex = s + .replaceAll("(?:(^|[^\\w\\*])\\*\\*([^\\w\\*]|$))", "$1.*$2") // ** matches anything + .replaceAll("(?:(^|[^\\w\\*])\\*([^\\w\\*]|$))", "$1[^\\/]*$2"); // * matches anything but / + this.filters.add(Pattern.compile(regex)); + } + } + + @Override + public Artifact transform(Artifact artifact) { + if (!artifact.isPresent()) return Artifact.none(); + + if (artifact.getType() == ArtifactType.BINARY || artifact.getType() == ArtifactType.SOURCE) { + return exclude(artifact); + } else { + return Artifact.none(); + } + } + + @Override + public ArtifactMetadata withInfo(ArtifactMetadata metadata) { + return metadata.with("EXCLUDE", filters.stream().map(Pattern::pattern).collect(Collectors.joining(";"))); + } + + private Artifact exclude(Artifact artifact) { + return StreamableArtifact.ofStreamable(artifact.getIdentifier(), artifact.getType(), + () -> new ZipInputStream(artifact.openStream()) { + @Override + public ZipEntry getNextEntry() throws IOException { + ZipEntry next; + while ((next = super.getNextEntry()) != null) { + if (isAllowed(next.getName())) { + return next; + } + } + return null; + } + }); + } + + private boolean isAllowed(String name) { + if (whitelist) { + return filters.stream().anyMatch(p -> p.asPredicate().test(name)); + } else { + return filters.stream().noneMatch(p -> p.asPredicate().test(name)); + } + } + +} diff --git a/src/shared/java/com/amadornes/artifactural/base/transform/SimpleArtifactPipeline.java b/src/shared/java/com/amadornes/artifactural/base/transform/SimpleArtifactPipeline.java new file mode 100644 index 0000000..625b2e9 --- /dev/null +++ b/src/shared/java/com/amadornes/artifactural/base/transform/SimpleArtifactPipeline.java @@ -0,0 +1,46 @@ +package com.amadornes.artifactural.base.transform; + +import com.amadornes.artifactural.api.artifact.Artifact; +import com.amadornes.artifactural.api.artifact.ArtifactMetadata; +import com.amadornes.artifactural.api.cache.ArtifactCache; +import com.amadornes.artifactural.api.transform.ArtifactPipeline; +import com.amadornes.artifactural.api.transform.ArtifactTransformer; + +import java.util.function.UnaryOperator; + +public class SimpleArtifactPipeline implements ArtifactPipeline { + + public static ArtifactPipeline create() { + return new SimpleArtifactPipeline(); + } + + private static final ArtifactTransformer IDENTITY = ArtifactTransformer.of(UnaryOperator.identity()); + + private ArtifactTransformer transformer = IDENTITY; + + private SimpleArtifactPipeline() { + } + + @Override + public ArtifactPipeline apply(ArtifactTransformer transformer) { + this.transformer = this.transformer.andThen(transformer); + return this; + } + + @Override + public ArtifactPipeline cache(ArtifactCache cache, String specifier) { + transformer = transformer.andThen(ArtifactTransformer.of(artifact -> cache.store(artifact, specifier))); + return this; + } + + @Override + public Artifact transform(Artifact artifact) { + return transformer.transform(artifact); + } + + @Override + public ArtifactMetadata withInfo(ArtifactMetadata metadata) { + return transformer.withInfo(metadata); + } + +} |