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

org.onosproject.net.config.Config Maven / Gradle / Ivy

There is a newer version: 2.7.0
Show newest version
/*
 * Copyright 2015 Open Networking Laboratory
 *
 * Licensed 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.onosproject.net.config;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.google.common.annotations.Beta;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterators;
import com.google.common.collect.Lists;
import org.onlab.packet.IpAddress;
import org.onlab.packet.IpPrefix;
import org.onlab.packet.MacAddress;
import org.onosproject.net.ConnectPoint;

import java.util.Collection;
import java.util.List;
import java.util.Set;
import java.util.function.Function;

import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkState;

/**
 * Base abstraction of a configuration facade for a specific subject. Derived
 * classes should keep all state in the specified JSON tree as that is the
 * only state that will be distributed or persisted; this class is merely
 * a facade for interacting with a particular facet of configuration on a
 * given subject.
 *
 * @param  type of subject
 */
@Beta
public abstract class Config {

    private static final String TRUE_LITERAL = "true";
    private static final String FALSE_LITERAL = "false";

    protected S subject;
    protected String key;

    protected JsonNode node;
    protected ObjectNode object;
    protected ArrayNode array;
    protected ObjectMapper mapper;

    protected ConfigApplyDelegate delegate;

    /**
     * Indicator of whether a configuration JSON field is required.
     */
    public enum FieldPresence {
        /**
         * Signifies that config field is an optional one.
         */
        OPTIONAL,

        /**
         * Signifies that config field is mandatory.
         */
        MANDATORY
    }

    /**
     * Initializes the configuration behaviour with necessary context.
     *
     * @param subject  configuration subject
     * @param key      configuration key
     * @param node     JSON node where configuration data is stored
     * @param mapper   JSON object mapper
     * @param delegate delegate context, or null for detached configs.
     */
    public final void init(S subject, String key, JsonNode node, ObjectMapper mapper,
                     ConfigApplyDelegate delegate) {
        this.subject = checkNotNull(subject, "Subject cannot be null");
        this.key = key;
        this.node = checkNotNull(node, "Node cannot be null");
        this.object = node instanceof ObjectNode ? (ObjectNode) node : null;
        this.array = node instanceof ArrayNode ? (ArrayNode) node : null;
        this.mapper = checkNotNull(mapper, "Mapper cannot be null");
        this.delegate = delegate;
    }

