1
0
mirror of https://github.com/VCMP-SqMod/SqMod.git synced 2025-01-19 12:07:13 +01:00
SqMod/vendor/DPP/include/dpp/utility.h

495 lines
16 KiB
C
Raw Normal View History

2023-03-23 20:20:44 +02:00
/************************************************************************************
*
* D++, A Lightweight C++ library for Discord
*
* Copyright 2021 Craig Edwards and D++ contributors
* (https://github.com/brainboxdotcc/DPP/graphs/contributors)
*
* 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.
*
************************************************************************************/
#pragma once
#include <dpp/export.h>
#include <dpp/snowflake.h>
#include <dpp/misc-enum.h>
#include <string>
#include <vector>
#include <unordered_map>
#include <map>
#include <functional>
#ifndef MAX_CND_IMAGE_SIZE
#define MAX_CDN_IMAGE_SIZE 4096
#endif
#ifndef MIN_CDN_IMAGE_SIZE
#define MIN_CDN_IMAGE_SIZE 16
#endif
/**
* @brief The main namespace for D++ functions, classes and types
*/
namespace dpp {
/**
* @brief Utility helper functions, generally for logging, running programs, time/date manipulation, etc
*/
namespace utility {
/**
* @brief Timestamp formats for dpp::utility::timestamp()
*
* @note These values are the actual character values specified by the Discord API
* and should not be changed unless the Discord API changes the specification!
* They have been sorted into numerical order of their ASCII value to keep C++ happy.
*/
enum time_format : uint8_t {
/// "20 April 2021" - Long Date
tf_long_date = 'D',
/// "Tuesday, 20 April 2021 16:20" - Long Date/Time
tf_long_datetime = 'F',
/// "2 months ago" - Relative Time
tf_relative_time = 'R',
/// "16:20:30" - Long Time
tf_long_time = 'T',
/// "20/04/2021" - Short Date
tf_short_date = 'd',
/// "20 April 2021 16:20" - Short Date/Time
tf_short_datetime = 'f',
/// "16:20" - Short Time
tf_short_time = 't',
};
/**
* @brief The base URL for CDN content such as profile pictures and guild icons.
*/
const std::string cdn_host = "https://cdn.discordapp.com";
/**
* @brief Callback for the results of a command executed via dpp::utility::exec
*/
typedef std::function<void(const std::string& output)> cmd_result_t;
/**
* @brief Run a commandline program asynchronously. The command line program
* is spawned in a separate std::thread, and when complete, its output from
* stdout is passed to the callback function in its string parameter. For example
* ```
* dpp::utility::exec("/bin/ls", {"-al"}, [](const std::string& output) {
* std::cout << "Output of 'ls -al': " << output << "\n";
* });
* ```
*
* @param cmd The command to run.
* @param parameters Command line parameters. Each will be escaped using `std::quoted`.
* @param callback The callback to call on completion.
*/
void DPP_EXPORT exec(const std::string& cmd, std::vector<std::string> parameters = {}, cmd_result_t callback = {});
/**
* @brief Return a mentionable timestamp (used in a message). These timestamps will display the given timestamp in the user's timezone and locale.
*
* @param ts Time stamp to convert
* @param tf Format of timestamp using dpp::utility::time_format
* @return std::string The formatted timestamp
*/
std::string DPP_EXPORT timestamp(time_t ts, time_format tf = tf_short_datetime);
/**
* @brief Returns current date and time
*
* @return std::string Current date and time in "Y-m-d H:M:S" format
*/
std::string DPP_EXPORT current_date_time();
/**
* @brief Convert a dpp::loglevel enum value to a string
*
* @param in log level to convert
* @return std::string string form of log level
*/
std::string DPP_EXPORT loglevel(dpp::loglevel in);
/**
* @brief Store a 128 bit icon hash (profile picture, server icon etc)
* as a 128 bit binary value made of two uint64_t.
* Has a constructor to build one from a string, and a method to fetch
* the value back in string form.
*/
struct DPP_EXPORT iconhash {
uint64_t first; //!< High 64 bits
uint64_t second; //!< Low 64 bits
/**
* @brief Construct a new iconhash object
* @param _first Leftmost portion of the hash value
* @param _second Rightmost portion of the hash value
*/
iconhash(uint64_t _first = 0, uint64_t _second = 0);
/**
* @brief Construct a new iconhash object
*/
iconhash(const iconhash&);
/**
* @brief Destroy the iconhash object
*/
~iconhash();
/**
* @brief Construct a new iconhash object
*
* @param hash String hash to construct from.
* Must contain a 32 character hex string.
*
* @throws std::length_error if the provided
* string is not exactly 32 characters long.
*/
iconhash(const std::string &hash);
/**
* @brief Assign from std::string
*
* @param assignment string to assign from.
*
* @throws std::length_error if the provided
* string is not exactly 32 characters long.
*/
iconhash& operator=(const std::string &assignment);
/**
* @brief Check if one iconhash is equal to another
*
* @param other other iconhash to compare
* @return True if the iconhash objects match
*/
bool operator==(const iconhash& other) const;
/**
* @brief Change value of iconhash object
*
* @param hash String hash to change to.
* Must contain a 32 character hex string.
*
* @throws std::length_error if the provided
* string is not exactly 32 characters long.
*/
void set(const std::string &hash);
/**
* @brief Convert iconhash back to 32 character
* string value.
*
* @return std::string Hash value
*/
std::string to_string() const;
};
/**
* @brief Return the current time with fractions of seconds.
* This is a unix epoch time with the fractional seconds part
* after the decimal place.
*
* @return double time with fractional seconds
*/
double DPP_EXPORT time_f();
/**
* @brief Returns true if D++ was built with voice support
*
* @return bool True if voice support is compiled in (libsodium/libopus)
*/
bool DPP_EXPORT has_voice();
/**
* @brief Convert a byte count to display value
*
* @param c number of bytes
* @return std::string display value suffixed with M, G, T where necessary
*/
std::string DPP_EXPORT bytes(uint64_t c);
/**
* @brief A class used to represent an uptime in hours, minutes,
* seconds and days, with helper functions to convert from time_t
* and display as a string.
*/
struct DPP_EXPORT uptime {
uint16_t days; //!< Number of days
uint8_t hours; //!< Number of hours
uint8_t mins; //!< Number of minutes
uint8_t secs; //!< Number of seconds
/**
* @brief Construct a new uptime object
*/
uptime();
/**
* @brief Construct a new uptime object
*
* @param diff A time_t to initialise the object from
*/
uptime(time_t diff);
/**
* @brief Construct a new uptime object
*
* @param diff A time_t to initialise the object from
*/
uptime(double diff);
/**
* @brief Get uptime as string
*
* @return std::string Uptime as string
*/
std::string to_string() const;
/**
* @brief Get uptime as seconds
*
* @return uint64_t Uptime as seconds
*/
uint64_t to_secs() const;
/**
* @brief Get uptime as milliseconds
*
* @return uint64_t Uptime as milliseconds
*/
uint64_t to_msecs() const;
};
/**
* @brief Convert doubles to RGB for sending in embeds
*
* @param red red value, between 0 and 1 inclusive
* @param green green value, between 0 and 1 inclusive
* @param blue blue value, between 0 and 1 inclusive
* @return uint32_t returned integer colour value
*/
uint32_t DPP_EXPORT rgb(double red, double green, double blue);
/**
* @brief Convert ints to RGB for sending in embeds
*
* @param red red value, between 0 and 255 inclusive
* @param green green value, between 0 and 255 inclusive
* @param blue blue value, between 0 and 255 inclusive
* @return uint32_t returned integer colour value
*/
uint32_t DPP_EXPORT rgb(int red, int green, int blue);
/**
* @brief Convert doubles to CMYK for sending in embeds
*
* @param c cyan value, between 0 and 1 inclusive
* @param m magenta value, between 0 and 1 inclusive
* @param y yellow value, between 0 and 1 inclusive
* @param k key (black) value, between 0 and 1 inclusive
* @return uint32_t returned integer colour value
*/
uint32_t DPP_EXPORT cmyk(double c, double m, double y, double k);
/**
* @brief Convert ints to CMYK for sending in embeds
*
* @param c cyan value, between 0 and 255 inclusive
* @param m magenta value, between 0 and 255 inclusive
* @param y yellow value, between 0 and 255 inclusive
* @param k key (black) value, between 0 and 255 inclusive
* @return uint32_t returned integer colour value
*/
uint32_t DPP_EXPORT cmyk(int c, int m, int y, int k);
/**
* @brief Output hex values of a section of memory for debugging
*
* @param data The start of the data to display
* @param length The length of data to display
*/
std::string DPP_EXPORT debug_dump(uint8_t* data, size_t length);
/**
* @brief Returns the length of a UTF-8 string in codepoints
*
* @param str string to count length of
* @return size_t length of string (0 for invalid utf8)
*/
size_t DPP_EXPORT utf8len(const std::string &str);
/**
* @brief Return substring of a UTF-8 encoded string in codepoints
*
* @param str string to return substring from
* @param start start codepoint offset
* @param length length in codepoints
* @return std::string Substring in UTF-8 or empty string if invalid UTF-8 passed in
*/
std::string DPP_EXPORT utf8substr(const std::string& str, std::string::size_type start, std::string::size_type length);
/**
* @brief Read a whole file into a std::string.
* Be sure you have enough memory to read the file, if you are reading a large file.
* @note Be aware this function can block! If you are regularly reading large files, consider caching them.
* @param filename The path to the file to read
* @return std::string The file contents
* @throw dpp::exception on failure to read the entire file
*/
std::string DPP_EXPORT read_file(const std::string& filename);
/**
* @brief Validate a string value
* In the event the length of the string is less than _min, then an exception of type dpp:length_exception
* will be thrown. If the string is longer than _max UTF8 codepoints it will be truncated to fit.
*
* @param value The value to validate
* @param _min Minimum length
* @param _max Maximum length
* @param exception_message Exception message to throw if value length < _min
* @return std::string Validated string, truncated if necessary.
* @throw dpp::length_exception if value UTF8 length < _min
*/
std::string DPP_EXPORT validate(const std::string& value, size_t _min, size_t _max, const std::string& exception_message);
/**
* @brief Get the url query parameter for the cdn endpoint. Internally used to build url getters.
*
* @param size size to generate url parameter for. Must be any power of two between 16 and 4096 (inclusive) or it'll return an empty string.
* @return std::string url query parameter e.g. `?size=128`, or an empty string
*/
std::string DPP_EXPORT avatar_size(uint32_t size);
/**
* @brief Split (tokenize) a string into a vector, using the given separators
*
* @param in Input string
* @param sep Separator characters
* @return std::vector<std::string> Tokenized strings
*/
std::vector<std::string> DPP_EXPORT tokenize(std::string const &in, const char* sep = "\r\n");
/**
* @brief Create a bot invite
*
* @param bot_id Bot ID
* @param permissions Permission bitmask of the bot to invite
* @param scopes Scopes to use
* @return Invite URL
*/
std::string DPP_EXPORT bot_invite_url(const snowflake bot_id, const uint64_t permissions = 0, const std::vector<std::string>& scopes = {"bot", "applications.commands"});
/**
* @brief Escapes Discord's markdown sequences in a string
*
* @param text Text to escape
* @param escape_code_blocks If set to false, then code blocks are not escaped.
* This means that you can still use a code block, and the text within will be left as-is.
* If set to true, code blocks will also be escaped so that ` symbol may be used as a normal
* character.
* @return std::string The text with the markdown special characters escaped with a backslash
*/
std::string DPP_EXPORT markdown_escape(const std::string& text, bool escape_code_blocks = false);
/**
* @brief Encodes a url parameter similar to [php urlencode()](https://www.php.net/manual/en/function.urlencode.php)
*
* @param value String to encode
* @return std::string URL encoded string
*/
std::string DPP_EXPORT url_encode(const std::string &value);
/**
* @brief Create a mentionable slashcommand (used in a message).
* @param command_id The ID of the slashcommand
* @param command_name The command name
* @param subcommand Optional: The subcommand name (for mentioning a subcommand)
* @return std::string The formatted mention
*/
std::string DPP_EXPORT slashcommand_mention(snowflake command_id, const std::string &command_name, const std::string &subcommand = "");
/**
* @brief Create a mentionable slashcommand (used in a message).
* @param command_id The ID of the slashcommand
* @param command_name The command name
* @param subcommand_group The subcommand group name
* @param subcommand The subcommand name
* @return std::string The formatted mention of the slashcommand with its subcommand
*/
std::string DPP_EXPORT slashcommand_mention(snowflake command_id, const std::string &command_name, const std::string &subcommand_group, const std::string &subcommand);
/**
* @brief Create a mentionable user.
* @param id The ID of the user.
* @return std::string The formatted mention of the user.
*/
std::string DPP_EXPORT user_mention(const snowflake& id);
/**
* @brief Create a mentionable channel.
* @param id The ID of the channel.
* @return std::string The formatted mention of the channel.
*/
std::string DPP_EXPORT channel_mention(const snowflake& id);
/**
* @brief Create a mentionable emoji
* @param name The name of the emoji.
* @param id The ID of the emoji.
* @param is_animated is emoji animated.
* @return std::string The formatted mention of the emoji.
*/
std::string DPP_EXPORT emoji_mention(const std::string& name, const snowflake& id, bool is_animated = false);
/**
* @brief Create a mentionable role.
* @param id The ID of the role.
* @return std::string The formatted mention of the role.
*/
std::string DPP_EXPORT role_mention(const snowflake& id);
/**
* @brief Returns the library's version string
*
* @return std::string version
*/
std::string DPP_EXPORT version();
/**
* @brief Build a URL parameter string e.g. "?a=b&c=d&e=f" from a map of key/value pairs.
* Entries with empty key names or values are omitted.
*
* @param parameters parameters to create a url query string for
* @return std::string A correctly encoded url query string
*/
std::string DPP_EXPORT make_url_parameters(const std::map<std::string, std::string>& parameters);
/**
* @brief Build a URL parameter string e.g. "?a=b&c=d&e=f" from a map of key/value pairs.
* Entries with empty key names or zero values are omitted.
*
* @param parameters parameters to create a url query string for
* @return std::string A correctly encoded url query string
*/
std::string DPP_EXPORT make_url_parameters(const std::map<std::string, uint64_t>& parameters);
/**
* @brief Set the name of the current thread for debugging and statistical reporting
*
* @param name New name to set
*/
void DPP_EXPORT set_thread_name(const std::string& name);
};
};