com.spotify.helios.common.descriptors.JobId Maven / Gradle / Ivy
/*
* Copyright (c) 2014 Spotify AB.
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package com.spotify.helios.common.descriptors;
import com.google.common.base.Function;
import com.google.common.collect.ComparisonChain;
import com.google.common.collect.Ordering;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonValue;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
/**
* Uniquely Identifies Jobs
*
* Has a string representation in JSON of:
*
* "name:version:hashvalue"
*
*
* The hash value is so that if you are talking to multiple clusters, and the job definitions
* are slightly different, even though they have the same name and version, they will still
* be uniquely identifiable. This is most important when executing commands against multiple
* clusters.
*
* Many endpoints taking JobId can take an abbreviated JobId. That is, one without a the final
* colon and hash value.
*/
@JsonSerialize
@JsonIgnoreProperties(ignoreUnknown = true)
public class JobId extends Descriptor implements Comparable {
private final String name;
private final String version;
private final String hash;
/**
* Create a fully qualified job id with name, version and hash.
*
* @param name The name of the job.
* @param version The version of the job.
* @param hash The hash of the job.
*/
public JobId(final String name,
final String version,
final String hash) {
this.name = name;
this.version = version;
this.hash = hash;
}
/**
* Create a new job id with a specific name and version.
*
* @param name The name of the job.
* @param version the version of the job.
*/
public JobId(final String name,
final String version) {
checkNotNull(name, "name");
checkNotNull(version, "version");
checkArgument(!name.isEmpty(), "name is empty");
checkArgument(!version.isEmpty(), "version is empty");
checkArgument(name.indexOf(':') == -1, "name contains colon");
checkArgument(version.indexOf(':') == -1, "version contains colon");
this.name = name;
this.version = version;
this.hash = null;
}
/**
* Private constructor for use by jackson.
*
* @param id The ID of the job.
*/
@JsonCreator
private JobId(final String id) {
final String[] parts = id.split(":");
if (parts.length != 2 && parts.length != 3) {
throw new IllegalArgumentException("Invalid Job id: " + id);
}
this.name = parts[0];
this.version = parts[1];
this.hash = (parts.length == 3) ? parts[2] : null;
}
/**
* Private constructor for use by {@link #parse(String)}
*
* @param name The name of the job.
* @param b
*/
private JobId(final String name, boolean b) {
checkArgument(!checkNotNull(name, "name is null").isEmpty(), "name is empty");
this.name = name;
this.version = null;
this.hash = null;
}
/**
* Parse a job id string.
*
* This parsing method can be used when input is trusted, i.e. failing to parse it indicates
* programming error and not bad input.
* @see #parse(String)
*
* @param id A string representation of the job ID.
* @return The JobId object.
*/
public static JobId fromString(final String id) {
try {
return parse(id);
} catch (JobIdParseException e) {
throw new IllegalArgumentException(e);
}
}
public static final Function FROM_STRING = new Function() {
@Override
public JobId apply(final String s) {
return fromString(s);
}
};
/**
* Parse a job id string.
*
* This parsing method can be used when input is not know to be correct. I.e. when parsing a job
* id supplied by the user in the cli or when parsing a request in the master rest interface.
* @see #fromString(String)
*
* @param id A string representation of the job ID.
* @return The JobId object.
* @throws JobIdParseException If the ID cannot be parsed.
*/
public static JobId parse(final String id) throws JobIdParseException {
final String[] parts = id.split(":");
switch (parts.length) {
case 1:
return new JobId(parts[0], true);
case 2:
return new JobId(parts[0], parts[1]);
case 3:
return new JobId(parts[0], parts[1], parts[2]);
default:
throw new JobIdParseException("Invalid Job id: " + id);
}
}
@Override
@JsonValue
public String toString() {
if (hash == null) {
return name + ":" + version;
} else {
return name + ":" + version + ":" + hash;
}
}
public String toShortString() {
if (hash == null) {
return name + ":" + version;
} else {
return name + ":" + version + ":" + hash.substring(0, 7);
}
}
public String getName() {
return name;
}
public String getVersion() {
return version;
}
public String getHash() {
return hash;
}
@Override
public boolean equals(final Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
final JobId jobId = (JobId) o;
if (hash != null ? !hash.equals(jobId.hash) : jobId.hash != null) {
return false;
}
if (name != null ? !name.equals(jobId.name) : jobId.name != null) {
return false;
}
if (version != null ? !version.equals(jobId.version) : jobId.version != null) {
return false;
}
return true;
}
@Override
public int hashCode() {
int result = name != null ? name.hashCode() : 0;
result = 31 * result + (version != null ? version.hashCode() : 0);
result = 31 * result + (hash != null ? hash.hashCode() : 0);
return result;
}
@Override
public int compareTo(final JobId o) {
return ComparisonChain.start()
.compare(name, o.name)
.compare(version, o.version)
.compare(hash, o.hash, Ordering.natural().nullsFirst())
.result();
}
public static Builder newBuilder() {
return new Builder();
}
public boolean isFullyQualified() {
return name != null && version != null && hash != null && hash.length() == 40;
}
public static class Builder {
private String name;
private String version;
private String hash;
public Builder setName(final String name) {
this.name = name;
return this;
}
public Builder setVersion(final String version) {
this.version = version;
return this;
}
public Builder setHash(final String hash) {
this.hash = hash;
return this;
}
public JobId build() {
if (hash == null) {
return new JobId(name, version);
} else {
return new JobId(name, version, hash);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy