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
1.75k
    locationInfo(false),
40
1.75k
    prettyPrint(false),
41
1.75k
    ppIndentL1(LOG4CXX_STR("  ")),
42
1.75k
    ppIndentL2(LOG4CXX_STR("    ")),
43
1.75k
    expectedPatternLength(100),
44
1.75k
    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
1.75k
  m_priv(std::make_unique<JSONLayoutPrivate>())
67
1.75k
{
68
1.75k
}
Unexecuted instantiation: log4cxx::JSONLayout::JSONLayout()
log4cxx::JSONLayout::JSONLayout()
Line
Count
Source
66
1.75k
  m_priv(std::make_unique<JSONLayoutPrivate>())
67
1.75k
{
68
1.75k
}
69
70
1.75k
JSONLayout::~JSONLayout(){}
71
72
void JSONLayout::setLocationInfo(bool locationInfoFlag)
73
1.11k
{
74
1.11k
  m_priv->locationInfo = locationInfoFlag;
75
1.11k
}
76
77
bool JSONLayout::getLocationInfo() const
78
0
{
79
0
  return m_priv->locationInfo;
80
0
}
81
82
void JSONLayout::setPrettyPrint(bool prettyPrintFlag)
83
989
{
84
989
  m_priv->prettyPrint = prettyPrintFlag;
85
989
}
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.04k
{
94
1.04k
  m_priv->threadInfo = newValue;
95
1.04k
}
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.14k
{
114
3.14k
  if (StringHelper::equalsIgnoreCase(option,
115
3.14k
      LOG4CXX_STR("LOCATIONINFO"), LOG4CXX_STR("locationinfo")))
116
1.11k
  {
117
1.11k
    setLocationInfo(OptionConverter::toBoolean(value, false));
118
1.11k
  }
119
2.03k
  else if (StringHelper::equalsIgnoreCase(option,
120
2.03k
      LOG4CXX_STR("THREADINFO"), LOG4CXX_STR("threadinfo")))
121
1.04k
  {
122
1.04k
    setThreadInfo(OptionConverter::toBoolean(value, false));
123
1.04k
  }
124
989
  else if (StringHelper::equalsIgnoreCase(option,
125
989
      LOG4CXX_STR("PRETTYPRINT"), LOG4CXX_STR("prettyprint")))
126
989
  {
127
989
    setPrettyPrint(OptionConverter::toBoolean(value, false));
128
989
  }
129
3.14k
}
130
131
void JSONLayout::format(LogString& output,
132
  const spi::LoggingEventPtr& event,
133
  Pool& p) const