    /**
     * Indicates whether or not the backing JSON node contains valid data.
     * 

* Default implementation returns true. * Subclasses are expected to override this with their own validation. * Implementations are free to throw a RuntimeException if data is invalid. * *

* * @return true if the data is valid; false otherwise * @throws RuntimeException if configuration is invalid or completely foobar */ public boolean isValid() { // Derivatives should use the provided set of predicates to test // validity of their fields, e.g.: // isString(path) // isBoolean(path) // isNumber(path, [min, max]) // isDecimal(path, [min, max]) // isMacAddress(path) // isIpAddress(path) // isIpPrefix(path) // isConnectPoint(path) return true; } /** * Returns the specific subject to which this configuration pertains. * * @return configuration subject */ public S subject() { return subject; } /** * Returns the configuration key. This is primarily aimed for use in * composite JSON trees in external representations and has no bearing on * the internal behaviours. * * @return configuration key */ public String key() { return key; } /** * Returns the JSON node that contains the configuration data. * * @return JSON node backing the configuration */ public JsonNode node() { return node; } /** * Applies any configuration changes made via this configuration. * * Not effective for detached configs. */ public void apply() { checkState(delegate != null, "Cannot apply detached config"); delegate.onApply(this); } // Miscellaneous helpers for interacting with JSON /** * Gets the specified property as a string. * * @param name property name * @param defaultValue default value if property not set * @return property value or default value */ protected String get(String name, String defaultValue) { return object.path(name).asText(defaultValue); } /** * Sets the specified property as a string or clears it if null value given. * * @param name property name * @param value new value or null to clear the property * @return self */ protected Config setOrClear(String name, String value) { if (value != null) { object.put(name, value); } else { object.remove(name); } return this; } /** * Gets the specified property as a boolean. * * @param name property name * @param defaultValue default value if property not set * @return property value or default value */ protected boolean get(String name, boolean defaultValue) { return object.path(name).asBoolean(defaultValue); } /** * Clears the specified property. * * @param name property name * @return self */ protected Config clear(String name) { object.remove(name); return this; } /** * Sets the specified property as a boolean or clears it if null value given. * * @param name property name * @param value new value or null to clear the property * @return self */ protected Config setOrClear(String name, Boolean value) { if (value != null) { object.put(name, value.booleanValue()); } else { object.remove(name); } return this; } /** * Gets the specified property as an integer. * * @param name property name * @param defaultValue default value if property not set * @return property value or default value */ protected int get(String name, int defaultValue) { return object.path(name).asInt(defaultValue); } /** * Sets the specified property as an integer or clears it if null value given. * * @param name property name * @param value new value or null to clear the property * @return self */ protected Config setOrClear(String name, Integer value) { if (value != null) { object.put(name, value.intValue()); } else { object.remove(name); } return this; } /** * Gets the specified property as a long. * * @param name property name * @param defaultValue default value if property not set * @return property value or default value */ protected long get(String name, long defaultValue) { return object.path(name).asLong(defaultValue); } /** * Sets the specified property as a long or clears it if null value given. * * @param name property name * @param value new value or null to clear the property * @return self */ protected Config setOrClear(String name, Long value) { if (value != null) { object.put(name, value.longValue()); } else { object.remove(name); } return this; } /** * Gets the specified property as a double. * * @param name property name * @param defaultValue default value if property not set * @return property value or default value */ protected double get(String name, double defaultValue) { return object.path(name).asDouble(defaultValue); } /** * Sets the specified property as a double or clears it if null value given. * * @param name property name * @param value new value or null to clear the property * @return self */ protected Config setOrClear(String name, Double value) { if (value != null) { object.put(name, value.doubleValue()); } else { object.remove(name); } return this; } /** * Gets the specified property as an enum. * * @param name property name * @param defaultValue default value if property not set * @param enumClass the enum class * @param type of enum * @return property value or default value */ protected > E get(String name, E defaultValue, Class enumClass) { return Enum.valueOf(enumClass, object.path(name).asText(defaultValue.toString())); } /** * Sets the specified property as a double or clears it if null value given. * * @param name property name * @param value new value or null to clear the property * @param type of enum * @return self */ protected Config setOrClear(String name, E value) { if (value != null) { object.put(name, value.toString()); } else { object.remove(name); } return this; } /** * Gets the specified array property as a list of items. * * @param name property name * @param function mapper from string to item * @param type of item * @return list of items */ protected List getList(String name, Function function) { List list = Lists.newArrayList(); ArrayNode arrayNode = (ArrayNode) object.path(name); arrayNode.forEach(i -> list.add(function.apply(i.asText()))); return list; } /** * Gets the specified array property as a list of items. * * @param name property name * @param function mapper from string to item * @param defaultValue default value if property not set * @param type of item * @return list of items */ protected List getList(String name, Function function, List defaultValue) { List list = Lists.newArrayList(); JsonNode jsonNode = object.path(name); if (jsonNode.isMissingNode()) { return defaultValue; } ArrayNode arrayNode = (ArrayNode) jsonNode; arrayNode.forEach(i -> list.add(function.apply(i.asText()))); return list; } /** * Sets the specified property as an array of items in a given collection or * clears it if null is given. * * @param name propertyName * @param collection collection of items * @param type of items * @return self */ protected Config setOrClear(String name, Collection collection) { if (collection == null) { object.remove(name); } else { ArrayNode arrayNode = mapper.createArrayNode(); collection.forEach(i -> arrayNode.add(i.toString())); object.set(name, arrayNode); } return this; } /** * Indicates whether only the specified fields are present in the backing JSON. * * @param allowedFields allowed field names * @return true if all allowedFields are present; false otherwise */ protected boolean hasOnlyFields(String... allowedFields) { return hasOnlyFields(object, allowedFields); } /** * Indicates whether only the specified fields are present in a particular * JSON object. * * @param node node whose fields to check * @param allowedFields allowed field names * @return true if all allowedFields are present; false otherwise */ protected boolean hasOnlyFields(ObjectNode node, String... allowedFields) { Set fields = ImmutableSet.copyOf(allowedFields); return !Iterators.any(node.fieldNames(), f -> !fields.contains(f)); } /** * Indicates whether the specified field holds a valid MAC address. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid MAC */ protected boolean isMacAddress(String field, FieldPresence presence) { JsonNode node = object.path(field); return isValid(node, presence, node.isTextual() && MacAddress.valueOf(node.asText()) != null); } /** * Indicates whether the specified field of a particular node holds a valid * MAC address. * * @param objectNode JSON node * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid MAC */ protected boolean isMacAddress(ObjectNode objectNode, String field, FieldPresence presence) { JsonNode node = objectNode.path(field); return isValid(node, presence, node.isTextual() && MacAddress.valueOf(node.asText()) != null); } /** * Indicates whether the specified field holds a valid IP address. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid IP */ protected boolean isIpAddress(String field, FieldPresence presence) { return isIpAddress(object, field, presence); } /** * Indicates whether the specified field of a particular node holds a valid * IP address. * * @param objectNode node from whom to access the field * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid IP */ protected boolean isIpAddress(ObjectNode objectNode, String field, FieldPresence presence) { JsonNode node = objectNode.path(field); return isValid(node, presence, node.isTextual() && IpAddress.valueOf(node.asText()) != null); } /** * Indicates whether the specified field holds a valid IP prefix. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid IP * prefix */ protected boolean isIpPrefix(String field, FieldPresence presence) { return isIpPrefix(object, field, presence); } /** * Indicates whether the specified field of a particular node holds a valid * IP prefix. * * @param objectNode node from whom to access the field * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid IP * prefix */ protected boolean isIpPrefix(ObjectNode objectNode, String field, FieldPresence presence) { JsonNode node = objectNode.path(field); return isValid(node, presence, node.isTextual() && IpPrefix.valueOf(node.asText()) != null); } /** * Indicates whether the specified field holds a valid connect point string. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid * connect point string representation */ protected boolean isConnectPoint(String field, FieldPresence presence) { return isConnectPoint(object, field, presence); } /** * Indicates whether the specified field of a particular node holds a valid * connect point string. * * @param objectNode JSON node * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid * connect point string representation */ protected boolean isConnectPoint(ObjectNode objectNode, String field, FieldPresence presence) { JsonNode node = objectNode.path(field); return isValid(node, presence, node.isTextual() && ConnectPoint.deviceConnectPoint(node.asText()) != null); } /** * Indicates whether the specified field holds a valid string value. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param pattern optional regex pattern * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid string */ protected boolean isString(String field, FieldPresence presence, String... pattern) { return isString(object, field, presence, pattern); } /** * Indicates whether the specified field on a particular node holds a valid * string value. * * @param objectNode JSON node * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param pattern optional regex pattern * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid string */ protected boolean isString(ObjectNode objectNode, String field, FieldPresence presence, String... pattern) { JsonNode node = objectNode.path(field); return isValid(node, presence, node.isTextual() && (pattern.length > 0 && node.asText().matches(pattern[0]) || pattern.length < 1)); } /** * Indicates whether the specified field holds a valid number. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param minMax optional min/max values * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid */ protected boolean isNumber(String field, FieldPresence presence, long... minMax) { JsonNode node = object.path(field); return isValid(node, presence, node.isNumber() && (minMax.length > 0 && minMax[0] <= node.asLong() || minMax.length < 1) && (minMax.length > 1 && minMax[1] > node.asLong() || minMax.length < 2)); } /** * Indicates whether the specified field holds a valid number. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param minMax optional min/max values * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid */ protected boolean isNumber(String field, FieldPresence presence, double... minMax) { JsonNode node = object.path(field); return isValid(node, presence, node.isNumber() && (minMax.length > 0 && minMax[0] <= node.asDouble() || minMax.length < 1) && (minMax.length > 1 && minMax[1] > node.asDouble() || minMax.length < 2)); } /** * Indicates whether the specified field holds a valid integer. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param minMax optional min/max values * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid */ protected boolean isIntegralNumber(String field, FieldPresence presence, long... minMax) { return isIntegralNumber(object, field, presence, minMax); } /** * Indicates whether the specified field of a particular node holds a valid * integer. * * @param objectNode JSON node * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param minMax optional min/max values * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid */ protected boolean isIntegralNumber(ObjectNode objectNode, String field, FieldPresence presence, long... minMax) { JsonNode node = objectNode.path(field); return isValid(node, presence, n -> { long number = (node.isIntegralNumber()) ? n.asLong() : Long.parseLong(n.asText()); return (minMax.length > 0 && minMax[0] <= number || minMax.length < 1) && (minMax.length > 1 && minMax[1] > number || minMax.length < 2); }); } /** * Indicates whether the specified field holds a valid decimal number. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @param minMax optional min/max values * @return true if valid; false otherwise * @throws IllegalArgumentException if field is present, but not valid */ protected boolean isDecimal(String field, FieldPresence presence, double... minMax) { JsonNode node = object.path(field); return isValid(node, presence, (node.isDouble() || node.isFloat()) && (minMax.length > 0 && minMax[0] <= node.asDouble() || minMax.length < 1) && (minMax.length > 1 && minMax[1] > node.asDouble() || minMax.length < 2)); } /** * Indicates whether the specified field holds a valid boolean value. * * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise */ protected boolean isBoolean(String field, FieldPresence presence) { return isBoolean(object, field, presence); } /** * Indicates whether the specified field of a particular node holds a valid * boolean value. * * @param objectNode JSON object node * @param field JSON field name * @param presence specifies if field is optional or mandatory * @return true if valid; false otherwise */ protected boolean isBoolean(ObjectNode objectNode, String field, FieldPresence presence) { JsonNode node = objectNode.path(field); return isValid(node, presence, node.isBoolean() || (node.isTextual() && isBooleanString(node.asText()))); } /** * Indicates whether a string holds a boolean literal value. * * @param str string to test * @return true if the string contains "true" or "false" (case insensitive), * otherwise false */ private boolean isBooleanString(String str) { return str.equalsIgnoreCase(TRUE_LITERAL) || str.equalsIgnoreCase(FALSE_LITERAL); } /** * Indicates whether the node is present and of correct value or not * mandatory and absent. * * @param node JSON node * @param presence specifies if field is optional or mandatory * @param correctValue true if the value is correct * @return true if the field is as expected */ private boolean isValid(JsonNode node, FieldPresence presence, boolean correctValue) { return isValid(node, presence, n -> correctValue); } /** * Indicates whether the node is present and of correct value or not * mandatory and absent. * * @param node JSON node * @param presence specified if field is optional or mandatory * @param validationFunction function which can be used to verify if the * node has the correct value * @return true if the field is as expected */ private boolean isValid(JsonNode node, FieldPresence presence, Function validationFunction) { boolean isMandatory = presence == FieldPresence.MANDATORY; if (isMandatory && validationFunction.apply(node)) { return true; } if (!isMandatory && (node.isNull() || node.isMissingNode())) { return true; } return validationFunction.apply(node); } }