/reflect

C++20 reflection library

Boost Licence Version build Try it online


C++20 minimal static reflection library

https://en.wikipedia.org/wiki/Reflective_programming

Features

Requirements


#include <reflect>

int main() {
  struct foo { int a; int b; };
  enum E { A, B };

  // reflect::visit
  static_assert(2 == reflect::visit([](auto&&... args) { return sizeof...(args); }, foo{}));

  // reflect::size
  static_assert(2 == reflect::size<foo>);

  // reflect::type_name
  static_assert("foo"sv == reflect::type_name<foo>());
  static_assert("foo"sv == reflect::type_name(foo{}));

  // reflect::enum_name
  static_assert("A"sv == reflect::enum_name(E::A));
  static_assert("B"sv == reflect::enum_name(E::B));

  // reflect::member_name
  static_assert("a"sv == reflect::member_name<0, foo>());
  static_assert("a"sv == reflect::member_name<0>(foo{}));
  static_assert("b"sv == reflect::member_name<1, foo>());
  static_assert("b"sv == reflect::member_name<1>(foo{}));

  constexpr auto f = foo{.a=4, .b=2};

  // reflect::get (SFINAE friendly)
  static_assert(4 == reflect::get<0>(f)); // by index
  static_assert(2 == reflect::get<1>(f));
  static_assert(4 == reflect::get<"a">(f)); // by name
  static_assert(2 == reflect::get<"b">(f));

  // reflect::has_member_name
  static_assert(reflect::has_member_name<foo, "a">);
  static_assert(reflect::has_member_name<foo, "b">);
  static_assert(not reflect::has_member_name<foo, "c">);

  struct bar { int a{}; int b{}; };

  // reflect::to
  constexpr auto t = reflect::to<std::tuple>(foo{.a=4, .b=2});
  static_assert(4 == std::get<0>(t));
  static_assert(2 == std::get<1>(t));

  struct bar { int a{}; int c{}; };

  // reflect::to (row polymorphism / member wise copy)
  constexpr auto b = reflect::to<bar>(foo{.a=4, .b=2});
  static_assert(4 == b.a and 0 == b.c);

  // reflect::debug
  reflect::debug(f); // compile-error: debug<foo>
}

API

template <class Fn, class T>
[[nodiscard]] constexpr auto visit(Fn&& fn, T&& t) noexcept;
struct foo { int a; int b; };
static_assert(2 == visit([](auto&&... args) { return sizeof...(args); }, foo{}));
template<class T> inline constexpr auto size = /*unspecified*/
struct foo { int a; int b; };
static_assert(2 == size<foo>);
template <class T>
[[nodiscard]] consteval auto type_name(const T& = {}) noexcept;
struct foo { int a; int b; };
static_assert(std::string_view{"foo"} == type_name<foo>());
static_assert(std::string_view{"foo"} == type_name(foo{}));
template <std::size_t Min = REFLECT_ENUM_MIN, std::size_t Max = REFLECT_ENUM_MAX, class E>
  requires (std::is_enum_v<E> and Max > Min)
[[nodiscard]] constexpr auto enum_name(const E e = {}) noexcept;
enum class Enum { foo = 1, bar = 2 };
static_assert(std::string_view{"foo"} == enum_name(Enum::foo));
static_assert(std::string_view{"bar"} == enum_name(Enum::bar));
template <std::size_t N, class T> requires (N < size<T>)
[[nodiscard]] consteval auto member_name(const T& = {}) noexcept;
struct foo { int a; int b; };
static_assert(std::string_view{"a"} == member_name<0, foo>());
static_assert(std::string_view{"a"} == member_name<0>(foo{}));
static_assert(std::string_view{"b"} == member_name<1, foo>());
static_assert(std::string_view{"b"} == member_name<1>(foo{}));
template<std::size_t N, class T> requires (N < size<std::remove_cvref_t<T>>)
[[nodiscard]] constexpr decltype(auto) get(T&& t) noexcept;
struct foo { int a; bool b; };
constexpr auto f = foo{.i=42, .b=true};
static_assert(42 == get<0>(f));
static_assert(true == get<1>(f));
template <class T, fixed_string Name> concept has_member_name = /*unspecified*/
struct foo { int a; int b; };
static_assert(has_member_name<foo, "a">);
static_assert(has_member_name<foo, "b">);
static_assert(not has_member_name<foo, "c">);
template<fixed_string Name, class T> requires has_member_name<T, Name>
constexpr decltype(auto) get(T&& t) noexcept;
struct foo { int a; int b; };
constexpr auto f = foo{.i=42, .b=true};
static_assert(42 == get<"a">(f));
static_assert(true == get<"b">(f));
template<template<class...> class R, class T>
[[nodiscard]] constexpr auto to(T&& t) noexcept;
struct foo { int a; int b; };