134
1.75k
{
135
1.75k
  auto& lsMsg = event->getRenderedMessage();
136
1.75k
  output.reserve(m_priv->expectedPatternLength + lsMsg.size());
137
1.75k
  output.append(LOG4CXX_STR("{"));
138
1.75k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
139
140
1.75k
  if (m_priv->prettyPrint)
141
989
  {
142
989
    output.append(m_priv->ppIndentL1);
143
989
  }
144
145
1.75k
  output.append(LOG4CXX_STR("\"timestamp\": \""));
146
1.75k
  m_priv->dateFormat.format(output, event->getTimeStamp(), p);
147
1.75k
  output.append(LOG4CXX_STR("\","));
148
1.75k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
149
150
1.75k
  if (m_priv->threadInfo)
151
1.04k
  {
152
1.04k
    if (m_priv->prettyPrint)
153
747
    {
154
747
      output.append(m_priv->ppIndentL1);
155
747
    }
156
1.04k
    appendQuotedEscapedString(output, LOG4CXX_STR("thread"));
157
1.04k
    output.append(LOG4CXX_STR(": "));
158
1.04k
    appendQuotedEscapedString(output, event->getThreadName());
159
1.04k
    output.append(LOG4CXX_STR(","));
160
1.04k
    output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
161
1.04k
  }
162
163
1.75k
  if (m_priv->prettyPrint)
164
989
  {
165
989
    output.append(m_priv->ppIndentL1);
166
989
  }
167
168
1.75k
  output.append(LOG4CXX_STR("\"level\": "));
169
1.75k
  LogString level;
170
1.75k
  event->getLevel()->toString(level);
171
1.75k
  appendQuotedEscapedString(output, level);
172
1.75k
  output.append(LOG4CXX_STR(","));
173
1.75k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
174
175
1.75k
  if (m_priv->prettyPrint)
176
989
  {
177
989
    output.append(m_priv->ppIndentL1);
178
989
  }
179
180
1.75k
  output.append(LOG4CXX_STR("\"logger\": "));
181
1.75k
  appendQuotedEscapedString(output, event->getLoggerName());
182
1.75k
  output.append(LOG4CXX_STR(","));
183
1.75k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
184
185
1.75k
  if (m_priv->prettyPrint)
186
989
  {
187
989
    output.append(m_priv->ppIndentL1);
188
989
  }
189
190
1.75k
  output.append(LOG4CXX_STR("\"message\": "));
191
1.75k
  appendQuotedEscapedString(output, lsMsg);
192
193
1.75k
  appendSerializedMDC(output, event);
194
1.75k
  appendSerializedNDC(output, event);
195
196
1.75k
  if (m_priv->locationInfo)
197
1.11k
  {
198
1.11k
    output.append(LOG4CXX_STR(","));
199
1.11k
    output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
200
1.11k
    appendSerializedLocationInfo(output, event, p);
201
1.11k
  }
202
203
1.75k
  output.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
204
1.75k
  output.append(LOG4CXX_STR("}"));
205
1.75k
  output.append(LOG4CXX_EOL);
206
1.75k
}
207
208
void JSONLayout::appendQuotedEscapedString(LogString& buf,
209
  const LogString& input) const
210
27.5k
{
211
27.5k
  appendItem(input, buf);
212
27.5k
}
213
214
void JSONLayout::appendItem(const LogString& input, LogString& buf)
215
27.5k
{
216
27.5k
  auto toHexDigit = [](int ch) -> int
217
1.81M
  {
218
1.81M
    return (10 <= ch ? (0x61 - 10) : 0x30) + ch;
219
1.81M
  };
220
  /* add leading quote */
221
27.5k
  buf.push_back(0x22);
222
223
27.5k
  size_t start = 0;
224
27.5k
  size_t index = 0;
225
226
27.5k
  for (int ch : input)
227
786k
  {
228
786k
    if (0x22 == ch || 0x5c == ch)
229
6.07k
      ;
230
780k
    else if (0x20 <= ch)
231
290k
    {
232
290k
      ++index;
233
290k
      continue;
234
290k
    }
235
496k
    if (start < index)
236
20.9k
    {
237
20.9k
      buf.append(input, start, index - start);
238
20.9k
    }
239
240
496k
    switch (ch)
241
496k
    {
242
2.04k
      case 0x08:
243
        /* \b backspace */
244
2.04k
        buf.push_back(0x5c);
245
2.04k
        buf.push_back('b');
246
2.04k
        break;
247
248
3.07k
      case 0x09:
249
        /* \t tab */
250
3.07k
        buf.push_back(0x5c);
251
3.07k
        buf.push_back('t');
252
3.07k
        break;
253
254
27.9k
      case 0x0a:
255
        /* \n newline */
256
27.9k
        buf.push_back(0x5c);
257
27.9k
        buf.push_back('n');
258
27.9k
        break;
259
260
1.20k
      case 0x0c:
261
        /* \f form feed */
262
1.20k
        buf.push_back(0x5c);
263
1.20k
        buf.push_back('f');
264
1.20k
        break;
265
266
1.94k
      case 0x0d:
267
        /* \r carriage return */
268
1.94k
        buf.push_back(0x5c);
269
1.94k
        buf.push_back('r');
270
1.94k
        break;
271
272
3.11k
      case 0x22:
273
        /* \" double quote */
274
3.11k
        buf.push_back(0x5c);
275
3.11k
        buf.push_back(0x22);
276
3.11k
        break;
277
278
2.96k
      case 0x5c:
279
        /* \\ backslash */
280
2.96k
        buf.push_back(0x5c);
281
2.96k
        buf.push_back(0x5c);
282
2.96k
        break;
283
284
454k
      default:
285
454k
        buf.push_back(0x5c);
286
454k
        buf.push_back(0x75); // 'u'
287
454k
        buf.push_back(toHexDigit((ch & 0xF000) >> 12));
288
454k
        buf.push_back(toHexDigit((ch & 0xF00) >> 8));
289
454k
        buf.push_back(toHexDigit((ch & 0xF0) >> 4));
290
454k
        buf.push_back(toHexDigit(ch & 0xF));
291
454k
        break;
292
496k
    }
293
496k
    start = ++index;
294
496k
  }
295
296
27.5k
  if (start < input.size())
297
16.7k
  {
298
16.7k
    buf.append(input, start, input.size() - start);
299
16.7k
  }
300
301
  /* add trailing quote */
302
27.5k
  buf.push_back(0x22);
303
27.5k
}
304
305
void JSONLayout::appendSerializedMDC(LogString& buf,
306
  const LoggingEventPtr& event) const
