Coverage Report

Created: 2025-11-16 06:34

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/logging-log4cxx/src/main/cpp/jsonlayout.cpp
Line
Count
Source
1
/*
2
 * Licensed to the Apache Software Foundation (ASF) under one or more
3
 * contributor license agreements.  See the NOTICE file distributed with
4
 * this work for additional information regarding copyright ownership.
5
 * The ASF licenses this file to You under the Apache License, Version 2.0
6
 * (the "License"); you may not use this file except in compliance with
7
 * the License.  You may obtain a copy of the License at
8
 *
9
 *      http://www.apache.org/licenses/LICENSE-2.0
10
 *
11
 * Unless required by applicable law or agreed to in writing, software
12
 * distributed under the License is distributed on an "AS IS" BASIS,
13
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
 * See the License for the specific language governing permissions and
15
 * limitations under the License.
16
 */
17
18
#include <log4cxx/logstring.h>
19
#include <log4cxx/jsonlayout.h>
20
#include <log4cxx/spi/loggingevent.h>
21
#include <log4cxx/level.h>
22
#include <log4cxx/helpers/optionconverter.h>
23
#include <log4cxx/helpers/cacheddateformat.h>
24
#include <log4cxx/helpers/simpledateformat.h>
25
#include <log4cxx/helpers/stringhelper.h>
26
#include <log4cxx/helpers/transcoder.h>
27
28
#include <string.h>
29
30
using namespace LOG4CXX_NS;
31
using namespace LOG4CXX_NS::helpers;
32
using namespace LOG4CXX_NS::spi;
33
34
IMPLEMENT_LOG4CXX_OBJECT(JSONLayout)
35
36
struct JSONLayout::JSONLayoutPrivate
37
{
38
  JSONLayoutPrivate() :
39
2.38k
    locationInfo(false),
40
2.38k
    prettyPrint(false),
41
2.38k
    ppIndentL1(LOG4CXX_STR("  ")),
42
2.38k
    ppIndentL2(LOG4CXX_STR("    ")),
43
2.38k
    expectedPatternLength(100),
44
2.38k
    threadInfo(false) {}
45
46
  // Print no location info by default
47
  bool locationInfo; //= false
48
  bool prettyPrint; //= false
49
50
  pattern::CachedDateFormat dateFormat
51
    { std::make_shared<helpers::SimpleDateFormat>(LOG4CXX_STR("yyyy-MM-dd HH:mm:ss,SSS"))
52
    , pattern::CachedDateFormat::getMaximumCacheValidity(LOG4CXX_STR("yyyy-MM-dd HH:mm:ss,SSS"))
53
    };
54
55
  LogString ppIndentL1;
56
  LogString ppIndentL2;
57
58
  // Expected length of a formatted event excluding the message text
59
  size_t expectedPatternLength;
60
61
  // Thread info is not included by default
62
  bool threadInfo; //= false
63
};
64
65
JSONLayout::JSONLayout() :
66
2.38k
  m_priv(std::make_unique<JSONLayoutPrivate>())
67
2.38k
{
68
2.38k
}
Unexecuted instantiation: log4cxx::JSONLayout::JSONLayout()
log4cxx::JSONLayout::JSONLayout()
Line
Count
Source
66
2.38k
  m_priv(std::make_unique<JSONLayoutPrivate>())
67
2.38k
{
68
2.38k
}
69
70
2.38k
JSONLayout::~JSONLayout(){}
71
72
void JSONLayout::setLocationInfo(bool locationInfoFlag)
73
1.41k
{
74
1.41k
  m_priv->locationInfo = locationInfoFlag;
75
1.41k
}
76
77
bool JSONLayout::getLocationInfo() const
78
0
{
79
0
  return m_priv->locationInfo;
80
0
}
81
82
void JSONLayout::setPrettyPrint(bool prettyPrintFlag)
83
1.13k
{
84
1.13k
  m_priv->prettyPrint = prettyPrintFlag;
85
1.13k
}
86
87
bool JSONLayout::getPrettyPrint() const
88
0
{
89
0
  return m_priv->prettyPrint;
90
0
}
91
92
void JSONLayout::setThreadInfo(bool newValue)
93
1.28k
{
94
1.28k
  m_priv->threadInfo = newValue;
95
1.28k
}
96
97
bool JSONLayout::getThreadInfo() const
98
0
{
99
0
  return m_priv->threadInfo;
100
0
}
101
102
LogString JSONLayout::getContentType() const
103
0
{
104
0
  return LOG4CXX_STR("application/json");
105
0
}
106
107
void JSONLayout::activateOptions(helpers::Pool& /* p */)
108
0
{
109
0
  m_priv->expectedPatternLength = getFormattedEventCharacterCount() * 2;
110
0
}
111
112
void JSONLayout::setOption(const LogString& option, const LogString& value)
113
3.83k
{
114
3.83k
  if (StringHelper::equalsIgnoreCase(option,
115
3.83k
      LOG4CXX_STR("LOCATIONINFO"), LOG4CXX_STR("locationinfo")))
116
1.41k
  {
117
1.41k
    setLocationInfo(OptionConverter::toBoolean(value, false));
118
1.41k
  }
119
2.41k
  else if (StringHelper::equalsIgnoreCase(option,
120
2.41k
      LOG4CXX_STR("THREADINFO"), LOG4CXX_STR("threadinfo")))
121
1.28k
  {
122
1.28k
    setThreadInfo(OptionConverter::toBoolean(value, false));
123
1.28k
  }
124
1.13k
  else if (StringHelper::equalsIgnoreCase(option,
125
1.13k
      LOG4CXX_STR("PRETTYPRINT"), LOG4CXX_STR("prettyprint")))
126
1.13k
  {
127
1.13k
    setPrettyPrint(OptionConverter::toBoolean(value, false));
128
1.13k
  }
129
3.83k
}
130
131
void JSONLayout::format(LogString& output,
132
  const spi::LoggingEventPtr& event,
133
  Pool& p) const
