// // Copyright (c) 2019-2023 Ruben Perez Hidalgo (rubenperez038 at gmail dot com) // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #ifndef BHO_MYSQL_METADATA_HPP #define BHO_MYSQL_METADATA_HPP #include #include #include #include #include #include namespace bho { namespace mysql { /** * \brief Metadata about a column in a SQL query. * \details This is a regular, value type. Instances of this class are not created by the user * directly, but by the library. */ class metadata { public: /** * \brief Default constructor. * \details The constructed metadata object has undefined * values for all of its members. * * \par Exception safety * No-throw guarantee. */ metadata() = default; /** * \brief Move constructor. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * `string_view`s obtained by calling accessor functions on `other` are invalidated. */ metadata(metadata&& other) = default; /** * \brief Copy constructor. * * \par Exception safety * Strong guarantee. Internal allocations may throw. */ metadata(const metadata& other) = default; /** * \brief Move assignment. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * `string_view`s obtained by calling accessor functions on both `*this` and `other` * are invalidated. */ metadata& operator=(metadata&& other) = default; /** * \brief Copy assignment. * * \par Exception safety * Basic guarantee. Internal allocations may throw. * * \par Object lifetimes * `string_view`s obtained by calling accessor functions on `*this` * are invalidated. */ metadata& operator=(const metadata& other) = default; /// Destructor. ~metadata() = default; /** * \brief Returns the name of the database (schema) the column belongs to. * \details * This is optional information - it won't be populated unless * the connection executing the query has `meta_mode() == metadata_mode::full`. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * The returned reference is valid as long as `*this` is alive and hasn't been * assigned to or moved from. */ string_view database() const noexcept { return schema_; } /** * \brief Returns the name of the virtual table the column belongs to. * \details If the table was aliased, this will be the name of the alias * (e.g. in `"SELECT * FROM employees emp"`, `table()` will be `"emp"`). *\n * This is optional information - it won't be populated unless * the connection executing the query has `meta_mode() == metadata_mode::full`. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * The returned reference is valid as long as `*this` is alive and hasn't been * assigned to or moved from. */ string_view table() const noexcept { return table_; } /** * \brief Returns the name of the physical table the column belongs to. * \details E.g. in `"SELECT * FROM employees emp"`, * `original_table()` will be `"employees"`. * \n * This is optional information - it won't be populated unless * the connection executing the query has `meta_mode() == metadata_mode::full`. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * The returned reference is valid as long as `*this` is alive and hasn't been * assigned to or moved from. */ string_view original_table() const noexcept { return org_table_; } /** * \brief Returns the actual name of the column. * \details If the column was aliased, this will be the name of the alias * (e.g. in `"SELECT id AS employee_id FROM employees"`, * `column_name()` will be `"employee_id"`). *\n * This is optional information - it won't be populated unless * the connection executing the query has `meta_mode() == metadata_mode::full`. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * The returned reference is valid as long as `*this` is alive and hasn't been * assigned to or moved from. */ string_view column_name() const noexcept { return name_; } /** * \brief Returns the original (physical) name of the column. * \details E.g. in `"SELECT id AS employee_id FROM employees"`, * `original_column_name()` will be `"id"`. * \n * This is optional information - it won't be populated unless * the connection executing the query has `meta_mode() == metadata_mode::full`. * * \par Exception safety * No-throw guarantee. * * \par Object lifetimes * The returned reference is valid as long as `*this` is alive and hasn't been * assigned to or moved from. */ string_view original_column_name() const noexcept { return org_name_; } /** * \brief Returns the ID of the collation that fields belonging to this column use. * \details This is not the collation used when defining the column * in a `CREATE TABLE` statement, but the collation that fields that belong to * this column and are sent to the client have. It usually matches the connection's collation. * * \par Exception safety * No-throw guarantee. */ std::uint16_t column_collation() const noexcept { return character_set_; } /** * \brief Returns the maximum length of the column. * \par Exception safety * No-throw guarantee. */ unsigned column_length() const noexcept { return column_length_; } /** * \brief Returns the type of the column (see \ref column_type for more info). * \par Exception safety * No-throw guarantee. */ column_type type() const noexcept { return type_; } /** * \brief Returns the number of decimals of the column. * \par Exception safety * No-throw guarantee. */ unsigned decimals() const noexcept { return decimals_; } /** * \brief Returns `true` if the column is not allowed to be NULL, `false` if it is nullable. * \par Exception safety * No-throw guarantee. */ bool is_not_null() const noexcept { return flag_set(detail::column_flags::not_null); } /** * \brief Returns `true` if the column is part of a `PRIMARY KEY`. * \par Exception safety * No-throw guarantee. */ bool is_primary_key() const noexcept { return flag_set(detail::column_flags::pri_key); } /** * \brief Returns `true` if the column is part of a `UNIQUE KEY` (but not a `PRIMARY KEY`). * \par Exception safety * No-throw guarantee. */ bool is_unique_key() const noexcept { return flag_set(detail::column_flags::unique_key); } /** * \brief Returns `true` if the column is part of a `KEY` (but not a `UNIQUE KEY` or `PRIMARY KEY`). * \par Exception safety * No-throw guarantee. */ bool is_multiple_key() const noexcept { return flag_set(detail::column_flags::multiple_key); } /** * \brief Returns `true` if the column has no sign (is `UNSIGNED`). * \par Exception safety * No-throw guarantee. */ bool is_unsigned() const noexcept { return flag_set(detail::column_flags::unsigned_); } /** * \brief Returns `true` if the column is defined as `ZEROFILL` (padded to its maximum length by * zeros). * \par Exception safety * No-throw guarantee. */ bool is_zerofill() const noexcept { return flag_set(detail::column_flags::zerofill); } /** * \brief Returns `true` if the column is defined as `AUTO_INCREMENT`. * \par Exception safety * No-throw guarantee. */ bool is_auto_increment() const noexcept { return flag_set(detail::column_flags::auto_increment); } /** * \brief Returns `true` if the column does not have a default value. * \par Exception safety * No-throw guarantee. */ bool has_no_default_value() const noexcept { return flag_set(detail::column_flags::no_default_value); } /** * \brief Returns `true` if the column is defined as `ON UPDATE CURRENT_TIMESTAMP`. * \par Exception safety * No-throw guarantee. */ bool is_set_to_now_on_update() const noexcept { return flag_set(detail::column_flags::on_update_now); } private: std::string schema_; std::string table_; // virtual table std::string org_table_; // physical table std::string name_; // virtual column name std::string org_name_; // physical column name std::uint16_t character_set_; std::uint32_t column_length_; // maximum length of the field column_type type_; // type of the column std::uint16_t flags_; // Flags as defined in Column Definition Flags std::uint8_t decimals_; // max shown decimal digits. 0x00 for int/static strings; 0x1f for // dynamic strings, double, float metadata(const detail::coldef_view& coldef, bool copy_strings) : schema_(copy_strings ? coldef.database : string_view()), table_(copy_strings ? coldef.table : string_view()), org_table_(copy_strings ? coldef.org_table : string_view()), name_(copy_strings ? coldef.name : string_view()), org_name_(copy_strings ? coldef.org_name : string_view()), character_set_(coldef.collation_id), column_length_(coldef.column_length), type_(coldef.type), flags_(coldef.flags), decimals_(coldef.decimals) { } bool flag_set(std::uint16_t flag) const noexcept { return flags_ & flag; } #ifndef BHO_MYSQL_DOXYGEN friend struct detail::access; #endif }; } // namespace mysql } // namespace bho #endif