relative_pointer.h 1.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. #pragma once
  2. #include <ostream>
  3. #include <string>
  4. #include <string_view>
  5. #include <jvalidate/detail/expect.h>
  6. #include <jvalidate/detail/number.h>
  7. #include <jvalidate/detail/pointer.h>
  8. #include <jvalidate/forward.h>
  9. namespace jvalidate::detail {
  10. class RelativePointer {
  11. public:
  12. RelativePointer(std::string_view path) {
  13. if (path == "0") {
  14. return;
  15. }
  16. if (auto pos = path.find('/'); pos != path.npos) {
  17. pointer_ = Pointer(path.substr(pos));
  18. path.remove_suffix(path.size() - pos);
  19. } else if (path.ends_with('#')) {
  20. requests_key_ = true;
  21. path.remove_suffix(1);
  22. }
  23. EXPECT_M(path == "0" || not path.starts_with("0"), "Cannot zero-prefix a relative pointer");
  24. parent_steps_ = from_str<size_t>(path);
  25. }
  26. template <Adapter A>
  27. std::variant<std::string, A> inspect(Pointer const & where, A const & root) const {
  28. if (requests_key_) {
  29. return where.parent(parent_steps_).back();
  30. }
  31. auto rval = where.parent(parent_steps_).walk(root);
  32. return pointer_.walk(rval);
  33. }
  34. friend std::ostream & operator<<(std::ostream & os, RelativePointer const & rel) {
  35. os << rel.parent_steps_;
  36. if (rel.requests_key_) {
  37. return os << '#';
  38. }
  39. os << rel.pointer_;
  40. return os;
  41. }
  42. private:
  43. size_t parent_steps_ = 0;
  44. bool requests_key_ = false;
  45. Pointer pointer_ = {};
  46. };
  47. }