134
2.38k
{
135
2.38k
  auto& lsMsg = event->getRenderedMessage();
136
2.38k
  output.reserve(m_priv->expectedPatternLength + lsMsg.size());
137
2.38k
  output.append(LOG4CXX_STR("{"));
138
2.38k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
139
140
2.38k
  if (m_priv->prettyPrint)
141
1.13k
  {
142
1.13k
    output.append(m_priv->ppIndentL1);
143
1.13k
  }
144
145
2.38k
  output.append(LOG4CXX_STR("\"timestamp\": \""));
146
2.38k
  m_priv->dateFormat.format(output, event->getTimeStamp(), p);
147
2.38k
  output.append(LOG4CXX_STR("\","));
148
2.38k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
149
150
2.38k
  if (m_priv->threadInfo)
151
1.28k
  {
152
1.28k
    if (m_priv->prettyPrint)
153
781
    {
154
781
      output.append(m_priv->ppIndentL1);
155
781
    }
156
1.28k
    appendQuotedEscapedString(output, LOG4CXX_STR("thread"));
157
1.28k
    output.append(LOG4CXX_STR(": "));
158
1.28k
    appendQuotedEscapedString(output, event->getThreadName());
159
1.28k
    output.append(LOG4CXX_STR(","));
160
1.28k
    output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
161
1.28k
  }
162
163
2.38k
  if (m_priv->prettyPrint)
164
1.13k
  {
165
1.13k
    output.append(m_priv->ppIndentL1);
166
1.13k
  }
167
168
2.38k
  output.append(LOG4CXX_STR("\"level\": "));
169
2.38k
  LogString level;
170
2.38k
  event->getLevel()->toString(level);
171
2.38k
  appendQuotedEscapedString(output, level);
172
2.38k
  output.append(LOG4CXX_STR(","));
173
2.38k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
174
175
2.38k
  if (m_priv->prettyPrint)
176
1.13k
  {
177
1.13k
    output.append(m_priv->ppIndentL1);
178
1.13k
  }
179
180
2.38k
  output.append(LOG4CXX_STR("\"logger\": "));
181
2.38k
  appendQuotedEscapedString(output, event->getLoggerName());
182
2.38k
  output.append(LOG4CXX_STR(","));
183
2.38k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
184
185
2.38k
  if (m_priv->prettyPrint)
186
1.13k
  {
187
1.13k
    output.append(m_priv->ppIndentL1);
188
1.13k
  }
189
190
2.38k
  output.append(LOG4CXX_STR("\"message\": "));
191
2.38k
  appendQuotedEscapedString(output, lsMsg);
192
193
2.38k
  appendSerializedMDC(output, event);
194
2.38k
  appendSerializedNDC(output, event);
195
196
2.38k
  if (m_priv->locationInfo)
197
1.41k
  {
198
1.41k
    output.append(LOG4CXX_STR(","));
199
1.41k
    output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
200
1.41k
    appendSerializedLocationInfo(output, event, p);
201
1.41k
  }
202
203
2.38k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
204
2.38k
  output.append(LOG4CXX_STR("}"));
205
2.38k
  output.append(LOG4CXX_EOL);
206
2.38k
}
207
208
void JSONLayout::appendQuotedEscapedString(LogString& buf,
209
  const LogString& input) const
210
36.2k
{
211
36.2k
  appendItem(input, buf);
212
36.2k
}
213
214
void JSONLayout::appendItem(const LogString& input, LogString& buf)
215
36.2k
{
216
36.2k
  auto toHexDigit = [](int ch) -> int
217
2.59M
  {
218
2.59M
    return (10 <= ch ? (0x61 - 10) : 0x30) + ch;
219
2.59M
  };
220
  /* add leading quote */
221
36.2k
  buf.push_back(0x22);
222
223
36.2k
  size_t start = 0;
224
36.2k
  size_t index = 0;
225
226
36.2k
  for (int ch : input)
227
1.31M
  {
228
1.31M
    if (0x22 == ch || 0x5c == ch)
229
11.1k
      ;
230
1.30M
    else if (0x20 <= ch)
231
565k
    {
232
565k
      ++index;
233
565k
      continue;
234
565k
    }
235
752k
    if (start < index)
236
28.6k
    {
237
28.6k
      buf.append(input, start, index - start);
238
28.6k
    }
239
240
752k
    switch (ch)
241
752k
    {
242
6.41k
      case 0x08:
243
        /* \b backspace */
244
6.41k
        buf.push_back(0x5c);
245
6.41k
        buf.push_back('b');
246
6.41k
        break;
247
248
8.03k
      case 0x09:
249
        /* \t tab */
250
8.03k
        buf.push_back(0x5c);
251
8.03k
        buf.push_back('t');
252
8.03k
        break;
253
254
10.5k
      case 0x0a:
255
        /* \n newline */
256
10.5k
        buf.push_back(0x5c);
257
10.5k
        buf.push_back('n');
258
10.5k
        break;
259
260
43.1k
      case 0x0c:
261
        /* \f form feed */
262
43.1k
        buf.push_back(0x5c);
263
43.1k
        buf.push_back('f');
264
43.1k
        break;
265
266
24.5k
      case 0x0d:
267
        /* \r carriage return */
268
24.5k
        buf.push_back(0x5c);
269
24.5k
        buf.push_back('r');
270
24.5k
        break;
271
272
2.33k
      case 0x22:
273
        /* \" double quote */
274
2.33k
        buf.push_back(0x5c);
275
2.33k
        buf.push_back(0x22);
276
2.33k
        break;
277
278
8.79k
      case 0x5c:
279
        /* \\ backslash */
280
8.79k
        buf.push_back(0x5c);
281
8.79k
        buf.push_back(0x5c);
282
8.79k
        break;
283
284
648k
      default:
285
648k
        buf.push_back(0x5c);
286
648k
        buf.push_back(0x75); // 'u'
287
648k
        buf.push_back(toHexDigit((ch & 0xF000) >> 12));
288
648k
        buf.push_back(toHexDigit((ch & 0xF00) >> 8));
289
648k
        buf.push_back(toHexDigit((ch & 0xF0) >> 4));
290
648k
        buf.push_back(toHexDigit(ch & 0xF));
291
648k
        break;
292
752k
    }
293
752k
    start = ++index;
294
752k
  }
295
296
36.2k
  if (start < input.size())
297
22.1k
  {
298
22.1k
    buf.append(input, start, input.size() - start);
299
22.1k
  }
300
301
  /* add trailing quote */
302
36.2k
  buf.push_back(0x22);
303
36.2k
}
304
305
void JSONLayout::appendSerializedMDC(LogString& buf,
306
  const LoggingEventPtr& event) const
307
2.38k
{
308
2.38k
  LoggingEvent::KeySet keys = event->getMDCKeySet();
309
310
2.38k
  if (keys.empty())
311
0
  {
312
0
    return;
313
0
  }
314
315
2.38k
  buf.append(LOG4CXX_STR(","));
316
2.38k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
317
318
2.38k
  if (m_priv->prettyPrint)
319
1.13k
  {
320
1.13k
    buf.append(m_priv->ppIndentL1);
321
1.13k
  }
322
323
2.38k
  appendQuotedEscapedString(buf, LOG4CXX_STR("context_map"));
324
2.38k
  buf.append(LOG4CXX_STR(": {"));
325
2.38k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
326
327
2.38k
  for (LoggingEvent::KeySet::iterator it = keys.begin();
328
5.71k
    it != keys.end(); ++it)
329
3.33k
  {
330
3.33k
    if (m_priv->prettyPrint)
331
1.63k
    {
332
1.63k
      buf.append(m_priv->ppIndentL2);
333
1.63k
    }
334
335
3.33k
    appendQuotedEscapedString(buf, *it);
336
3.33k
    buf.append(LOG4CXX_STR(": "));
337
3.33k
    LogString value;
338
3.33k
    event->getMDC(*it, value);
339
3.33k
    appendQuotedEscapedString(buf, value);
340
341
    /* if this isn't the last k:v pair, we need a comma */
342
3.33k
    if (it + 1 != keys.end())
343
945
    {
344
945
      buf.append(LOG4CXX_STR(","));
345
945
      buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
346
945
    }
347
2.38k
    else
348
2.38k
    {
349
2.38k
      buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
350
2.38k
    }
351
3.33k
  }
352
353
2.38k
  if (m_priv->prettyPrint)
354
1.13k
  {
355
1.13k
    buf.append(m_priv->ppIndentL1);
356
1.13k
  }
357
358
2.38k
  buf.append(LOG4CXX_STR("}"));
359
2.38k
}
360
361
void JSONLayout::appendSerializedNDC(LogString& buf,
362
  const LoggingEventPtr& event) const
