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

org.jclouds.compute.domain.TemplateBuilderSpec Maven / Gradle / Ivy

The newest version!
/*
 * 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 org.jclouds.compute.domain;

import static com.google.common.base.Objects.equal;
import static com.google.common.base.Objects.toStringHelper;
import static com.google.common.base.Preconditions.checkArgument;

import java.util.List;

import org.jclouds.compute.options.TemplateOptions;
import org.jclouds.domain.LoginCredentials;
import org.jclouds.domain.LoginCredentials.Builder;
import org.jclouds.javax.annotation.Nullable;

import com.google.common.annotations.Beta;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Objects;
import com.google.common.base.Splitter;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;

/**
 * A specification of a {@link TemplateBuilder} configuration.
 * 
 * 

* {@code TemplateBuilderSpec} supports parsing configuration off of a string, * which makes it especially useful for command-line configuration of a * {@code TemplateBuilder}. * *

* The string syntax is a series of comma-separated keys or key-value pairs, * each corresponding to a {@code TemplateBuilder} method. *

    *
  • {@code hardwareId=[String]}: sets {@link TemplateBuilder#hardwareId}. *
  • {@code minCores=[double]}: sets {@link TemplateBuilder#minCores}. *
  • {@code minRam=[integer]}: sets {@link TemplateBuilder#minRam}. *
  • {@code hypervisorMatches=[String]}: sets * {@link TemplateBuilder#hypervisorMatches}. *
  • {@code imageId=[String]}: sets {@link TemplateBuilder#imageId}. *
  • {@code imageNameMatches=[String]}: sets * {@link TemplateBuilder#imageNameMatches}. *
  • {@code osFamily=[OsFamily]}: sets {@link TemplateBuilder#osFamily}. *
  • {@code osVersionMatches=[String]}: sets * {@link TemplateBuilder#osVersionMatches}. *
  • {@code os64Bit=[boolean]}: sets {@link TemplateBuilder#os64Bit}. *
  • {@code osArchMatches=[String]}: sets * {@link TemplateBuilder#osArchMatches}. *
  • {@code osDescriptionMatches=[String]}: sets * {@link TemplateBuilder#osDescriptionMatches}. *
  • {@code loginUser=[String]}: sets * {@link TemplateOptions#overrideLoginCredentials} parsing password, if colon * delimited. *
  • {@code authenticateSudo=[Boolean]}: sets * {@link TemplateOptions#overrideLoginCredentials}, but only if * {@code loginUser} is set. *
  • {@code locationId=[String]}: sets {@link TemplateBuilder#locationId}. *
* * The set of supported keys will grow as {@code TemplateBuilder} evolves, but * existing keys will never be removed. * *

* Whitespace before and after commas and equal signs is ignored. Keys may not * be repeated. * *

* It is also illegal to use the following combination of keys *

    *
  • {@code hardwareId} and any of *
      *
    • {@code minCores} *
    • {@code minRam} *
    • {@code hypervisorMatches} *
    *
  • {@code imageId} and any of *
      *
    • {@code imageNameMatches} *
    • {@code osFamily} *
    • {@code osVersionMatches} *
    • {@code os64Bit} *
    • {@code osArchMatches} *
    • {@code osDescriptionMatches} *
    *
* *

* {@code TemplateBuilderSpec} does not support configuring * {@code TemplateBuilder} methods with non-value parameters. These must be * configured in code. * *

* A new {@code TemplateBuilder} can be instantiated from a * {@code TemplateBuilderSpec} using * {@link TemplateBuilder#from(TemplateBuilderSpec)} or * {@link TemplateBuilder#from(String)}. * *

* Design inspired by {@link CacheBuilderSpec} * * @author Adrian Cole * @since 1.5 */ @Beta public class TemplateBuilderSpec { /** Parses a single value. */ protected interface ValueParser { void parse(TemplateBuilderSpec spec, String key, @Nullable String value); } /** Splits each key-value pair. */ protected static final Splitter KEYS_SPLITTER = Splitter.on(',').trimResults(); /** Splits the key from the value. */ protected static final Splitter KEY_VALUE_SPLITTER = Splitter.on('=').trimResults(); /** Map of names to ValueParser. */ protected static final ImmutableMap VALUE_PARSERS = ImmutableMap. builder() .put("hardwareId", new HardwareIdParser()) .put("minCores", new MinCoresParser()) .put("minRam", new MinRamParser()) .put("minDisk", new MinDiskParser()) .put("hypervisorMatches", new HypervisorMatchesMatchesParser()) .put("imageId", new ImageIdParser()) .put("imageNameMatches", new ImageNameMatchesParser()) .put("osFamily", new OsFamilyParser()) .put("osVersionMatches", new OsVersionMatchesParser()) .put("os64Bit", new Os64BitParser()) .put("osArchMatches", new OsArchMatchesParser()) .put("osDescriptionMatches", new OsDescriptionMatchesParser()) .put("loginUser", new LoginUserParser()) .put("authenticateSudo", new AuthenticateSudoParser()) .put("locationId", new LocationIdParser()) .build(); @VisibleForTesting String hardwareId; @VisibleForTesting Double minCores; @VisibleForTesting Integer minRam; @VisibleForTesting Double minDisk; @VisibleForTesting String hypervisorMatches; @VisibleForTesting String imageId; @VisibleForTesting String imageNameMatches; @VisibleForTesting OsFamily osFamily; @VisibleForTesting String osVersionMatches; @VisibleForTesting Boolean os64Bit; @VisibleForTesting String osArchMatches; @VisibleForTesting String osDescriptionMatches; @VisibleForTesting String loginUser; @VisibleForTesting Boolean authenticateSudo; @VisibleForTesting String locationId; /** Specification; used for toParseableString(). */ // transient in case people using serializers don't want this to show up protected transient String specification; protected TemplateBuilderSpec() { // we want serializers like Gson to work w/o using sun.misc.Unsafe, // prohibited in GAE. This also implies fields are not final. // see http://code.google.com/p/jclouds/issues/detail?spec=925 } protected TemplateBuilderSpec(String specification) { this.specification = specification; } /** * Creates a TemplateBuilderSpec from a string. * * @param templateBuilderSpecification * the string form */ public static TemplateBuilderSpec parse(String templateBuilderSpecification) { TemplateBuilderSpec spec = new TemplateBuilderSpec(templateBuilderSpecification); if (!templateBuilderSpecification.isEmpty()) { for (String keyValuePair : KEYS_SPLITTER.split(templateBuilderSpecification)) { List keyAndValue = ImmutableList.copyOf(KEY_VALUE_SPLITTER.split(keyValuePair)); checkArgument(!keyAndValue.isEmpty(), "blank key-value pair"); checkArgument(keyAndValue.size() <= 2, "key-value pair %s with more than one equals sign", keyValuePair); // Find the ValueParser for the current key. String key = keyAndValue.get(0); ValueParser valueParser = VALUE_PARSERS.get(key); checkArgument(valueParser != null, "unknown key %s", key); String value = keyAndValue.size() == 1 ? null : keyAndValue.get(1); valueParser.parse(spec, key, value); } } return spec; } /** * Returns a TemplateBuilder configured according to this instance's * specification. * @param templateOptions */ public TemplateBuilder copyTo(TemplateBuilder builder, TemplateOptions templateOptions) { if (hardwareId != null) { builder.hardwareId(hardwareId); } if (minCores != null) { builder.minCores(minCores); } if (minRam != null) { builder.minRam(minRam); } if (minDisk != null) { builder.minDisk(minDisk); } if (hypervisorMatches != null) { builder.hypervisorMatches(hypervisorMatches); } if (imageId != null) { builder.imageId(imageId); } if (imageNameMatches != null) { builder.imageNameMatches(imageNameMatches); } if (osFamily != null) { builder.osFamily(osFamily); } if (osVersionMatches != null) { builder.osVersionMatches(osVersionMatches); } if (os64Bit != null) { builder.os64Bit(os64Bit); } if (osArchMatches != null) { builder.osArchMatches(osArchMatches); } if (osDescriptionMatches != null) { builder.osDescriptionMatches(osDescriptionMatches); } if (loginUser != null) { Builder loginBuilder = LoginCredentials.builder(); int pos = loginUser.indexOf(':'); if (pos != -1) { loginBuilder.user(loginUser.substring(0, pos)).password(loginUser.substring(pos + 1)); } else loginBuilder.user(loginUser); if (authenticateSudo != null) { loginBuilder.authenticateSudo(authenticateSudo); } LoginCredentials creds = loginBuilder.build(); templateOptions.overrideLoginCredentials(creds); } if (locationId != null) { builder.locationId(locationId); } return builder; } /** * Returns a string that can be used to parse an equivalent * {@code TemplateBuilderSpec}. The order and form of this representation is * not guaranteed, except that reparsing its output will produce a * {@code TemplateBuilderSpec} equal to this instance. */ public String toParsableString() { return specification; } /** * Returns a string representation for this TemplateBuilderSpec instance. The * form of this representation is not guaranteed. */ @Override public String toString() { return toStringHelper(this).addValue(toParsableString()).toString(); } @Override public int hashCode() { return Objects.hashCode(hardwareId, minCores, minRam, hypervisorMatches, imageId, imageNameMatches, osFamily, osVersionMatches, os64Bit, osArchMatches, osDescriptionMatches, loginUser, authenticateSudo, locationId); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (!(obj instanceof TemplateBuilderSpec)) { return false; } TemplateBuilderSpec that = (TemplateBuilderSpec) obj; return equal(hardwareId, that.hardwareId) && equal(minCores, that.minCores) && equal(minRam, that.minRam) && equal(hypervisorMatches, that.hypervisorMatches) && equal(imageId, that.imageId) && equal(imageNameMatches, that.imageNameMatches) && equal(osFamily, that.osFamily) && equal(osVersionMatches, that.osVersionMatches) && equal(os64Bit, that.os64Bit) && equal(osArchMatches, that.osArchMatches) && equal(osDescriptionMatches, that.osDescriptionMatches) && equal(loginUser, that.loginUser) && equal(authenticateSudo, that.authenticateSudo) && equal(locationId, that.locationId); } /** Base class for parsing doubles. */ abstract static class DoubleParser implements ValueParser { protected abstract void parseDouble(TemplateBuilderSpec spec, double value); @Override public void parse(TemplateBuilderSpec spec, String key, String value) { checkArgument(value != null && !value.isEmpty(), "value of key %s omitted", key); try { parseDouble(spec, Double.parseDouble(value)); } catch (NumberFormatException e) { throw new IllegalArgumentException(String.format("key %s value set to %s, must be double", key, value), e); } } } /** Base class for parsing ints. */ abstract static class IntegerParser implements ValueParser { protected abstract void parseInteger(TemplateBuilderSpec spec, int value); @Override public void parse(TemplateBuilderSpec spec, String key, String value) { checkArgument(value != null && !value.isEmpty(), "value of key %s omitted", key); try { parseInteger(spec, Integer.parseInt(value)); } catch (NumberFormatException e) { throw new IllegalArgumentException(String.format("key %s value set to %s, must be integer", key, value), e); } } } /** Base class for parsing enums. */ abstract static class EnumParser> implements ValueParser { private final Class type; protected EnumParser(Class type) { this.type = type; } protected abstract void parseEnum(TemplateBuilderSpec spec, E value); @Override public void parse(TemplateBuilderSpec spec, String key, String value) { checkArgument(value != null && !value.isEmpty(), "value of key %s omitted", key); try { parseEnum(spec, Enum.valueOf(type, value)); } catch (IllegalArgumentException e) { throw new IllegalArgumentException(String.format("key %s value set to %s, must be a name in enum %s", key, value, type), e); } } } /** Base class for parsing strings. */ abstract static class StringParser implements ValueParser { protected abstract void set(TemplateBuilderSpec spec, String value); @Override public void parse(TemplateBuilderSpec spec, String key, String value) { checkArgument(value != null && !value.isEmpty(), "value of key %s omitted", key); set(spec, value); } } /** Base class for parsing booleans. */ abstract static class BooleanParser implements ValueParser { protected abstract void parseBoolean(TemplateBuilderSpec spec, boolean value); @Override public void parse(TemplateBuilderSpec spec, String key, String value) { checkArgument(value != null && !value.isEmpty(), "value of key %s omitted", key); try { parseBoolean(spec, Boolean.parseBoolean(value)); } catch (NumberFormatException e) { throw new IllegalArgumentException(String.format("key %s value set to %s, must be booleans", key, value), e); } } } /** Parse hardwareId */ static class HardwareIdParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.hardwareId == null, "hardware id was already set to ", spec.hardwareId); checkArgument(spec.minCores == null, "min cores was already set to ", spec.minCores); checkArgument(spec.minRam == null, "min ram was already set to ", spec.minRam); checkArgument(spec.hypervisorMatches == null, "hypervisor matches was already set to ", spec.hypervisorMatches); spec.hardwareId = value; } } /** Parse minCores */ static class MinCoresParser extends DoubleParser { @Override protected void parseDouble(TemplateBuilderSpec spec, double value) { checkArgument(spec.minCores == null, "min cores was already set to ", spec.minCores); checkArgument(spec.hardwareId == null, "hardware id was already set to ", spec.hardwareId); spec.minCores = value; } } /** Parse minRam */ static class MinRamParser extends IntegerParser { @Override protected void parseInteger(TemplateBuilderSpec spec, int value) { checkArgument(spec.minRam == null, "min ram was already set to ", spec.minRam); checkArgument(spec.hardwareId == null, "hardware id was already set to ", spec.hardwareId); spec.minRam = value; } } /** Parse minDisk */ static class MinDiskParser extends DoubleParser { @Override protected void parseDouble(TemplateBuilderSpec spec, double value) { checkArgument(spec.minDisk == null, "min disk was already set to ", spec.minDisk); checkArgument(spec.hardwareId == null, "hardware id was already set to ", spec.hardwareId); spec.minDisk = value; } } /** Parse hypervisorMatches */ static class HypervisorMatchesMatchesParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.hypervisorMatches == null, "hypervisor matches was already set to ", spec.hypervisorMatches); checkArgument(spec.hardwareId == null, "hardware id was already set to ", spec.hardwareId); spec.hypervisorMatches = value; } } /** Parse imageId */ static class ImageIdParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); checkArgument(spec.imageNameMatches == null, "image name matches was already set to ", spec.imageNameMatches); checkArgument(spec.osFamily == null, "operating system family was already set to ", spec.osFamily); checkArgument(spec.osVersionMatches == null, "os version matches was already set to ", spec.osVersionMatches); checkArgument(spec.os64Bit == null, "os 64 bit was already set to ", spec.os64Bit); checkArgument(spec.osArchMatches == null, "os arch matches was already set to ", spec.osArchMatches); checkArgument(spec.osDescriptionMatches == null, "os description matches was already set to ", spec.osDescriptionMatches); spec.imageId = value; } } /** Parse imageNameMatches */ static class ImageNameMatchesParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.imageNameMatches == null, "image name matches was already set to ", spec.imageNameMatches); checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); spec.imageNameMatches = value; } } /** Parse osFamily */ static class OsFamilyParser extends EnumParser { protected OsFamilyParser() { super(OsFamily.class); } @Override protected void parseEnum(TemplateBuilderSpec spec, OsFamily value) { checkArgument(spec.osFamily == null, "operating system family was already set to ", spec.osFamily); checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); spec.osFamily = value; } } /** Parse osVersionMatches */ static class OsVersionMatchesParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.osVersionMatches == null, "os version matches was already set to ", spec.osVersionMatches); checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); spec.osVersionMatches = value; } } /** Parse os64Bit */ static class Os64BitParser extends BooleanParser { @Override protected void parseBoolean(TemplateBuilderSpec spec, boolean value) { checkArgument(spec.os64Bit == null, "os 64 bit was already set to ", spec.os64Bit); checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); spec.os64Bit = value; } } /** Parse osArchMatches */ static class OsArchMatchesParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.osArchMatches == null, "os arch matches was already set to ", spec.osArchMatches); checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); spec.osArchMatches = value; } } /** Parse osDescriptionMatches */ static class OsDescriptionMatchesParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.osDescriptionMatches == null, "os description matches was already set to ", spec.osDescriptionMatches); checkArgument(spec.imageId == null, "image id was already set to ", spec.imageId); spec.osDescriptionMatches = value; } } /** Parse loginUser */ static class LoginUserParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.loginUser == null, "login user was already set to ", spec.loginUser); spec.loginUser = value; } } /** Parse authenticateSudo */ static class AuthenticateSudoParser extends BooleanParser { @Override protected void parseBoolean(TemplateBuilderSpec spec, boolean value) { checkArgument(spec.loginUser != null, "login user must be set to use authenticateSudo"); checkArgument(spec.authenticateSudo == null, "authenticate sudo was already set to ", spec.authenticateSudo); spec.authenticateSudo = value; } } /** Parse locationId */ static class LocationIdParser extends StringParser { @Override protected void set(TemplateBuilderSpec spec, String value) { checkArgument(spec.locationId == null, "location id was already set to ", spec.locationId); spec.locationId = value; } } public String getHardwareId() { return hardwareId; } public Double getMinCores() { return minCores; } public Integer getMinRam() { return minRam; } public String getHypervisorMatches() { return hypervisorMatches; } public String getImageId() { return imageId; } public String getImageNameMatches() { return imageNameMatches; } public OsFamily getOsFamily() { return osFamily; } public String getOsVersionMatches() { return osVersionMatches; } public Boolean getOs64Bit() { return os64Bit; } public String getOsArchMatches() { return osArchMatches; } public String getOsDescriptionMatches() { return osDescriptionMatches; } public String getLoginUser() { return loginUser; } public Boolean getAuthenticateSudo() { return authenticateSudo; } public String getLocationId() { return locationId; } public String getSpecification() { return specification; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy