Coverage Report

Created: 2026-05-30 06:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/logging-log4cxx/src/main/cpp/appenderskeleton.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/spi/loggingevent.h>
19
#include <log4cxx/appenderskeleton.h>
20
#include <log4cxx/helpers/loglog.h>
21
#include <log4cxx/helpers/onlyonceerrorhandler.h>
22
#include <log4cxx/level.h>
23
#include <log4cxx/helpers/stringhelper.h>
24
#include <log4cxx/private/appenderskeleton_priv.h>
25
#include <mutex>
26
27
using namespace LOG4CXX_NS;
28
using namespace LOG4CXX_NS::spi;
29
using namespace LOG4CXX_NS::helpers;
30
31
IMPLEMENT_LOG4CXX_OBJECT(AppenderSkeleton)
32
33
AppenderSkeleton::AppenderSkeleton( std::unique_ptr<AppenderSkeletonPrivate> priv )
34
0
  :   m_priv(std::move(priv))
35
0
{
36
37
0
}
38
39
AppenderSkeleton::AppenderSkeleton()
40
0
  :   m_priv(std::make_unique<AppenderSkeletonPrivate>())
41
0
{
42
43
0
}
44
45
AppenderSkeleton::AppenderSkeleton(const LayoutPtr& layout)
46
0
  :   m_priv(std::make_unique<AppenderSkeletonPrivate>(layout))
47
0
{
48
49
0
}
50
51
0
AppenderSkeleton::~AppenderSkeleton() {}
52
53
void AppenderSkeleton::activateOptions( LOG4CXX_ACTIVATE_OPTIONS_FORMAL_PARAMETERS )
54
0
{
55
0
}
56
57
#if LOG4CXX_ABI_VERSION <= 15
58
void AppenderSkeleton::finalize()
59
0
{
60
  // An appender might be closed then garbage collected. There is no
61
  // point in closing twice.
62
0
  if (m_priv->closed)
63
0
  {
64
0
    return;
65
0
  }
66
67
0
  close();
68
0
}
69
#endif
70
71
void AppenderSkeleton::addFilter(const spi::FilterPtr newFilter)
72
0
{
73
0
  std::lock_guard<std::recursive_mutex> lock(m_priv->mutex);
74
75
0
  if (m_priv->headFilter == nullptr)
76
0
  {
77
0
    m_priv->headFilter = m_priv->tailFilter = newFilter;
78
0
  }
79
0
  else
80
0
  {
81
0
    m_priv->tailFilter->setNext(newFilter);
82
0
    m_priv->tailFilter = newFilter;
83
0
  }
84
0
}
85
86
void AppenderSkeleton::clearFilters()
87
0
{
88
0
  std::lock_guard<std::recursive_mutex> lock(m_priv->mutex);
89
0
  m_priv->headFilter = m_priv->tailFilter = nullptr;
90
0
}
91
92
bool AppenderSkeleton::isAsSevereAsThreshold(const LevelPtr& level) const
93
0
{
94
0
  return ((level == 0) || level->isGreaterOrEqual(m_priv->threshold));
95
0
}
96
97
void AppenderSkeleton::doAppend( LOG4CXX_APPEND_FORMAL_PARAMETERS )
98
0
{
99
0
  std::lock_guard<std::recursive_mutex> lock(m_priv->mutex);
100
101
0
  doAppendImpl( LOG4CXX_APPEND_PARAMETERS );
102
0
}
103
104
void AppenderSkeleton::doAppendImpl( LOG4CXX_APPEND_FORMAL_PARAMETERS )
105
0
{
106
0
  if (m_priv->closed)
107
0
  {
108
0
    LogLog::warn(LOG4CXX_STR("Attempted to append to closed appender named [")
109
0
      + m_priv->name + LOG4CXX_STR("]."));
110
0
  }
111
0
  else if (isAsSevereAsThreshold(event->getLevel()) && isAccepted(event))
112
0
  {
113
0
    append( LOG4CXX_APPEND_PARAMETERS );
114
0
  }
115
0
}
116
117
bool AppenderSkeleton::isAccepted(const spi::LoggingEventPtr& event) const
118
0
{
119
0
  FilterPtr f = m_priv->headFilter;
120
0
  while (f != 0)
121
0
  {
122
0
    switch (f->decide(event))
123
0
    {
124
0
      case Filter::DENY:
125
0
        return false;
126
127
0
      case Filter::ACCEPT:
128
0
        f = nullptr;
129
0
        break;
130
131
0
      default:
132
0
      case Filter::NEUTRAL:
133
0
        f = f->getNext();
134
0
    }
135
0
  }
136
0
  return true;
137
0
}
138
139
bool AppenderSkeleton::AppenderSkeletonPrivate::checkNotClosed()
140
0
{
141
0
  if (this->closed)
142
0
  {
143
0
    if (!this->warnedClosed)
144
0
    {
145
0
      LogLog::warn(LOG4CXX_STR("Not allowed to write to a closed appender."));
146
0
      this->warnedClosed = true;
147
0
    }
148
0
    return false;
149
0
  }
150
0
  return true;
151
0
}
152
153
bool AppenderSkeleton::AppenderSkeletonPrivate::setClosed()
154
0
{
155
0
  std::lock_guard<std::recursive_mutex> lock(this->mutex);
156
0
  bool wasOpen = !this->closed;
157
0
  this->closed = true;
158
0
  return wasOpen;
159
0
}
160
161
bool AppenderSkeleton::AppenderSkeletonPrivate::checkLayout()
162
0
{
163
0
  if (!this->layout)
164
0
  {
165
0
    if (!this->warnedNoLayout)
166
0
    {
167
0
      this->errorHandler->error
168
0
        ( LogString(LOG4CXX_STR("No layout set for the appender named ["))
169
0
        + this->name + LOG4CXX_STR("].")
170
0
        );
171
0
      this->warnedNoLayout = true;
172
0
    }
173
0
    return false;
174
0
  }
175
0
  return true;
176
0
}
177
178
void AppenderSkeleton::setErrorHandler(const spi::ErrorHandlerPtr errorHandler1)
179
0
{
180
0
  std::lock_guard<std::recursive_mutex> lock(m_priv->mutex);
181
182
0
  if (errorHandler1 == nullptr)
183
0
  {
184
    // We do not throw exception here since the cause is probably a
185
    // bad config file.
186
0
    LogLog::warn(LOG4CXX_STR("You have tried to set a null error-handler."));
187
0
  }
188
0
  else
189
0
  {
190
0
    m_priv->errorHandler = errorHandler1;
191
0
  }
192
0
}
193
194
void AppenderSkeleton::setThreshold(const LevelPtr& threshold1)
195
0
{
196
0
  std::lock_guard<std::recursive_mutex> lock(m_priv->mutex);
197
0
  m_priv->threshold = threshold1;
198
0
}
199
200
void AppenderSkeleton::setOption(const LogString& option,
201
  const LogString& value)