363
2.38k
{
364
2.38k
  LogString ndcVal;
365
366
2.38k
  if (!event->getNDC(ndcVal))
367
0
  {
368
0
    return;
369
0
  }
370
371
2.38k
  buf.append(LOG4CXX_STR(","));
372
2.38k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
373
374
2.38k
  if (m_priv->prettyPrint)
375
1.13k
  {
376
1.13k
    buf.append(m_priv->ppIndentL1);
377
1.13k
  }
378
379
2.38k
  appendQuotedEscapedString(buf, LOG4CXX_STR("context_stack"));
380
2.38k
  buf.append(LOG4CXX_STR(": ["));
381
2.38k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
382
383
2.38k
  if (m_priv->prettyPrint)
384
1.13k
  {
385
1.13k
    buf.append(m_priv->ppIndentL2);
386
1.13k
  }
387
388
2.38k
  appendQuotedEscapedString(buf, ndcVal);
389
2.38k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
390
391
2.38k
  if (m_priv->prettyPrint)
392
1.13k
  {
393
1.13k
    buf.append(m_priv->ppIndentL1);
394
1.13k
  }
395
396
2.38k
  buf.append(LOG4CXX_STR("]"));
397
2.38k
}
398
399
void JSONLayout::appendSerializedLocationInfo(LogString& buf,
400
  const LoggingEventPtr& event, Pool& p) const
401
1.41k
{
402
1.41k
  if (m_priv->prettyPrint)
403
814
  {
404
814
    buf.append(m_priv->ppIndentL1);
405
814
  }
406
407
1.41k
  appendQuotedEscapedString(buf, LOG4CXX_STR("location_info"));
408
1.41k
  buf.append(LOG4CXX_STR(": {"));
409
1.41k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
410
1.41k
  const LocationInfo& locInfo = event->getLocationInformation();
411
412
1.41k
  if (m_priv->prettyPrint)
413
814
  {
414
814
    buf.append(m_priv->ppIndentL2);
415
814
  }
416
417
1.41k
  appendQuotedEscapedString(buf, LOG4CXX_STR("file"));
418
1.41k
  buf.append(LOG4CXX_STR(": "));
419
1.41k
  LOG4CXX_DECODE_CHAR(fileName, locInfo.getFileName());
420
1.41k
  appendQuotedEscapedString(buf, fileName);
421
1.41k
  buf.append(LOG4CXX_STR(","));
422
1.41k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
423
424
1.41k
  if (m_priv->prettyPrint)
425
814
  {
426
814
    buf.append(m_priv->ppIndentL2);
427
814
  }
428
429
1.41k
  appendQuotedEscapedString(buf, LOG4CXX_STR("line"));
430
1.41k
  buf.append(LOG4CXX_STR(": "));
431
1.41k
  LogString lineNumber;
432
1.41k
  StringHelper::toString(locInfo.getLineNumber(), p, lineNumber);
433
1.41k
  appendQuotedEscapedString(buf, lineNumber);
434
1.41k
  buf.append(LOG4CXX_STR(","));
435
1.41k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
436
437
1.41k
  if (m_priv->prettyPrint)
438
814
  {
439
814
    buf.append(m_priv->ppIndentL2);
440
814
  }
441
442
1.41k
  appendQuotedEscapedString(buf, LOG4CXX_STR("class"));
443
1.41k
  buf.append(LOG4CXX_STR(": "));
444
1.41k
  LOG4CXX_DECODE_CHAR(className, locInfo.getClassName());
445
1.41k
  appendQuotedEscapedString(buf, className);
446
1.41k
  buf.append(LOG4CXX_STR(","));
447
1.41k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
448
449
1.41k
  if (m_priv->prettyPrint)
450
814
  {
451
814
    buf.append(m_priv->ppIndentL2);
452
814
  }
453
454
1.41k
  appendQuotedEscapedString(buf, LOG4CXX_STR("method"));
455
1.41k
  buf.append(LOG4CXX_STR(": "));
456
1.41k
  LOG4CXX_DECODE_CHAR(methodName, locInfo.getMethodName());
457
1.41k
  appendQuotedEscapedString(buf, methodName);
458
1.41k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
459
460
1.41k
  if (m_priv->prettyPrint)
461
814
  {
462
814
    buf.append(m_priv->ppIndentL1);
463
814
  }
464
465
1.41k
  buf.append(LOG4CXX_STR("}"));
466
1.41k
}
467