307
1.75k
{
308
1.75k
  LoggingEvent::KeySet keys = event->getMDCKeySet();
309
310
1.75k
  if (keys.empty())
311
0
  {
312
0
    return;
313
0
  }
314
315
1.75k
  buf.append(LOG4CXX_STR(","));
316
1.75k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
317
318
1.75k
  if (m_priv->prettyPrint)
319
989
  {
320
989
    buf.append(m_priv->ppIndentL1);
321
989
  }
322
323
1.75k
  appendQuotedEscapedString(buf, LOG4CXX_STR("context_map"));
324
1.75k
  buf.append(LOG4CXX_STR(": {"));
325
1.75k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
326
327
1.75k
  for (LoggingEvent::KeySet::iterator it = keys.begin();
328
4.21k
    it != keys.end(); ++it)
329
2.45k
  {
330
2.45k
    if (m_priv->prettyPrint)
331
1.48k
    {
332
1.48k
      buf.append(m_priv->ppIndentL2);
333
1.48k
    }
334
335
2.45k
    appendQuotedEscapedString(buf, *it);
336
2.45k
    buf.append(LOG4CXX_STR(": "));
337
2.45k
    LogString value;
338
2.45k
    event->getMDC(*it, value);
339
2.45k
    appendQuotedEscapedString(buf, value);
340
341
    /* if this isn't the last k:v pair, we need a comma */
342
2.45k
    if (it + 1 != keys.end())
343
695
    {
344
695
      buf.append(LOG4CXX_STR(","));
345
695
      buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
346
695
    }
347
1.75k
    else
348
1.75k
    {
349
1.75k
      buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
350
1.75k
    }
351
2.45k
  }
352
353
1.75k
  if (m_priv->prettyPrint)
354
989
  {
355
989
    buf.append(m_priv->ppIndentL1);
356
989
  }
357
358
1.75k
  buf.append(LOG4CXX_STR("}"));
359
1.75k
}
360
361
void JSONLayout::appendSerializedNDC(LogString& buf,
362
  const LoggingEventPtr& event) const
363
1.75k
{
364
1.75k
  LogString ndcVal;
365
366
1.75k
  if (!event->getNDC(ndcVal))
367
0
  {
368
0
    return;
369
0
  }
370
371
1.75k
  buf.append(LOG4CXX_STR(","));
372
1.75k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
373
374
1.75k
  if (m_priv->prettyPrint)
375
989
  {
376
989
    buf.append(m_priv->ppIndentL1);
377
989
  }
378
379
1.75k
  appendQuotedEscapedString(buf, LOG4CXX_STR("context_stack"));
380
1.75k
  buf.append(LOG4CXX_STR(": ["));
381
1.75k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
382
383
1.75k
  if (m_priv->prettyPrint)
384
989
  {
385
989
    buf.append(m_priv->ppIndentL2);
386
989
  }
387
388
1.75k
  appendQuotedEscapedString(buf, ndcVal);
389
1.75k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
390
391
1.75k
  if (m_priv->prettyPrint)
392
989
  {
393
989
    buf.append(m_priv->ppIndentL1);
394
989
  }
395
396
1.75k
  buf.append(LOG4CXX_STR("]"));
397
1.75k
}
398
399
void JSONLayout::appendSerializedLocationInfo(LogString& buf,
400
  const LoggingEventPtr& event, Pool& p) const
401
1.11k
{
402
1.11k
  if (m_priv->prettyPrint)
403
738
  {
404
738
    buf.append(m_priv->ppIndentL1);
405
738
  }
406
407
1.11k
  appendQuotedEscapedString(buf, LOG4CXX_STR("location_info"));
408
1.11k
  buf.append(LOG4CXX_STR(": {"));
409
1.11k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
410
1.11k
  const LocationInfo& locInfo = event->getLocationInformation();
411
412
1.11k
  if (m_priv->prettyPrint)
413
738
  {
414
738
    buf.append(m_priv->ppIndentL2);
415
738
  }
416
417
1.11k
  appendQuotedEscapedString(buf, LOG4CXX_STR("file"));
418
1.11k
  buf.append(LOG4CXX_STR(": "));
419
1.11k
  LOG4CXX_DECODE_CHAR(fileName, locInfo.getFileName());
420
1.11k
  appendQuotedEscapedString(buf, fileName);
421
1.11k
  buf.append(LOG4CXX_STR(","));
422
1.11k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
423
424
1.11k
  if (m_priv->prettyPrint)
425
738
  {
426
738
    buf.append(m_priv->ppIndentL2);
427
738
  }
428
429
1.11k
  appendQuotedEscapedString(buf, LOG4CXX_STR("line"));
430
1.11k
  buf.append(LOG4CXX_STR(": "));
431
1.11k
  LogString lineNumber;
432
1.11k
  StringHelper::toString(locInfo.getLineNumber(), p, lineNumber);
433
1.11k
  appendQuotedEscapedString(buf, lineNumber);
434
1.11k
  buf.append(LOG4CXX_STR(","));
435
1.11k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
436
437
1.11k
  if (m_priv->prettyPrint)
438
738
  {
439
738
    buf.append(m_priv->ppIndentL2);
440
738
  }
441
442
1.11k
  appendQuotedEscapedString(buf, LOG4CXX_STR("class"));
443
1.11k
  buf.append(LOG4CXX_STR(": "));
444
1.11k
  LOG4CXX_DECODE_CHAR(className, locInfo.getClassName());
445
1.11k
  appendQuotedEscapedString(buf, className);
446
1.11k
  buf.append(LOG4CXX_STR(","));
447
1.11k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
448
449
1.11k
  if (m_priv->prettyPrint)
450
738
  {
451
738
    buf.append(m_priv->ppIndentL2);
452
738
  }
453
454
1.11k
  appendQuotedEscapedString(buf, LOG4CXX_STR("method"));
455
1.11k
  buf.append(LOG4CXX_STR(": "));
456
1.11k
  LOG4CXX_DECODE_CHAR(methodName, locInfo.getMethodName());
457
1.11k
  appendQuotedEscapedString(buf, methodName);
458
1.11k
  buf.append(m_priv->prettyPrint ? LOG4CXX_EOL : LOG4CXX_STR(" "));
459
460
1.11k
  if (m_priv->prettyPrint)
461
738
  {
462
738
    buf.append(m_priv->ppIndentL1);
463
738
  }
464
465
1.11k
  buf.append(LOG4CXX_STR("}"));
466
1.11k
}
467