/* -------------------------------------------------------------------------- */ /* Copyright 2002-2019, OpenNebula Project, OpenNebula Systems */ /* */ /* 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. */ /* -------------------------------------------------------------------------- */ #ifndef ATTRIBUTE_H_ #define ATTRIBUTE_H_ #include #include #include #include #include "NebulaUtil.h" using namespace std; /** * Attribute base class for name-value pairs. This class provides a generic * interface to implement */ class Attribute { public: Attribute(const string& aname):attribute_name(aname) { transform ( attribute_name.begin(), attribute_name.end(), attribute_name.begin(), (int(*)(int))toupper); // FIX Attribute name if it does not conform XML element // naming conventions int size = attribute_name.size(); if ((size >0 && !(isalpha(aname[0]) || aname[0] == '_')) || (size >=3 && (aname[0]=='X' && aname[1]=='M' && aname[2]=='L'))) { attribute_name.insert(0,"ONE_"); } }; virtual ~Attribute(){}; enum AttributeType { SIMPLE = 0, VECTOR = 1 }; /** * Gets the name of the attribute. * @return the attribute name */ const string& name() const { return attribute_name; }; /** * Marshall the attribute in a single string. The string MUST be freed * by the calling function. * @return a string (allocated in the heap) holding the attribute value. */ virtual string * marshall(const char * _sep = 0) const = 0; /** * Write the attribute using a simple XML format. The string MUST be freed * by the calling function. * @return a string (allocated in the heap) holding the attribute value. */ virtual void to_xml(std::ostringstream& s) const = 0; virtual void to_json(std::ostringstream& s) const = 0; virtual void to_token(std::ostringstream& s) const = 0; /** * Builds a new attribute from a string. */ virtual void unmarshall(const string& sattr, const char * _sep = 0) = 0; /** * Returns the attribute type */ virtual AttributeType type() = 0; /** * Clones the current attribute */ virtual Attribute* clone() const = 0; /** * Encrypt all secret attributes */ virtual void encrypt(const string& one_key, const set& eas) {}; /** * Decrypt all secret attributes */ virtual void decrypt(const string& one_key, const set& eas) {}; protected: /** * The attribute name. */ string attribute_name; }; /* -------------------------------------------------------------------------- */ /* -------------------------------------------------------------------------- */ /** * The SingleAttribute class represents a simple attribute in the form * NAME = VALUE. */ class SingleAttribute : public Attribute { public: SingleAttribute(const string& name):Attribute(name){}; SingleAttribute(const string& name, const string& value): Attribute(name),attribute_value(value){}; SingleAttribute(const SingleAttribute& sa):Attribute(sa.attribute_name) { attribute_value = sa.attribute_value; }; ~SingleAttribute(){}; /** * Returns the attribute value, a string. */ const string& value() const { return attribute_value; }; /** * Marshall the attribute in a single string. The string MUST be freed * by the calling function. * @return a string (allocated in the heap) holding the attribute value. */ string * marshall(const char * _sep = 0) const { string * rs = new string; *rs = attribute_value; return rs; }; /** * Write the attribute using a simple XML format: * * attribute_value * * @paran s the stream to write the attribute. */ void to_xml(std::ostringstream& s) const { s << "<" << attribute_name << ">" << one_util::escape_xml(attribute_value) << ""; } void to_json(std::ostringstream& s) const { one_util::escape_json(attribute_value, s); } void to_token(std::ostringstream& s) const { if (attribute_name.empty() || attribute_value.empty()) { return; } one_util::escape_token(attribute_name, s); s << "="; one_util::escape_token(attribute_value, s); s << std::endl; } /** * Builds a new attribute from a string. */ void unmarshall(const string& sattr, const char * _sep = 0) { attribute_value = sattr; }; /** * Replaces the attribute value from a string. */ void replace(const string& sattr) { attribute_value = sattr; }; /** * Returns the attribute type */ AttributeType type() { return SIMPLE; }; /** * Clones the current attribute */ Attribute* clone() const { return new SingleAttribute(*this); }; /** * Encrypt all secret attributes */ virtual void encrypt(const string& one_key, const set& eas) override; /** * Decrypt all secret attributes */ virtual void decrypt(const string& one_key, const set& eas) override; private: string attribute_value; }; /* -------------------------------------------------------------------------- */ /* -------------------------------------------------------------------------- */ /** * The VectorAttribute class represents an array attribute in the form * NAME = [ VAL_NAME_1=VAL_VALUE_1,...,VAL_NAME_N=VAL_VALUE_N]. */ class VectorAttribute : public Attribute { public: VectorAttribute(const string& name):Attribute(name){}; VectorAttribute(const string& name,const map& value): Attribute(name),attribute_value(value){}; VectorAttribute(const VectorAttribute& va):Attribute(va.attribute_name) { attribute_value = va.attribute_value; }; VectorAttribute(const VectorAttribute* va):Attribute(va->attribute_name) { attribute_value = va->attribute_value; }; ~VectorAttribute(){}; /** * Returns the attribute value, a string. */ const map& value() const { return attribute_value; }; /** * Returns the string value * @param name of the attribute * * @return the value of the attribute if found, empty otherwise */ string vector_value(const string& name) const; /** * Returns the value of the given element of the VectorAttribute * * @param name of the attribute * @param value, not set if the element is not found of has invalid type * * @return 0 on success, -1 otherwise */ template int vector_value(const string& name, T& value) const { map::const_iterator it; it = attribute_value.find(name); if ( it == attribute_value.end() ) { return -1; } if ( it->second.empty() ) { return -1; } istringstream iss(it->second); iss >> value; if (iss.fail() || !iss.eof()) { return -1; } return 0; } int vector_value(const string& name, string& value) const; int vector_value(const string& name, bool& value) const; /** * Returns the value of the given element of the VectorAttribute * * @param name Name of the attribute * @param value Integer value, if an error occurred the string returned is * empty and value is not set * * @return the value in string form on success, "" otherwise */ template string vector_value_str(const string& name, T& value) const { map::const_iterator it; it = attribute_value.find(name); if ( it == attribute_value.end() ) { return ""; } if ( it->second.empty() ) { return ""; } istringstream iss(it->second); iss >> value; if (iss.fail() || !iss.eof()) { return ""; } return it->second; } /** * Marshall the attribute in a single string. The string MUST be freed * by the calling function. The string is in the form: * "VAL_NAME_1=VAL_VALUE_1,...,VAL_NAME_N=VAL_VALUE_N". * @return a string (allocated in the heap) holding the attribute value. */ string * marshall(const char * _sep = 0) const; /** * Write the attribute using a simple XML format: * * * val_value_1 * ... * val_value_n * * * The string MUST be freed by the calling function. * @return a string (allocated in the heap) holding the attribute value. */ void to_xml(std::ostringstream& s) const; void to_json(std::ostringstream& s) const; void to_token(std::ostringstream& s) const; /** * Builds a new attribute from a string of the form: * "VAL_NAME_1=VAL_VALUE_1,...,VAL_NAME_N=VAL_VALUE_N". */ void unmarshall(const string& sattr, const char * _sep = 0); /** * Replace the value of the given attribute with the provided map */ void replace(const map& attr); /** * The attributes from vattr will be copied to this vector * @param attr Vector attribute to merge * @param replace True to replace existing values, false to copy values * only if they don't exist in this vector attribute */ void merge(VectorAttribute* vattr, bool replace); /** * Replace the value of the given vector attribute */ template void replace(const string& name, T value) { ostringstream oss; oss << value; replace(name, oss.str()); } void replace(const string& name, bool value) { if (value == true) { replace(name, "YES"); } else { replace(name, "NO"); } } void replace(const string& name, const string& value); /** * Removes given the vector attribute * @param name of the vector attribute */ void remove(const string& name); /** * Returns the attribute type */ AttributeType type() { return VECTOR; }; /** * Clones the current attribute */ VectorAttribute* clone() const { return new VectorAttribute(*this); }; /** * Clear the vector attribute values */ void clear() { attribute_value.clear(); } /** * @return true if the vector attribute contains no values */ bool empty() { return attribute_value.empty(); } /** * Encrypt all secret attributes */ virtual void encrypt(const string& one_key, const set& eas) override; /** * Decrypt all secret attributes */ virtual void decrypt(const string& one_key, const set& eas) override; private: static const char * magic_sep; static const int magic_sep_size; map attribute_value; }; #endif /*ATTRIBUTE_H_*/