All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.bdeploy.bhive.model.Manifest Maven / Gradle / Ivy

Go to download

Public API including dependencies, ready to be used for integrations and plugins.

There is a newer version: 7.4.0
Show newest version
/*
 * Copyright (c) SSI Schaefer IT Solutions GmbH
 */
package io.bdeploy.bhive.model;

import java.io.Serializable;
import java.util.Collections;
import java.util.Map;
import java.util.SortedMap;
import java.util.SortedSet;
import java.util.TreeMap;
import java.util.TreeSet;

import javax.annotation.Generated;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;

import io.bdeploy.bhive.BHiveExecution;
import io.bdeploy.bhive.objects.view.MissingObjectView;
import io.bdeploy.bhive.objects.view.TreeView;
import io.bdeploy.bhive.objects.view.scanner.TreeVisitor;
import io.bdeploy.bhive.op.ManifestExistsOperation;
import io.bdeploy.bhive.op.ManifestLoadOperation;
import io.bdeploy.bhive.op.ManifestRefScanOperation;
import io.bdeploy.bhive.op.ScanOperation;

/**
 * Represents a manifest. A manifest is a top-level object making {@link Tree}s
 * more easy to remember, associating a name and additional labels (meta-data)
 * with it.
 * 

* Note that the labels are NOT part of the manifests ID. */ public class Manifest implements Serializable, Comparable { private static final long serialVersionUID = 1L; private final Key key; private final ObjectId root; private final SortedMap labels; private final SortedSet references; @JsonCreator private Manifest(@JsonProperty("key") Key key, @JsonProperty("root") ObjectId root, @JsonProperty("labels") SortedMap labels, @JsonProperty("references") SortedSet references) { this.key = key; this.root = root; this.labels = labels; this.references = references; } /** * @return the manifest key used to uniquely identify the manifest across hives */ public Key getKey() { return key; } /** * @return the {@link ObjectId} of the root {@link Tree}. Always references a * {@link Tree}. */ public ObjectId getRoot() { return root; } /** * @return additional meta-data associated with the manifest. */ public Map getLabels() { return Collections.unmodifiableMap(labels); } /** * Used to figure out an order of {@link Manifest} insertion during transfers when references * do not yet exist in the target BHive. * * @return the existing set of cached references, regardless of their existence in any BHive. */ SortedSet internalGetCachedReferences() { return references; } /** * Retrieves a set of cached dependencies. This cache is created when inserting the {@link Manifest}. {@link Manifest}s * created by previous versions before the cache existed will return null always. *

* This cache is meant to be used internally in {@link ManifestRefScanOperation} only, not directly by business logic. * * @param hive used to verify existence of referenced manifests. * @param depth up to which depth references should be returned. * @param allowMissing whether it is acceptable to have references to missing manifests. * @return all referenced {@link Manifest}s if cached information is available, null otherwise. */ public SortedMap getCachedReferences(BHiveExecution hive, int depth, boolean allowMissing) { if (references == null) { return null; } SortedMap result = new TreeMap<>(); for (ReferenceKey rk : references) { if (rk.depth > depth) { continue; } if (!allowMissing) { Boolean exists = hive.execute(new ManifestExistsOperation().setManifest(rk.key)); if (!Boolean.TRUE.equals(exists)) { throw new IllegalStateException( "Missing referenced manifest: " + rk.key + ", referenced from " + key + " [" + rk.path + "]"); } } result.put(rk.path, rk.key); } return result; } @Override public int compareTo(Manifest o) { return key.compareTo(o.key); } @Override public String toString() { return "Manifest(" + root + ")"; } @Generated("Eclipse") @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((key == null) ? 0 : key.hashCode()); result = prime * result + ((root == null) ? 0 : root.hashCode()); return result; } @Generated("Eclipse") @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } Manifest other = (Manifest) obj; if (key == null) { if (other.key != null) { return false; } } else if (!key.equals(other.key)) { return false; } if (root == null) { if (other.root != null) { return false; } } else if (!root.equals(other.root)) { return false; } return true; } /** * Builder to create a Manifest. */ public static final class Builder { private final Key key; private ObjectId root; private final SortedMap labels = new TreeMap<>(); /** * @param key the immutable key of the manifest to use. */ public Builder(Key key) { this.key = key; } /** * @param root the root {@link Tree} reference. The {@link ObjectId} MUST * reference a {@link Tree}. * @return this for chaining */ public Builder setRoot(ObjectId root) { this.root = root; return this; } /** * Add a label to the manifest. A label can be any domain/user specific * key/value pair. * * @param key the label key * @param value the label value * @return this for chaining */ public Builder addLabel(String key, String value) { labels.put(key, value); return this; } /** * @return create the immutable manifest. */ public Manifest build(BHiveExecution hive) { return new Manifest(key, root, labels, findReferences(hive)); } private SortedSet findReferences(BHiveExecution hive) { if (hive == null) { return null; } return findNestedReferences(hive, root, "", 0); } private SortedSet findNestedReferences(BHiveExecution hive, ObjectId tree, String path, long depth) { if (hive == null) { return null; } // calculate references directly, and persistently cache them here. SortedSet referenced = new TreeSet<>(); TreeView state = hive.execute(new ScanOperation().setTree(tree).setFollowReferences(false)); state.visit(new TreeVisitor.Builder().onMissing(this::missing).onManifestRef(m -> { referenced.add(new ReferenceKey(path + m.getPathString(), m.getReferenced(), depth + m.getPath().size())); Manifest mf = hive.execute(new ManifestLoadOperation().setManifest(m.getReferenced())); if (mf.references != null) { // manifest has cached references, expand instead of scan for (ReferenceKey rk : mf.references) { // adjust to the current depth and path. referenced.add(new ReferenceKey(path + m.getPathString() + "/" + rk.getPath(), rk.key, depth + m.getPath().size() + rk.depth)); } } else { // need to actually scan the manifest for references. could yield nested manifest references as well. referenced.addAll( findNestedReferences(hive, mf.getRoot(), path + m.getPathString() + "/", depth + m.getPath().size())); } }).build()); return referenced; } private void missing(MissingObjectView m) { throw new IllegalStateException("Missing object: " + m.getElementId() + " at " + m.getPath()); } public Key getKey() { return key; } } /** * Represents the unique ID of a manifest. Consists of a name and a tag. *

* Usually the tag will represent some form of version number, but there are no * assumptions about the actual format. */ public static final class Key implements Serializable, Comparable { private static final long serialVersionUID = 1L; private final String name; private final String tag; /** * Creates a new key which can be used to create new manifests or lookup * existing manifests which match the key. Note that name and tag are used to * name filesystem objects ultimately, so they should not contain characters * which cause issues in filesystems. * * @param name the name, may not contain ':' specifically, and any * filesystem-problematic character in general. * @param tag the tag, may not contain ':' specifically, and any * filesystem-problematic character in general. */ @JsonCreator public Key(@JsonProperty("name") String name, @JsonProperty("tag") String tag) { this.name = name; this.tag = tag; if (name.contains("..")) { throw new IllegalArgumentException( "Manifest key may not contain the '..' character sequence: " + name + " / " + tag); } if (name.contains(":") || tag.contains(":")) { throw new IllegalArgumentException("Manifest key may not contain the ':' character: " + name + " / " + tag); } if (name.contains("\\")) { throw new IllegalArgumentException("Manifest key may not contain the '\\' character: " + name + " / " + tag); } } /** * @return the name of the key'd manifest */ public String getName() { return name; } /** * @return the tag of the key'd manifest */ public String getTag() { return tag; } public String directoryFriendlyName() { return name.replace('/', '-') + "_" + tag; } @Override public int compareTo(Key o) { int x = name.compareTo(o.name); if (x != 0) { return x; } return tag.compareTo(o.tag); } @Override public String toString() { return name + ":" + tag; } /** * Parses a key from it's {@link String} representation. */ public static Key parse(String key) { int indexOf = key.indexOf(':'); if (indexOf < 0) { throw new IllegalArgumentException("invalid manifest key format: " + key); } return new Key(key.substring(0, indexOf), key.substring(indexOf + 1)); } @Generated("Eclipse") @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((name == null) ? 0 : name.hashCode()); result = prime * result + ((tag == null) ? 0 : tag.hashCode()); return result; } @Generated("Eclipse") @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } Key other = (Key) obj; if (name == null) { if (other.name != null) { return false; } } else if (!name.equals(other.name)) { return false; } if (tag == null) { if (other.tag != null) { return false; } } else if (!tag.equals(other.tag)) { return false; } return true; } } public static class ReferenceKey implements Serializable, Comparable { private static final long serialVersionUID = 1L; private final String path; private final Manifest.Key key; private final long depth; @JsonCreator public ReferenceKey(@JsonProperty("path") String path, @JsonProperty("key") Manifest.Key key, @JsonProperty("depth") long depth) { this.path = path; this.key = key; this.depth = depth; } public long getDepth() { return depth; } public String getPath() { return path; } public Manifest.Key getKey() { return key; } @Override public int compareTo(ReferenceKey o) { return path.compareTo(o.path); } @Generated("Eclipse") @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((path == null) ? 0 : path.hashCode()); return result; } @Generated("Eclipse") @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } ReferenceKey other = (ReferenceKey) obj; if (path == null) { if (other.path != null) { return false; } } else if (!path.equals(other.path)) { return false; } return true; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy