mirror of
https://github.com/OpenNebula/one.git
synced 2025-01-20 14:03:36 +03:00
03add6d9ac
* Update smoke test checkout action (cherry picked from commit 466d336d7e264b22e19576e42e39252921f1f0a6)
369 lines
8.8 KiB
C++
369 lines
8.8 KiB
C++
/* -------------------------------------------------------------------------- */
|
|
/* Copyright 2002-2023, 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 DRIVER_MESSAGE_H
|
|
#define DRIVER_MESSAGE_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
|
|
#include "EnumString.h"
|
|
#include "SSLUtil.h"
|
|
|
|
/**
|
|
* This class represents a generic message used by the Monitoring Protocol.
|
|
* The structure of the message is:
|
|
*
|
|
* +------+-----+--------+-----+-----+-----+----+-----+---------+------+
|
|
* | TYPE | ' ' | STATUS | ' ' | OID | ' ' | TS | ' ' | PAYLOAD | '\n' |
|
|
* +------+-----+--------+-----+-----+-----+----+-----+---------+------+
|
|
*
|
|
* TYPE String (non-blanks) identifying the message type
|
|
* ' ' A single white space to separate fields
|
|
* STATUS String (non-blanks), status of the message depends on message
|
|
* type, could contain result of operation ("SUCCESS" or "FAILURE")
|
|
* OID Number, id of affected object, -1 if not object related
|
|
* TS timestamp for the message in epoch.
|
|
* PAYLOAD of the message XML base64 encoded
|
|
* '\n' End of message delimiter
|
|
*
|
|
*/
|
|
template<typename E, //Enum class for the message types
|
|
bool encode = false, //Payload is base64 encoded
|
|
bool compress = false, //Payload is compressed
|
|
bool encrypt = false, //Payload is encrypted
|
|
bool has_timestamp = false> //Message includes timestamp
|
|
class Message
|
|
{
|
|
public:
|
|
using msg_enum = E;
|
|
|
|
Message() = default;
|
|
|
|
Message(E type, std::string &&status, int oid, const std::string& msg_payload);
|
|
|
|
/**
|
|
* Parse the Message from an input string
|
|
* @param input string with the message
|
|
*/
|
|
int parse_from(const std::string& input);
|
|
|
|
/**
|
|
* Writes this object to the given string
|
|
*/
|
|
int write_to(std::string& out) const;
|
|
|
|
/**
|
|
* Writes this object to the given file descriptor
|
|
*/
|
|
int write_to(int fd) const;
|
|
|
|
/**
|
|
* Writes this object to the given output stream
|
|
*/
|
|
int write_to(std::ostream& oss) const;
|
|
|
|
/**
|
|
*
|
|
*/
|
|
E type() const
|
|
{
|
|
return _type;
|
|
}
|
|
|
|
void type(E t)
|
|
{
|
|
_type = t;
|
|
}
|
|
|
|
/**
|
|
* Returns type of the message as string
|
|
*/
|
|
const std::string& type_str() const
|
|
{
|
|
return _type_str._to_str(_type);
|
|
}
|
|
|
|
static const std::string& type_str(E t)
|
|
{
|
|
return _type_str._to_str(t);
|
|
}
|
|
|
|
/**
|
|
* Status of the message, can't contain blanks.
|
|
* Depends on message type, could contain result of
|
|
* operation ("SUCCESS" or "FAILURE")
|
|
* Default value is "-"
|
|
*/
|
|
const std::string& status() const
|
|
{
|
|
return _status;
|
|
}
|
|
|
|
void status(const std::string& status)
|
|
{
|
|
_status = status;
|
|
}
|
|
|
|
/**
|
|
* Object ID, -1 if not object related
|
|
*/
|
|
int oid() const
|
|
{
|
|
return _oid;
|
|
}
|
|
|
|
void oid(int oid)
|
|
{
|
|
_oid = oid;
|
|
}
|
|
|
|
/**
|
|
* Message data, could be empty
|
|
*/
|
|
const std::string& payload() const
|
|
{
|
|
return _payload;
|
|
}
|
|
|
|
std::string payload64() const
|
|
{
|
|
std::string buffer;
|
|
|
|
ssl_util::base64_decode(_payload, buffer);
|
|
|
|
return buffer;
|
|
}
|
|
|
|
void payload(const std::string& p)
|
|
{
|
|
_payload = p;
|
|
}
|
|
|
|
/**
|
|
* Message timestamp, optional
|
|
*/
|
|
time_t timestamp() const
|
|
{
|
|
static_assert(has_timestamp == true, "Timestamp disabled");
|
|
return _timestamp;
|
|
}
|
|
|
|
void timestamp(time_t ts)
|
|
{
|
|
static_assert(has_timestamp == true, "Timestamp disabled");
|
|
_timestamp = ts;
|
|
}
|
|
|
|
private:
|
|
/**
|
|
* Message fields
|
|
*/
|
|
E _type = E::UNDEFINED;
|
|
|
|
std::string _status = std::string("-");
|
|
|
|
int _oid = -1;
|
|
|
|
std::string _payload;
|
|
|
|
time_t _timestamp = 0;
|
|
|
|
static const EString<E> _type_str;
|
|
};
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
/* -------------------------------------------------------------------------- */
|
|
/* Message Template Implementation */
|
|
/* -------------------------------------------------------------------------- */
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
template<typename E, bool compress, bool encode, bool encrypt, bool has_timestamp>
|
|
Message<E, compress, encode, encrypt, has_timestamp>
|
|
::Message(E type, std::string &&status, int oid, const std::string& payload)
|
|
: _type(type)
|
|
, _status(std::move(status))
|
|
, _oid(oid)
|
|
, _payload(payload)
|
|
{
|
|
}
|
|
|
|
template<typename E, bool compress, bool encode, bool encrypt, bool has_timestamp>
|
|
int Message<E, compress, encode, encrypt, has_timestamp>
|
|
::parse_from(const std::string& input)
|
|
{
|
|
std::istringstream is(input);
|
|
std::string buffer;
|
|
|
|
if (!is.good())
|
|
{
|
|
goto error;
|
|
}
|
|
|
|
is >> buffer;
|
|
|
|
_type = _type_str._from_str(buffer);
|
|
|
|
if ( !is.good() || _type == E::UNDEFINED )
|
|
{
|
|
goto error;
|
|
}
|
|
|
|
buffer.clear();
|
|
|
|
is >> _status;
|
|
|
|
is >> _oid >> std::ws;
|
|
|
|
if (has_timestamp)
|
|
{
|
|
is >> _timestamp >> std::ws;
|
|
}
|
|
|
|
getline(is, buffer);
|
|
|
|
if (buffer.empty())
|
|
{
|
|
_payload.clear();
|
|
return 0;
|
|
}
|
|
|
|
if (encode)
|
|
{
|
|
ssl_util::base64_decode(buffer, _payload);
|
|
|
|
if ( compress && ssl_util::zlib_decompress(_payload, _payload) == -1 )
|
|
{
|
|
goto error;
|
|
}
|
|
|
|
if ( encrypt && ssl_util::is_rsa_set() )
|
|
{
|
|
if ( ssl_util::rsa_private_decrypt(_payload, _payload) == -1 )
|
|
{
|
|
// Accept also not encrypted messages
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_payload = buffer;
|
|
}
|
|
|
|
return 0;
|
|
|
|
error:
|
|
_type = E::UNDEFINED;
|
|
_payload = input;
|
|
return -1;
|
|
}
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
template<typename E, bool compress, bool encode, bool encrypt, bool has_timestamp>
|
|
int Message<E, compress, encode, encrypt, has_timestamp>
|
|
::write_to(std::string& out) const
|
|
{
|
|
out.clear();
|
|
|
|
out = _type_str._to_str(_type);
|
|
out += ' ';
|
|
out += _status;
|
|
out += ' ';
|
|
out += std::to_string(_oid);
|
|
out += ' ';
|
|
|
|
if (has_timestamp)
|
|
{
|
|
out += std::to_string(_timestamp);
|
|
out += ' ';
|
|
}
|
|
|
|
if (!_payload.empty())
|
|
{
|
|
if (encode)
|
|
{
|
|
std::string msg;
|
|
|
|
msg = _payload;
|
|
|
|
if (compress && ssl_util::zlib_compress(msg, msg) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
if ( ssl_util::base64_encode(msg, msg) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
out += msg;
|
|
}
|
|
else
|
|
{
|
|
out += _payload;
|
|
}
|
|
}
|
|
|
|
out += '\n';
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
template<typename E, bool compress, bool encode, bool encrypt, bool has_timestamp>
|
|
int Message<E, compress, encode, encrypt, has_timestamp>
|
|
::write_to(int fd) const
|
|
{
|
|
std::string out;
|
|
|
|
if ( write_to(out) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
::write(fd, (const void *) out.c_str(), out.size());
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
template<typename E, bool compress, bool encode, bool encrypt, bool has_timestamp>
|
|
int Message<E, compress, encode, encrypt, has_timestamp>
|
|
::write_to(std::ostream& oss) const
|
|
{
|
|
std::string out;
|
|
|
|
if ( write_to(out) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
oss << out;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#endif /*DRIVER_MESSAGE_H_*/
|