Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
S
spdlog
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Libraries
spdlog
Commits
397d4866
Commit
397d4866
authored
Mar 28, 2017
by
gabime
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fixed issue #396 and added some tests to catch it
parent
27df6eb4
Changes
6
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
359 additions
and
297 deletions
+359
-297
include/spdlog/async_logger.h
include/spdlog/async_logger.h
+5
-0
include/spdlog/details/async_log_helper.h
include/spdlog/details/async_log_helper.h
+244
-246
include/spdlog/details/async_logger_impl.h
include/spdlog/details/async_logger_impl.h
+13
-0
include/spdlog/logger.h
include/spdlog/logger.h
+5
-5
tests/errors.cpp
tests/errors.cpp
+87
-42
tests/utils.cpp
tests/utils.cpp
+5
-4
No files found.
include/spdlog/async_logger.h
View file @
397d4866
...
@@ -63,6 +63,11 @@ public:
...
@@ -63,6 +63,11 @@ public:
//Wait for the queue to be empty, and flush synchronously
//Wait for the queue to be empty, and flush synchronously
//Warning: this can potentialy last forever as we wait it to complete
//Warning: this can potentialy last forever as we wait it to complete
void
flush
()
override
;
void
flush
()
override
;
// Error handler
virtual
void
set_error_handler
(
log_err_handler
)
override
;
virtual
log_err_handler
error_handler
()
override
;
protected:
protected:
void
_sink_it
(
details
::
log_msg
&
msg
)
override
;
void
_sink_it
(
details
::
log_msg
&
msg
)
override
;
void
_set_formatter
(
spdlog
::
formatter_ptr
msg_formatter
)
override
;
void
_set_formatter
(
spdlog
::
formatter_ptr
msg_formatter
)
override
;
...
...
include/spdlog/details/async_log_helper.h
View file @
397d4866
...
@@ -32,11 +32,11 @@
...
@@ -32,11 +32,11 @@
namespace
spdlog
namespace
spdlog
{
{
namespace
details
namespace
details
{
{
class
async_log_helper
class
async_log_helper
{
{
// Async msg to move to/from the queue
// Async msg to move to/from the queue
// Movable only. should never be copied
// Movable only. should never be copied
enum
class
async_msg_type
enum
class
async_msg_type
...
@@ -58,7 +58,7 @@ class async_log_helper
...
@@ -58,7 +58,7 @@ class async_log_helper
~
async_msg
()
=
default
;
~
async_msg
()
=
default
;
async_msg
(
async_msg
&&
other
)
SPDLOG_NOEXCEPT
:
async_msg
(
async_msg
&&
other
)
SPDLOG_NOEXCEPT
:
logger_name
(
std
::
move
(
other
.
logger_name
)),
logger_name
(
std
::
move
(
other
.
logger_name
)),
level
(
std
::
move
(
other
.
level
)),
level
(
std
::
move
(
other
.
level
)),
time
(
std
::
move
(
other
.
time
)),
time
(
std
::
move
(
other
.
time
)),
...
@@ -66,7 +66,7 @@ async_msg(async_msg&& other) SPDLOG_NOEXCEPT:
...
@@ -66,7 +66,7 @@ async_msg(async_msg&& other) SPDLOG_NOEXCEPT:
msg_type
(
std
::
move
(
other
.
msg_type
))
msg_type
(
std
::
move
(
other
.
msg_type
))
{}
{}
async_msg
(
async_msg_type
m_type
)
:
msg_type
(
m_type
)
async_msg
(
async_msg_type
m_type
)
:
msg_type
(
m_type
)
{}
{}
async_msg
&
operator
=
(
async_msg
&&
other
)
SPDLOG_NOEXCEPT
async_msg
&
operator
=
(
async_msg
&&
other
)
SPDLOG_NOEXCEPT
...
@@ -85,7 +85,7 @@ async_msg(async_msg&& other) SPDLOG_NOEXCEPT:
...
@@ -85,7 +85,7 @@ async_msg(async_msg&& other) SPDLOG_NOEXCEPT:
async_msg
&
operator
=
(
const
async_msg
&
other
)
=
delete
;
async_msg
&
operator
=
(
const
async_msg
&
other
)
=
delete
;
// construct from log_msg
// construct from log_msg
async_msg
(
const
details
::
log_msg
&
m
)
:
async_msg
(
const
details
::
log_msg
&
m
)
:
level
(
m
.
level
),
level
(
m
.
level
),
time
(
m
.
time
),
time
(
m
.
time
),
thread_id
(
m
.
thread_id
),
thread_id
(
m
.
thread_id
),
...
@@ -109,7 +109,7 @@ async_msg(async_msg&& other) SPDLOG_NOEXCEPT:
...
@@ -109,7 +109,7 @@ async_msg(async_msg&& other) SPDLOG_NOEXCEPT:
}
}
};
};
public:
public:
using
item_type
=
async_msg
;
using
item_type
=
async_msg
;
using
q_type
=
details
::
mpmc_bounded_queue
<
item_type
>
;
using
q_type
=
details
::
mpmc_bounded_queue
<
item_type
>
;
...
@@ -135,8 +135,9 @@ public:
...
@@ -135,8 +135,9 @@ public:
void
flush
(
bool
wait_for_q
);
void
flush
(
bool
wait_for_q
);
void
set_error_handler
(
spdlog
::
log_err_handler
err_handler
);
private:
private:
formatter_ptr
_formatter
;
formatter_ptr
_formatter
;
std
::
vector
<
std
::
shared_ptr
<
sinks
::
sink
>>
_sinks
;
std
::
vector
<
std
::
shared_ptr
<
sinks
::
sink
>>
_sinks
;
...
@@ -182,8 +183,8 @@ private:
...
@@ -182,8 +183,8 @@ private:
// wait until the queue is empty
// wait until the queue is empty
void
wait_empty_q
();
void
wait_empty_q
();
};
};
}
}
}
}
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
...
@@ -215,13 +216,13 @@ inline spdlog::details::async_log_helper::async_log_helper(
...
@@ -215,13 +216,13 @@ inline spdlog::details::async_log_helper::async_log_helper(
// and wait for it to finish gracefully
// and wait for it to finish gracefully
inline
spdlog
::
details
::
async_log_helper
::~
async_log_helper
()
inline
spdlog
::
details
::
async_log_helper
::~
async_log_helper
()
{
{
try
try
{
{
push_msg
(
async_msg
(
async_msg_type
::
terminate
));
push_msg
(
async_msg
(
async_msg_type
::
terminate
));
_worker_thread
.
join
();
_worker_thread
.
join
();
}
}
catch
(...)
// don't crash in destructor
catch
(...)
// don't crash in destructor
{}
{
}
}
}
...
@@ -233,16 +234,13 @@ inline void spdlog::details::async_log_helper::log(const details::log_msg& msg)
...
@@ -233,16 +234,13 @@ inline void spdlog::details::async_log_helper::log(const details::log_msg& msg)
inline
void
spdlog
::
details
::
async_log_helper
::
push_msg
(
details
::
async_log_helper
::
async_msg
&&
new_msg
)
inline
void
spdlog
::
details
::
async_log_helper
::
push_msg
(
details
::
async_log_helper
::
async_msg
&&
new_msg
)
{
{
if
(
!
_q
.
enqueue
(
std
::
move
(
new_msg
))
&&
_overflow_policy
!=
async_overflow_policy
::
discard_log_msg
)
if
(
!
_q
.
enqueue
(
std
::
move
(
new_msg
))
&&
_overflow_policy
!=
async_overflow_policy
::
discard_log_msg
)
{
{
auto
last_op_time
=
details
::
os
::
now
();
auto
last_op_time
=
details
::
os
::
now
();
auto
now
=
last_op_time
;
auto
now
=
last_op_time
;
do
do
{
{
now
=
details
::
os
::
now
();
now
=
details
::
os
::
now
();
sleep_or_yield
(
now
,
last_op_time
);
sleep_or_yield
(
now
,
last_op_time
);
}
}
while
(
!
_q
.
enqueue
(
std
::
move
(
new_msg
)));
while
(
!
_q
.
enqueue
(
std
::
move
(
new_msg
)));
}
}
}
}
...
@@ -250,28 +248,30 @@ inline void spdlog::details::async_log_helper::push_msg(details::async_log_helpe
...
@@ -250,28 +248,30 @@ inline void spdlog::details::async_log_helper::push_msg(details::async_log_helpe
inline
void
spdlog
::
details
::
async_log_helper
::
flush
(
bool
wait_for_q
)
inline
void
spdlog
::
details
::
async_log_helper
::
flush
(
bool
wait_for_q
)
{
{
push_msg
(
async_msg
(
async_msg_type
::
flush
));
push_msg
(
async_msg
(
async_msg_type
::
flush
));
if
(
wait_for_q
)
if
(
wait_for_q
)
wait_empty_q
();
//return only make after the above flush message was processed
wait_empty_q
();
//return only make after the above flush message was processed
}
}
inline
void
spdlog
::
details
::
async_log_helper
::
worker_loop
()
inline
void
spdlog
::
details
::
async_log_helper
::
worker_loop
()
{
{
try
{
if
(
_worker_warmup_cb
)
_worker_warmup_cb
();
if
(
_worker_warmup_cb
)
_worker_warmup_cb
();
auto
last_pop
=
details
::
os
::
now
();
auto
last_pop
=
details
::
os
::
now
();
auto
last_flush
=
last_pop
;
auto
last_flush
=
last_pop
;
while
(
process_next_msg
(
last_pop
,
last_flush
));
auto
active
=
true
;
if
(
_worker_teardown_cb
)
_worker_teardown_cb
();
while
(
active
)
{
try
{
active
=
process_next_msg
(
last_pop
,
last_flush
);
}
}
catch
(
const
std
::
exception
&
ex
)
catch
(
const
std
::
exception
&
ex
)
{
{
_err_handler
(
ex
.
what
());
_err_handler
(
ex
.
what
());
}
}
catch
(...)
catch
(...)
{
{
_err_handler
(
"Unknown exception"
);
_err_handler
(
"Unknown exception"
);
}
}
}
if
(
_worker_teardown_cb
)
_worker_teardown_cb
();
}
}
// process next message in the queue
// process next message in the queue
...
@@ -280,11 +280,9 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
...
@@ -280,11 +280,9 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
{
{
async_msg
incoming_async_msg
;
async_msg
incoming_async_msg
;
if
(
_q
.
dequeue
(
incoming_async_msg
))
if
(
_q
.
dequeue
(
incoming_async_msg
))
{
{
last_pop
=
details
::
os
::
now
();
last_pop
=
details
::
os
::
now
();
switch
(
incoming_async_msg
.
msg_type
)
switch
(
incoming_async_msg
.
msg_type
)
{
{
case
async_msg_type
:
:
flush
:
case
async_msg_type
:
:
flush
:
_flush_requested
=
true
;
_flush_requested
=
true
;
break
;
break
;
...
@@ -298,10 +296,8 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
...
@@ -298,10 +296,8 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
log_msg
incoming_log_msg
;
log_msg
incoming_log_msg
;
incoming_async_msg
.
fill_log_msg
(
incoming_log_msg
);
incoming_async_msg
.
fill_log_msg
(
incoming_log_msg
);
_formatter
->
format
(
incoming_log_msg
);
_formatter
->
format
(
incoming_log_msg
);
for
(
auto
&
s
:
_sinks
)
for
(
auto
&
s
:
_sinks
)
{
{
if
(
s
->
should_log
(
incoming_log_msg
.
level
))
{
if
(
s
->
should_log
(
incoming_log_msg
.
level
))
{
s
->
log
(
incoming_log_msg
);
s
->
log
(
incoming_log_msg
);
}
}
}
}
...
@@ -311,8 +307,7 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
...
@@ -311,8 +307,7 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
// Handle empty queue..
// Handle empty queue..
// This is the only place where the queue can terminate or flush to avoid losing messages already in the queue
// This is the only place where the queue can terminate or flush to avoid losing messages already in the queue
else
else
{
{
auto
now
=
details
::
os
::
now
();
auto
now
=
details
::
os
::
now
();
handle_flush_interval
(
now
,
last_flush
);
handle_flush_interval
(
now
,
last_flush
);
sleep_or_yield
(
now
,
last_pop
);
sleep_or_yield
(
now
,
last_pop
);
...
@@ -324,8 +319,7 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
...
@@ -324,8 +319,7 @@ inline bool spdlog::details::async_log_helper::process_next_msg(log_clock::time_
inline
void
spdlog
::
details
::
async_log_helper
::
handle_flush_interval
(
log_clock
::
time_point
&
now
,
log_clock
::
time_point
&
last_flush
)
inline
void
spdlog
::
details
::
async_log_helper
::
handle_flush_interval
(
log_clock
::
time_point
&
now
,
log_clock
::
time_point
&
last_flush
)
{
{
auto
should_flush
=
_flush_requested
||
(
_flush_interval_ms
!=
std
::
chrono
::
milliseconds
::
zero
()
&&
now
-
last_flush
>=
_flush_interval_ms
);
auto
should_flush
=
_flush_requested
||
(
_flush_interval_ms
!=
std
::
chrono
::
milliseconds
::
zero
()
&&
now
-
last_flush
>=
_flush_interval_ms
);
if
(
should_flush
)
if
(
should_flush
)
{
{
for
(
auto
&
s
:
_sinks
)
for
(
auto
&
s
:
_sinks
)
s
->
flush
();
s
->
flush
();
now
=
last_flush
=
details
::
os
::
now
();
now
=
last_flush
=
details
::
os
::
now
();
...
@@ -368,11 +362,15 @@ inline void spdlog::details::async_log_helper::sleep_or_yield(const spdlog::log_
...
@@ -368,11 +362,15 @@ inline void spdlog::details::async_log_helper::sleep_or_yield(const spdlog::log_
inline
void
spdlog
::
details
::
async_log_helper
::
wait_empty_q
()
inline
void
spdlog
::
details
::
async_log_helper
::
wait_empty_q
()
{
{
auto
last_op
=
details
::
os
::
now
();
auto
last_op
=
details
::
os
::
now
();
while
(
_q
.
approx_size
()
>
0
)
while
(
_q
.
approx_size
()
>
0
)
{
{
sleep_or_yield
(
details
::
os
::
now
(),
last_op
);
sleep_or_yield
(
details
::
os
::
now
(),
last_op
);
}
}
}
}
inline
void
spdlog
::
details
::
async_log_helper
::
set_error_handler
(
spdlog
::
log_err_handler
err_handler
)
{
_err_handler
=
err_handler
;
}
include/spdlog/details/async_logger_impl.h
View file @
397d4866
...
@@ -57,6 +57,19 @@ inline void spdlog::async_logger::flush()
...
@@ -57,6 +57,19 @@ inline void spdlog::async_logger::flush()
_async_log_helper
->
flush
(
true
);
_async_log_helper
->
flush
(
true
);
}
}
// Error handler
inline
void
spdlog
::
async_logger
::
set_error_handler
(
spdlog
::
log_err_handler
err_handler
)
{
_err_handler
=
err_handler
;
_async_log_helper
->
set_error_handler
(
err_handler
);
}
inline
spdlog
::
log_err_handler
spdlog
::
async_logger
::
error_handler
()
{
return
_err_handler
;
}
inline
void
spdlog
::
async_logger
::
_set_formatter
(
spdlog
::
formatter_ptr
msg_formatter
)
inline
void
spdlog
::
async_logger
::
_set_formatter
(
spdlog
::
formatter_ptr
msg_formatter
)
{
{
_formatter
=
msg_formatter
;
_formatter
=
msg_formatter
;
...
...
include/spdlog/logger.h
View file @
397d4866
...
@@ -59,10 +59,6 @@ public:
...
@@ -59,10 +59,6 @@ public:
void
set_pattern
(
const
std
::
string
&
);
void
set_pattern
(
const
std
::
string
&
);
void
set_formatter
(
formatter_ptr
);
void
set_formatter
(
formatter_ptr
);
// error handler
void
set_error_handler
(
log_err_handler
);
log_err_handler
error_handler
();
// automatically call flush() if message level >= log_level
// automatically call flush() if message level >= log_level
void
flush_on
(
level
::
level_enum
log_level
);
void
flush_on
(
level
::
level_enum
log_level
);
...
@@ -70,6 +66,10 @@ public:
...
@@ -70,6 +66,10 @@ public:
const
std
::
vector
<
sink_ptr
>&
sinks
()
const
;
const
std
::
vector
<
sink_ptr
>&
sinks
()
const
;
// error handler
virtual
void
set_error_handler
(
log_err_handler
);
virtual
log_err_handler
error_handler
();
protected:
protected:
virtual
void
_sink_it
(
details
::
log_msg
&
);
virtual
void
_sink_it
(
details
::
log_msg
&
);
virtual
void
_set_pattern
(
const
std
::
string
&
);
virtual
void
_set_pattern
(
const
std
::
string
&
);
...
...
tests/errors.cpp
View file @
397d4866
...
@@ -6,6 +6,19 @@
...
@@ -6,6 +6,19 @@
#include<iostream>
#include<iostream>
class
failing_sink
:
public
spdlog
::
sinks
::
sink
{
void
log
(
const
spdlog
::
details
::
log_msg
&
msg
)
override
{
throw
std
::
runtime_error
(
"some error happened during log"
);
}
void
flush
()
{}
};
TEST_CASE
(
"default_error_handler"
,
"[errors]]"
)
TEST_CASE
(
"default_error_handler"
,
"[errors]]"
)
{
{
prepare_logdir
();
prepare_logdir
();
...
@@ -22,15 +35,17 @@ TEST_CASE("default_error_handler", "[errors]]")
...
@@ -22,15 +35,17 @@ TEST_CASE("default_error_handler", "[errors]]")
}
}
struct
custom_ex
{};
struct
custom_ex
{};
TEST_CASE
(
"custom_error_handler"
,
"[errors]]"
)
TEST_CASE
(
"custom_error_handler"
,
"[errors]]"
)
{
{
prepare_logdir
();
prepare_logdir
();
std
::
string
filename
=
"logs/simple_log.txt"
;
std
::
string
filename
=
"logs/simple_log.txt"
;
auto
logger
=
spdlog
::
create
<
spdlog
::
sinks
::
simple_file_sink_mt
>
(
"logger"
,
filename
,
true
);
auto
logger
=
spdlog
::
create
<
spdlog
::
sinks
::
simple_file_sink_mt
>
(
"logger"
,
filename
,
true
);
logger
->
flush_on
(
spdlog
::
level
::
info
);
logger
->
flush_on
(
spdlog
::
level
::
info
);
logger
->
set_error_handler
([
=
](
const
std
::
string
&
msg
)
logger
->
set_error_handler
([
=
](
const
std
::
string
&
msg
)
{
{
throw
custom_ex
();
throw
custom_ex
();
});
});
logger
->
info
(
"Good message #1"
);
logger
->
info
(
"Good message #1"
);
...
@@ -39,6 +54,16 @@ TEST_CASE("custom_error_handler", "[errors]]")
...
@@ -39,6 +54,16 @@ TEST_CASE("custom_error_handler", "[errors]]")
REQUIRE
(
count_lines
(
filename
)
==
2
);
REQUIRE
(
count_lines
(
filename
)
==
2
);
}
}
TEST_CASE
(
"default_error_handler2"
,
"[errors]]"
)
{
auto
logger
=
spdlog
::
create
<
failing_sink
>
(
"failed_logger"
);
logger
->
set_error_handler
([
=
](
const
std
::
string
&
msg
)
{
throw
custom_ex
();
});
REQUIRE_THROWS_AS
(
logger
->
info
(
"Some message"
),
custom_ex
);
}
TEST_CASE
(
"async_error_handler"
,
"[errors]]"
)
TEST_CASE
(
"async_error_handler"
,
"[errors]]"
)
{
{
prepare_logdir
();
prepare_logdir
();
...
@@ -47,8 +72,7 @@ TEST_CASE("async_error_handler", "[errors]]")
...
@@ -47,8 +72,7 @@ TEST_CASE("async_error_handler", "[errors]]")
std
::
string
filename
=
"logs/simple_async_log.txt"
;
std
::
string
filename
=
"logs/simple_async_log.txt"
;
{
{
auto
logger
=
spdlog
::
create
<
spdlog
::
sinks
::
simple_file_sink_mt
>
(
"logger"
,
filename
,
true
);
auto
logger
=
spdlog
::
create
<
spdlog
::
sinks
::
simple_file_sink_mt
>
(
"logger"
,
filename
,
true
);
logger
->
set_error_handler
([
=
](
const
std
::
string
&
msg
)
logger
->
set_error_handler
([
=
](
const
std
::
string
&
msg
)
{
{
std
::
ofstream
ofs
(
"logs/custom_err.txt"
);
std
::
ofstream
ofs
(
"logs/custom_err.txt"
);
if
(
!
ofs
)
throw
std
::
runtime_error
(
"Failed open logs/custom_err.txt"
);
if
(
!
ofs
)
throw
std
::
runtime_error
(
"Failed open logs/custom_err.txt"
);
ofs
<<
err_msg
;
ofs
<<
err_msg
;
...
@@ -62,3 +86,24 @@ TEST_CASE("async_error_handler", "[errors]]")
...
@@ -62,3 +86,24 @@ TEST_CASE("async_error_handler", "[errors]]")
REQUIRE
(
count_lines
(
filename
)
==
2
);
REQUIRE
(
count_lines
(
filename
)
==
2
);
REQUIRE
(
file_contents
(
"logs/custom_err.txt"
)
==
err_msg
);
REQUIRE
(
file_contents
(
"logs/custom_err.txt"
)
==
err_msg
);
}
}
// Make sure async error handler is executed
TEST_CASE
(
"async_error_handler2"
,
"[errors]]"
)
{
prepare_logdir
();
std
::
string
err_msg
(
"This is async handler error message"
);
spdlog
::
set_async_mode
(
128
);
{
auto
logger
=
spdlog
::
create
<
failing_sink
>
(
"failed_logger"
);
logger
->
set_error_handler
([
=
](
const
std
::
string
&
msg
)
{
std
::
ofstream
ofs
(
"logs/custom_err2.txt"
);
if
(
!
ofs
)
throw
std
::
runtime_error
(
"Failed open logs/custom_err2.txt"
);
ofs
<<
err_msg
;
});
logger
->
info
(
"Hello failure"
);
spdlog
::
drop
(
"failed_logger"
);
//force logger to drain the queue and shutdown
spdlog
::
set_sync_mode
();
}
REQUIRE
(
file_contents
(
"logs/custom_err2.txt"
)
==
err_msg
);
}
tests/utils.cpp
View file @
397d4866
...
@@ -4,11 +4,12 @@ void prepare_logdir()
...
@@ -4,11 +4,12 @@ void prepare_logdir()
{
{
spdlog
::
drop_all
();
spdlog
::
drop_all
();
#ifdef _WIN32
#ifdef _WIN32
auto
rv
=
system
(
"del /F /Q logs
\\
*"
);
system
(
"if not exist logs mkdir logs"
);
system
(
"del /F /Q logs
\\
*"
);
#else
#else
auto
rv
=
system
(
"rm -f logs/*"
);
system
(
"mkdir -p logs"
);
system
(
"rm -f logs/*"
);
#endif
#endif
(
void
)
rv
;
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment