1
0
mirror of https://github.com/OpenNebula/one.git synced 2025-01-25 06:03:36 +03:00
one/include/Message.h
Pavel Czerný 2fe396ae15
L #-: Add Astyle linting
* Add AStyle to smoke tests
* Format C++ files using AStyle
2024-06-03 11:40:24 +02:00

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_*/