Program Listing for File rpc_util.cpp
↰ Return to documentation for file (src/common/rpc/rpc_util.cpp)
/*
Copyright 2018-2025, Barcelona Supercomputing Center (BSC), Spain
Copyright 2015-2025, Johannes Gutenberg Universitaet Mainz, Germany
This software was partially supported by the
EC H2020 funded project NEXTGenIO (Project ID: 671951, www.nextgenio.eu).
This software was partially supported by the
ADA-FS project under the SPPEXA project funded by the DFG.
This software was partially supported by the
the European Union’s Horizon 2020 JTI-EuroHPC research and
innovation programme, by the project ADMIRE (Project ID: 956748,
admire-eurohpc.eu)
This project was partially promoted by the Ministry for Digital Transformation
and the Civil Service, within the framework of the Recovery,
Transformation and Resilience Plan - Funded by the European Union
-NextGenerationEU.
This file is part of GekkoFS.
GekkoFS is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
GekkoFS is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with GekkoFS. If not, see <https://www.gnu.org/licenses/>.
SPDX-License-Identifier: GPL-3.0-or-later
*/
#include <common/rpc/rpc_util.hpp>
extern "C" {
#include <unistd.h>
#include <sys/socket.h>
#include <netdb.h>
}
#include <system_error>
using namespace std;
namespace gkfs::rpc {
hg_bool_t
bool_to_merc_bool(const bool state) {
return state ? static_cast<hg_bool_t>(HG_TRUE)
: static_cast<hg_bool_t>(HG_FALSE);
}
string
get_my_hostname(bool short_hostname) {
char hostname[1024];
auto ret = gethostname(hostname, 1024);
if(ret == 0) {
string hostname_s(hostname);
if(!short_hostname)
return hostname_s;
// get short hostname
auto pos = hostname_s.find("."s);
if(pos != string::npos)
hostname_s = hostname_s.substr(0, pos);
return hostname_s;
} else
return ""s;
}
#ifdef GKFS_ENABLE_UNUSED_FUNCTIONS
string
get_host_by_name(const string& hostname) {
int err = 0;
struct addrinfo hints;
memset(&hints, 0, sizeof(struct addrinfo));
hints.ai_flags = (AI_V4MAPPED | AI_ADDRCONFIG);
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_RAW;
struct addrinfo* addr = nullptr;
err = getaddrinfo(hostname.c_str(), nullptr, &hints, &addr);
if(err) {
throw runtime_error("Error getting address info for '" + hostname +
"': " + gai_strerror(err));
}
char addr_str[INET6_ADDRSTRLEN];
err = getnameinfo(addr->ai_addr, addr->ai_addrlen, addr_str,
INET6_ADDRSTRLEN, nullptr, 0,
(NI_NUMERICHOST | NI_NOFQDN));
if(err) {
throw runtime_error("Error on getnameinfo(): "s + gai_strerror(err));
}
freeaddrinfo(addr);
return addr_str;
}
#endif
bool
get_bitset(const std::vector<uint8_t>& data, const uint16_t position) {
return (data[(position) / 8] & 1 << ((position) % 8));
}
void
set_bitset(std::vector<uint8_t>& data, const uint16_t position) {
data[(position) / 8] |= 1 << ((position) % 8); // set
}
std::string
base64_encode(const std::vector<uint8_t>& data) {
static const std::string base64_chars =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
std::ostringstream encoded;
uint32_t buffer = 0;
int padding = 0;
for(uint8_t byte : data) {
buffer = (buffer << 8) | byte;
padding += 8;
while(padding >= 6) {
padding -= 6;
encoded << base64_chars[(buffer >> padding) & 0x3F];
}
}
if(padding > 0) {
buffer <<= 6 - padding;
encoded << base64_chars[buffer & 0x3F];
}
while(encoded.str().length() % 4 != 0) {
encoded << '=';
}
return encoded.str();
}
std::vector<uint8_t>
base64_decode(const std::string& encoded) {
static const std::string base64_chars =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
std::vector<uint8_t> data;
uint32_t buffer = 0;
int padding = 0;
size_t count = 0;
for(char c : encoded) {
if(c == '=')
break;
std::size_t value = base64_chars.find(c);
if(value == std::string::npos)
continue;
buffer = (buffer << 6) | value;
padding += 6;
if(padding >= 8) {
padding -= 8;
data.push_back(static_cast<uint8_t>((buffer >> padding) & 0xFF));
count++;
}
}
// Handle padding characters
if(padding > 0 && padding < 6 && (buffer & ((1 << padding) - 1)) == 0) {
// Remove the padding bits
buffer >>= padding;
padding = 0;
data.push_back(static_cast<uint8_t>((buffer >> 8) & 0xFF));
count++;
}
if(count == 0 || padding % 8 != 0)
return {};
return data;
}
std::string
compress_bitset(const std::vector<uint8_t>& bytes) {
return base64_encode(bytes);
}
std::vector<uint8_t>
decompress_bitset(const std::string& compressedString) {
return base64_decode(compressedString);
}
} // namespace gkfs::rpc