constexpr auto t = to<std::tuple>(foo{.a=4, .b=2});
static_assert(4 == std::get<0>(t));
static_assert(2 == std::get<1>(t));

auto f = foo{.a=4, .b=2};
auto t = to<std::tuple>(f);
std::get<0>(t) *= 10;
f.b=42;
assert(40 == std::get<0>(t) and 40 == f.a);
assert(42 == std::get<1>(t) and 42 == f.b);
template<class R, class T>
[[nodiscard]] constexpr auto to(T&& t);
struct foo { int a; int b; };
struct baz { int a{}; int c{}; };

const auto b = to<baz>(foo{.a=4, .b=2});
assert(4 == b.a and 0 == b.c);
template <class T, std::size_t Size> struct fixed_string;
static_assert(0u == std::size(fixed_string{""}));
static_assert(fixed_string{""} == fixed_string{""});
static_assert(std::string_view{""} == std::string_view{fixed_string{""}});
static_assert(3u == std::size(fixed_string{"foo"}));
static_assert(std::string_view{"foo"} == std::string_view{fixed_string{"foo"}});
static_assert(fixed_string{"foo"} == fixed_string{"foo"});
consteval auto debug(auto&&...) -> void;
debug(foo{}); // compile-time error: debug(foo) is not defined

Configuration

#define REFLECT 1'0'0       // Current library version (SemVer)
#define REFLECT_MAX_SIZE 64 // Max number of member fields to be reflected
#define REFLECT_ENUM_MIN 0  // Min size for enum name
#define REFLECT_ENUM_MAX 64 // Min size for enum name lookup

FAQ

  • How reflect compares to https://wg21.link/P2996?

    reflect library only provides basic reflection primitvies, mostly via hacks and workarounds to deal with lack of the reflection. https://wg21.link/P2996 is a language proprosal with a lot of more features and capabilities. It's like comparing a drop in the ocean to the entire sea!

  • How reflect works under the hood?

    There are a many different ways to implement reflection. reflect uses C++20's structure bindings, concepts and source_location to do it. See visit implementation for more details.

  • How reflect can be compiler changes agnostic?

    reflect precomputes required prefixes/postfixes to find required names from the source_location::function_name() output for each compiler upon inclusion. Any compiler change will end up with new prefixes/postfixes and won't require additional maintanace.

  • What does it mean that reflect tests itself upon include?

    reflect runs all tests (via static_asserts) upon include. If the include compiled it means all tests are passing and the library works correctly on given compiler, enviornment.

  • What is compile-time overhead of reflect library?

    reflect include takes < 20ms (that includes running all tests). The most expensive calls are visit and enum_to_name which timing will depend on the number of reflected elements and/or min/max values provided. There are no recursive template instantiations in the library.

  • How to extend number of members to be reflected (default: 64)?

    Add new reflect::detail::visit overleads and extend REFLECT_MAX_SIZE

  • Similar projects?

    boost.pfr, glaze, reflect-cpp

  • How can I contribute?

    Please follow CONTRIBUTING.md

  • How to report a bug?

    Please follow PULL_REQUEST_TEMPLATE.md


Disclaimer reflect is not an official Boost library.