GCC Code Coverage Report
Directory: ./ Exec Total Coverage
File: json_utils.h Lines: 96 96 100.0 %
Date: 2022-12-07 04:23:16 Branches: 21 22 95.5 %

Line Branch Exec Source
1
#ifndef SRC_JSON_UTILS_H_
2
#define SRC_JSON_UTILS_H_
3
4
#if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
5
6
#include <iomanip>
7
#include <ostream>
8
#include <limits>
9
#include <string>
10
11
namespace node {
12
13
std::string EscapeJsonChars(const std::string& str);
14
std::string Reindent(const std::string& str, int indentation);
15
16
// JSON compiler definitions.
17
class JSONWriter {
18
 public:
19
35
  JSONWriter(std::ostream& out, bool compact)
20
35
    : out_(out), compact_(compact) {}
21
22
 private:
23
5311
  inline void indent() { indent_ += 2; }
24
5311
  inline void deindent() { indent_ -= 2; }
25
43170
  inline void advance() {
26
43170
    if (compact_) return;
27
321627
    for (int i = 0; i < indent_; i++) out_ << ' ';
28
  }
29
33406
  inline void write_one_space() {
30
33406
    if (compact_) return;
31
32401
    out_ << ' ';
32
  }
33
43170
  inline void write_new_line() {
34
43170
    if (compact_) return;
35
41845
    out_ << '\n';
36
  }
37
38
 public:
39
3926
  inline void json_start() {
40
3926
    if (state_ == kAfterValue) out_ << ',';
41
3926
    write_new_line();
42
3926
    advance();
43
3926
    out_ << '{';
44
3926
    indent();
45
3926
    state_ = kObjectStart;
46
3926
  }
47
48
3891
  inline void json_end() {
49
3891
    write_new_line();
50
3891
    deindent();
51
3891
    advance();
52
3891
    out_ << '}';
53
3891
    state_ = kAfterValue;
54
3891
  }
55
  template <typename T>
56
1116
  inline void json_objectstart(T key) {
57
1116
    if (state_ == kAfterValue) out_ << ',';
58
1116
    write_new_line();
59
1116
    advance();
60
1116
    write_string(key);
61
1116
    out_ << ':';
62
1116
    write_one_space();
63
1116
    out_ << '{';
64
1116
    indent();
65
1116
    state_ = kObjectStart;
66
1116
  }
67
68
  template <typename T>
69
269
  inline void json_arraystart(T key) {
70
269
    if (state_ == kAfterValue) out_ << ',';
71
269
    write_new_line();
72
269
    advance();
73
269
    write_string(key);
74
269
    out_ << ':';
75
269
    write_one_space();
76
269
    out_ << '[';
77
269
    indent();
78
269
    state_ = kObjectStart;
79
269
  }
80
1151
  inline void json_objectend() {
81
1151
    write_new_line();
82
1151
    deindent();
83
1151
    advance();
84
1151
    out_ << '}';
85
1151
    if (indent_ == 0) {
86
      // Top-level object is complete, so end the line.
87
35
      out_ << '\n';
88
    }
89
1151
    state_ = kAfterValue;
90
1151
  }
91
92
269
  inline void json_arrayend() {
93
269
    write_new_line();
94
269
    deindent();
95
269
    advance();
96
269
    out_ << ']';
97
269
    state_ = kAfterValue;
98
269
  }
99
  template <typename T, typename U>
100
64042
  inline void json_keyvalue(const T& key, const U& value) {
101
64042
    if (state_ == kAfterValue) out_ << ',';
102
64042
    write_new_line();
103
64042
    advance();
104
64042
    write_string(key);
105
64042
    out_ << ':';
106
64042
    write_one_space();
107
64042
    write_value(value);
108
64042
    state_ = kAfterValue;
109
  }
110
111
  template <typename U>
112
1054
  inline void json_element(const U& value) {
113
1054
    if (state_ == kAfterValue) out_ << ',';
114
1054
    write_new_line();
115
1054
    advance();
116
1054
    write_value(value);
117
1054
    state_ = kAfterValue;
118
1054
  }
119
120
  struct Null {};  // Usable as a JSON value.
121
122
  struct ForeignJSON {
123
    std::string as_string;
124
  };
125
126
 private:
127
  template <typename T,
128
            typename test_for_number = typename std::
129
                enable_if<std::numeric_limits<T>::is_specialized, bool>::type>
130
44682
  inline void write_value(T number) {
131
    if constexpr (std::is_same<T, bool>::value)
132
1576
      out_ << (number ? "true" : "false");
133
    else
134
43106
      out_ << number;
135
44682
  }
136
137
45
  inline void write_value(Null null) { out_ << "null"; }
138
7523
  inline void write_value(const char* str) { write_string(str); }
139
2637
  inline void write_value(const std::string& str) { write_string(str); }
140
141
2
  inline void write_value(const ForeignJSON& json) {
142
2
    out_ << Reindent(json.as_string, indent_);
143
2
  }
144
145
43566
  inline void write_string(const std::string& str) {
146
43566
    out_ << '"' << EscapeJsonChars(str) << '"';
147
43566
  }
148
40916
  inline void write_string(const char* str) { write_string(std::string(str)); }
149
150
  enum JSONState { kObjectStart, kAfterValue };
151
  std::ostream& out_;
152
  bool compact_;
153
  int indent_ = 0;
154
  int state_ = kObjectStart;
155
};
156
157
}  // namespace node
158
159
#endif  // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
160
161
#endif  // SRC_JSON_UTILS_H_