202
0
{
203
0
  if (StringHelper::equalsIgnoreCase(option,
204
0
      LOG4CXX_STR("THRESHOLD"), LOG4CXX_STR("threshold")))
205
0
  {
206
0
    setThreshold(Level::toLevelLS(value));
207
0
  }
208
0
  else if (StringHelper::equalsIgnoreCase(option,
209
0
      LOG4CXX_STR("NAME"), LOG4CXX_STR("name")))
210
0
  {
211
0
    setName(value);
212
0
  }
213
#if ENABLE_FAILING_APPENDER_SIMULATION_TESTING
214
  else if (StringHelper::equalsIgnoreCase(option,
215
      LOG4CXX_STR("FAILONMESSAGE"), LOG4CXX_STR("failonmessage")))
216
  {
217
    m_priv->exceptionTriggeringMessage = value;
218
  }
219
#endif
220
0
}
221
222
const spi::ErrorHandlerPtr AppenderSkeleton::getErrorHandler() const
223
0
{
224
0
  return m_priv->errorHandler;
225
0
}
226
227
spi::FilterPtr AppenderSkeleton::getFilter() const
228
0
{
229
0
  return m_priv->headFilter;
230
0
}
231
232
const spi::FilterPtr AppenderSkeleton::getFirstFilter() const
233
0
{
234
0
  return m_priv->headFilter;
235
0
}
236
237
LayoutPtr AppenderSkeleton::getLayout() const
238
0
{
239
0
  return m_priv->layout;
240
0
}
241
242
LogString AppenderSkeleton::getName() const
243
0
{
244
0
  return m_priv->name;
245
0
}
246
247
const LevelPtr AppenderSkeleton::getThreshold() const
248
0
{
249
0
  return m_priv->threshold;
250
0
}
251
252
void AppenderSkeleton::setLayout(const LayoutPtr layout1)
253
0
{
254
0
  m_priv->layout = layout1;
255
0
}
256
257
void AppenderSkeleton::setName(const LogString& name1)
258
0
{
259
0
  m_priv->name.assign(name1);
260
0
}