From 46b6e66b5b689452c563c661345b5e28b9c91ea3 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sat, 13 Apr 2019 21:08:44 +0700 Subject: [PATCH 01/26] Add 'color' option in %level placeholder and escaping in config file --- lib/LogP6/ConfigFile.pm6 | 25 +++++++++++-------- lib/LogP6/WriterConf/Pattern.pm6 | 32 +++++++++++++++++-------- t/00-config-file.t | 2 +- t/00-grammar.t | 4 +++- t/resource/00-config-file/log-p6-1.json | 2 +- 5 files changed, 42 insertions(+), 23 deletions(-) diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index 502cf70..53e603d 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -48,10 +48,10 @@ sub parse-config(IO() $file-path) is export { writers => list(conf, &writer), filters => list(conf, &filter), cliches => list(conf, &cliche), - default-pattern => string(conf), + default-pattern => string-e(conf), default-auto-exceptions => bool(conf), default-handle => handle(conf), - default-x-pattern => string(conf), + default-x-pattern => string-e(conf), default-level => level(conf), default-first-level-check => bool(conf), default-wrapper => wrapper(conf) @@ -68,7 +68,7 @@ sub writer(\json) { when 'std' { return LogP6::WriterConf::Std.new( name => json // Str, - pattern => json // Str, + pattern => string-e(json), handle => handle(json), auto-exceptions => bool(json) ); @@ -115,16 +115,21 @@ sub string(\json) { return Str; } +sub string-e(\json) { + return json.Str.trans(['\e', '\a', '\n'] => ["\e", "\a", "\n"]) with json; + return Str; +} + sub cliche(\json) { my $name = json; die 'Missing cliche\'s name' without $name; return LogP6::Cliche.new( name => $name, matcher => matcher(json), - default-pattern => string(json), + default-pattern => string-e(json), default-auto-exceptions => bool(json), default-handle => handle(json), - default-x-pattern => string(json), + default-x-pattern => string-e(json), default-level => level(json), default-first-level-check => bool(json), grooves => list(json, &string), @@ -210,11 +215,11 @@ sub custom(\json) { sub level(\json) { return LogP6::Level without json; given json { - when 'trace' { return trace; } - when 'debug' { return debug; } - when 'info' { return info; } - when 'warn' { return warn; } - when 'error' { return error; } + when 'trace' { return LogP6::Level::trace; } + when 'debug' { return LogP6::Level::debug; } + when 'info' { return LogP6::Level::info; } + when 'warn' { return LogP6::Level::warn; } + when 'error' { return LogP6::Level::error; } default { die 'wrong level value ' ~ json; } } } diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index a149420..cc4cc27 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -125,23 +125,33 @@ class FrameName does PatternPart { } my $lnames = []; -$lnames[trace.Int] = 'TRACE'; -$lnames[debug.Int] = 'DEBUG'; -$lnames[info.Int] = 'INFO'; -$lnames[warn.Int] = 'WARN'; -$lnames[error.Int] = 'ERROR'; +$lnames[LogP6::Level::trace.Int] = 'TRACE'; +$lnames[LogP6::Level::debug.Int] = 'DEBUG'; +$lnames[LogP6::Level::info.Int] = 'INFO'; +$lnames[LogP6::Level::warn.Int] = 'WARN'; +$lnames[LogP6::Level::error.Int] = 'ERROR'; $lnames .= List; +my $color = []; +$color[LogP6::Level::trace.Int] = "\e[33m"; # yellow +$color[LogP6::Level::debug.Int] = "\e[32m"; # green; +$color[LogP6::Level::info.Int] = "\e[34m"; # blue; +$color[LogP6::Level::warn.Int] = "\e[35m"; # magenta; +$color[LogP6::Level::error.Int] = "\e[31m"; # red; +$color .= List; + class LevelName does PatternPart { has $.levels; method new($conf) { my $levels = $lnames.clone.Array; my $length = $conf // 0; + my $colored = $conf // False; for 1..5 -> $i { $levels[$i] = $conf{$i.Str} // $levels[$i]; $levels[$i] = sprintf('%-*.*s', $length, $length, $levels[$i]) if $length > 0; + $levels[$i] = $color[$i] ~ $levels[$i] ~ "\e[0m" if $colored; } self.bless(levels => $levels.List); @@ -202,6 +212,7 @@ grammar Grammar is export { rule level-param:sym { 'WARN' '=' } rule level-param:sym { 'ERROR' '=' } rule level-param:sym { 'length' '=' } + rule level-param:sym { 'color' } # %framefile - frame file path token item:sym { '%framefile' } # %frameline - frame file line @@ -267,12 +278,13 @@ class Actions is export { } } method level-params($/) { make $>>.made.hash } - method level-param:sym($/) { make trace.Int.Str => $.Str } - method level-param:sym($/) { make debug.Int.Str => $.Str } - method level-param:sym($/) { make info.Int.Str => $.Str } - method level-param:sym($/) { make warn.Int.Str => $.Str } - method level-param:sym($/) { make error.Int.Str => $.Str } + method level-param:sym($/) { make Level::trace.Int.Str => $.Str } + method level-param:sym($/) { make Level::debug.Int.Str => $.Str } + method level-param:sym($/) { make Level::info.Int.Str => $.Str } + method level-param:sym($/) { make Level::warn.Int.Str => $.Str } + method level-param:sym($/) { make Level::error.Int.Str => $.Str } method level-param:sym($/) { make 'length' => $.Str } + method level-param:sym($/) { make 'color' => True } method item:sym($/) { make FrameFile } method item:sym($/) { make FrameLine } method item:sym($/) { make FrameName } diff --git a/t/00-config-file.t b/t/00-config-file.t index 81ad4a3..6d80350 100644 --- a/t/00-config-file.t +++ b/t/00-config-file.t @@ -147,7 +147,7 @@ subtest { is $cn.default-pattern, '%msg', 'default-pattern'; is $cn.default-auto-exceptions, False, 'default-auto-exceptions'; is $cn.default-handle, $*ERR, 'default-handle'; - is $cn.default-x-pattern, '%x', 'default-x-pattern'; + is $cn.default-x-pattern, "%x\n%x\a%x\e%x", 'default-x-pattern'; is $cn.default-level, $trace, 'default-level'; is $cn.default-first-level-check, True, 'default-first-level-check'; ok $cn.default-wrapper, 'default-wrapper ok'; diff --git a/t/00-grammar.t b/t/00-grammar.t index 295e0f4..e76564a 100644 --- a/t/00-grammar.t +++ b/t/00-grammar.t @@ -7,7 +7,7 @@ use LogP6::WriterConf::Pattern; use LogP6::Context; use IOString; -plan 14; +plan 15; my LogP6::Context $context .= new; $context.level-set($info); @@ -68,6 +68,8 @@ is parse-process($level-line), '', 'level length 0 warn'; $context.level-set($info); is parse-process($level-line), '', 'level length 0 info'; +is parse-process('colored %level{color}'), "colored \e[34mINFO\e[0m", + 'level color'; # %frame* sub info($frame) { diff --git a/t/resource/00-config-file/log-p6-1.json b/t/resource/00-config-file/log-p6-1.json index 7656ac2..8f435dc 100644 --- a/t/resource/00-config-file/log-p6-1.json +++ b/t/resource/00-config-file/log-p6-1.json @@ -5,7 +5,7 @@ "type": "std", "path": "err" }, - "default-x-pattern": "%x", + "default-x-pattern": "%x\\n%x\\a%x\\e%x", "default-level": "trace", "default-first-level-check": true, "default-wrapper": { From 9af7069390121a9bc3848cb7e97f0a3ef753b615 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sun, 14 Apr 2019 10:58:07 +0700 Subject: [PATCH 02/26] Add asynchronous handle and writer --- lib/LogP6/Context.pm6 | 20 +++++++++++++++ lib/LogP6/Handle/Async.pm6 | 24 +++++++++++++++++ lib/LogP6/Handle/AsyncSingle.pm6 | 38 +++++++++++++++++++++++++++ lib/LogP6/Writer/Async.pm6 | 16 ++++++++++++ lib/LogP6/WriterConf/Async.pm6 | 35 +++++++++++++++++++++++++ t/00-context.t | 44 ++++++++++++++++++++++++++++---- t/00-handle-async-single.t | 22 ++++++++++++++++ t/00-handle-async.t | 22 ++++++++++++++++ t/00-writer-async.t | 28 ++++++++++++++++++++ t/resource/Helpers/IOString.pm6 | 9 +++++++ 10 files changed, 253 insertions(+), 5 deletions(-) create mode 100644 lib/LogP6/Handle/Async.pm6 create mode 100644 lib/LogP6/Handle/AsyncSingle.pm6 create mode 100644 lib/LogP6/Writer/Async.pm6 create mode 100644 lib/LogP6/WriterConf/Async.pm6 create mode 100644 t/00-handle-async-single.t create mode 100644 t/00-handle-async.t create mode 100644 t/00-writer-async.t diff --git a/lib/LogP6/Context.pm6 b/lib/LogP6/Context.pm6 index 6aa6a1a..abcf375 100644 --- a/lib/LogP6/Context.pm6 +++ b/lib/LogP6/Context.pm6 @@ -26,6 +26,26 @@ submethod BUILD() { $!tname = $!thread.name; } +submethod TWEAK( + :$msg, :$date, :$level, :$x, :$trait, :@ndc, :%mdc, :$callframe +) { + $!msg = $msg; + $!date = $date; + $!level = $level; + $!x = $x; + $!trait = $trait; + @!ndc := @ndc; + %!mdc = %mdc; + $!callframe = $callframe; +} + +method copy() { + return LogP6::Context.new( + :$!msg, :$!date, :$!level, :$!x, :$!trait, + :ndc(@!ndc.clone), :mdc(%!mdc.clone), :$!callframe + ); +} + #|[Gets Context object for current Thread. # Throw exception in case Thread does not have a Context for now. # It is not recommend to use the method as part of LogP6 API. diff --git a/lib/LogP6/Handle/Async.pm6 b/lib/LogP6/Handle/Async.pm6 new file mode 100644 index 0000000..558d9b0 --- /dev/null +++ b/lib/LogP6/Handle/Async.pm6 @@ -0,0 +1,24 @@ +#|[Async IO::Handle which delegates all WRITE calls to another handle in +#| another threads.] +class LogP6::Handle::Async is IO::Handle { + has IO::Handle $.delegate is required; + has Scheduler $.scheduler; + + submethod TWEAK() { + self.encoding: $!delegate.encoding; + $!scheduler = $*SCHEDULER without $!scheduler; + } + + method WRITE(IO::Handle:D: Blob:D \data --> Bool:D) { + $!scheduler.cue({ $!delegate.WRITE(data) }); + True; + } + + method close() { + $!delegate.close; + } + + method READ(|) { #`[do nothing] } + + method EOF { #`[do nothing] } +} diff --git a/lib/LogP6/Handle/AsyncSingle.pm6 b/lib/LogP6/Handle/AsyncSingle.pm6 new file mode 100644 index 0000000..7415a88 --- /dev/null +++ b/lib/LogP6/Handle/AsyncSingle.pm6 @@ -0,0 +1,38 @@ +#|[Async IO::Handle which delegate all WRITE calls to another handle in +#| one separate thread.] +class LogP6::Handle::AsyncSingle is IO::Handle { + has IO::Handle $.delegate is required; + has Scheduler $.scheduler; + has Channel $!queue; + has Promise $!executor; + + submethod TWEAK() { + self.encoding: $!delegate.encoding; + $!scheduler = $*SCHEDULER without $!scheduler; + $!queue .= new; + $!executor .= new; + $!scheduler.cue({ + react { + whenever $!queue -> \blob { + $!delegate.WRITE(blob); + } + } + $!executor.keep(True); + }); + } + + method WRITE(IO::Handle:D: Blob:D \data --> Bool:D) { + $!queue.send(data) unless $!queue.closed; + True; + } + + method close() { + $!queue.close; + await $!executor; + $!delegate.close; + } + + method READ(|) { #`[do nothing] } + + method EOF { #`[do nothing] } +} \ No newline at end of file diff --git a/lib/LogP6/Writer/Async.pm6 b/lib/LogP6/Writer/Async.pm6 new file mode 100644 index 0000000..9c1a2e8 --- /dev/null +++ b/lib/LogP6/Writer/Async.pm6 @@ -0,0 +1,16 @@ +use LogP6::Writer; + +#| Async writer which delegates writing to another writer in another threads +class LogP6::Writer::Async does LogP6::Writer { + has LogP6::Writer $.delegate is required; + has Scheduler $.scheduler is required; + has Bool $.need-callframe is required; + + method write($context) { + # initialize date and callframe before copy context + $context.date(); + $context.callframe() if $!need-callframe; + my $copy = $context.copy; + $!scheduler.cue({ $!delegate.write($copy) }); + } +} \ No newline at end of file diff --git a/lib/LogP6/WriterConf/Async.pm6 b/lib/LogP6/WriterConf/Async.pm6 new file mode 100644 index 0000000..a690358 --- /dev/null +++ b/lib/LogP6/WriterConf/Async.pm6 @@ -0,0 +1,35 @@ +use LogP6::WriterConf; +use LogP6::Writer::Async; + +#| Conf for Async writer which delegates writing to another writer in another +#| threads. +class LogP6::WriterConf::Async does LogP6::WriterConf { + has LogP6::WriterConf $.delegate is required; + has Str $.name; + has Scheduler $.scheduler; + has Bool $.need-callframe; + + submethod TWEAK() { + $!name = $!delegate.name without $!name; + } + + method clone-with-name($name --> LogP6::WriterConf:D) { + return self.close: :$name; + } + + method make-writer(*%defaults --> LogP6::Writer:D) { + LogP6::Writer::Async.new( + :delegate($!delegate.make-writer(|%defaults)), + :scheduler($!scheduler // $*SCHEDULER), + :$!need-callframe + ); + } + + method name(--> Str) { + $!name; + } + + method close(--> Nil) { + $!delegate.close; + } +} diff --git a/t/00-context.t b/t/00-context.t index 8556815..490f0c3 100644 --- a/t/00-context.t +++ b/t/00-context.t @@ -4,7 +4,7 @@ use lib 'lib'; use LogP6::Logger; use LogP6::Level; -plan 44; +plan 54; my LogP6::Context $context = get-context(); @@ -22,14 +22,14 @@ is $context.msg, 'setted', 'setted msg'; nok $context.level.defined, 'default level'; $context.level-set('boom'); is $context.level, 'boom', 'can set any level'; -$context.level-set(info); -is $context.level, info, 'set properly level'; +$context.level-set(LogP6::Level::info); +is $context.level, LogP6::Level::info, 'set properly level'; nok $context.x.defined, 'default x'; $context.x-set(X::AdHoc.new); is $context.x, X::AdHoc.new, 'setted x'; -$context.reset('resetted', trace, X::AdHoc.new(:payload

)); +$context.reset('resetted', LogP6::Level::trace, X::AdHoc.new(:payload

)); is $context.msg, 'resetted', 'resetted msg'; -is $context.level, trace, 'resetted level'; +is $context.level, LogP6::Level::trace, 'resetted level'; is $context.x.payload, 'p', 'resetted x'; # trait @@ -97,4 +97,38 @@ nok $context.msg, 'clean all with msg'; nok $context.x, 'clean all with x'; nok $context.level, 'clean all with level'; +# clone +$context.msg-set('msg'); +my $date = $context.date(); +$context.level-set(LogP6::Level::info); +$context.x-set(X::AdHoc.new); +$context.trait-set('trait'); +$context.ndc-push('ndc'); +$context.mdc-put('key', 'value'); +sub sub-info() { $context.callframe } +sub info() { sub-info } +sub foo() { info() } +foo(); +my $copy = $context.copy; +$context.msg-set('gsm'); +is $copy.msg, 'msg', 'copy msg'; +$context.date-clean; +is $copy.date, $date, 'copy date'; +$context.level-set(LogP6::Level::warn); +is $copy.level, LogP6::Level::info, 'copy level'; +$context.x-set(Any); +is $copy.x, X::AdHoc.new, 'copy x'; +$context.trait-set('tiart'); +is $copy.trait, 'trait', 'copy trait'; +is $copy.tid, $context.tid, 'copy tid'; +is $copy.tname, $context.tname, 'copy tname'; +$context.ndc-push('cdn'); +is-deeply $copy.ndc, ['ndc'], 'copy ndc'; +$context.mdc-put('value', 'key'); +is-deeply $copy.mdc, %(:key), 'copy mdc'; +my $callframe = $context.callframe; +$context.clean; +is $copy.callframe, $callframe, 'copy callframe'; + + done-testing; diff --git a/t/00-handle-async-single.t b/t/00-handle-async-single.t new file mode 100644 index 0000000..db05cc4 --- /dev/null +++ b/t/00-handle-async-single.t @@ -0,0 +1,22 @@ +use Test; + +use lib 'lib'; +use LogP6::Handle::AsyncSingle; +use lib './t/resource/Helpers'; +use IOString; + +plan 2; + +my IOString $delegate .= new; +my LogP6::Handle::AsyncSingle $async .= new: :$delegate; + +$async.say('boom'); +$async.say('moob'); +sleep 2; + +is $delegate.clean, "boom\nmoob\n", 'delegate writes'; + +$async.close; +ok $delegate.closed, 'delegate closed'; + +done-testing; diff --git a/t/00-handle-async.t b/t/00-handle-async.t new file mode 100644 index 0000000..21b172b --- /dev/null +++ b/t/00-handle-async.t @@ -0,0 +1,22 @@ +use Test; + +use lib 'lib'; +use LogP6::Handle::Async; +use lib './t/resource/Helpers'; +use IOString; + +plan 2; + +my IOString $delegate .= new; +my LogP6::Handle::Async $async .= new: :$delegate; + +$async.say('boom'); +$async.say('moob'); +sleep 2; + +is $delegate.clean, "boom\nmoob\n", 'delegate writes'; + +$async.close; +ok $delegate.closed, 'delegate closed'; + +done-testing; diff --git a/t/00-writer-async.t b/t/00-writer-async.t new file mode 100644 index 0000000..3d9374a --- /dev/null +++ b/t/00-writer-async.t @@ -0,0 +1,28 @@ +use Test; + +use lib 'lib'; +use lib './t/resource/Helpers'; +use LogP6 :configure; +use IOString; +use LogP6::WriterConf::Async; + +plan 2; + +my IOString $handle .= new; +my $delegate = writer(:pattern('%msg'), :$handle); +my $async = LogP6::WriterConf::Async.new(:name, :$delegate); +writer($async); +cliche(:name, :matcher, + grooves => ('async-writer', level($info))); + +my $log = get-logger('async'); + +$log.info('boom'); + +sleep(1); + +is $handle.clean(), "boom\n", 'delegate writes'; +$async.close; +ok $handle.closed, 'handle closed'; + +done-testing; diff --git a/t/resource/Helpers/IOString.pm6 b/t/resource/Helpers/IOString.pm6 index 06b91b3..58cd88f 100644 --- a/t/resource/Helpers/IOString.pm6 +++ b/t/resource/Helpers/IOString.pm6 @@ -2,6 +2,7 @@ unit class IOString is IO::Handle; has Str $!writed; +has Bool $!closed; submethod TWEAK { self.encoding: 'utf8'; @@ -37,4 +38,12 @@ multi method Str(IOString:D:) { multi method gist(IOString:D:) { $!writed; +} + +method close() { + $!closed = True; +} + +method closed() { + $!closed; } \ No newline at end of file From 487a90dab33437857d8ebba31bde6aebbdb3bfba Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sun, 14 Apr 2019 11:24:28 +0700 Subject: [PATCH 03/26] Update README with color and async writing information --- README.md | 32 +++++++++++++++++++++++++------- lib/LogP6/WriterConf/Async.pm6 | 2 +- 2 files changed, 26 insertions(+), 8 deletions(-) diff --git a/README.md b/README.md index 56f29e1..9e9d304 100644 --- a/README.md +++ b/README.md @@ -29,6 +29,7 @@ even in your own libraries. - [WriterConf](#writerconf) - [Standard WriterConf](#standard-writerconf) - [Pattern](#pattern) + - [Async writing](#async-writing) - [Writer factory subroutines](#writer-factory-subroutines) - [Writer configuration file](#writer-configuration-file) - [Filter configuration](#filter-configuration) @@ -375,13 +376,16 @@ subpattern. `$msg` - optional exception message, `$name` - optional exception name, `$trace` - optional exception stacktrace. For example, `'%x{($name "$msg") Trace: $trace}'` can be converted into `'(X::AdHoc "test exception") Trace: ...'`; -- `%level{WARN=W DEBUG=D ERROR=E TRACE=T INFO=I length=2}` - log importance +- `%level{WARN=W DEBUG=D ERROR=E TRACE=T INFO=I length=2 color}` - log importance level. By default logger will use level name in upper case but you can specify synonyms for all or part of them in curly brackets in format -`=`. Also you can specify a fixed length of log level -name. Default length is 0 - write level as is. For example +`=`. You can specify a fixed length of log level name. +Default length is 0 - write level as is. For example `'[%level{WARN=hmm ERROR=alarm length=5}]'` can be converted into -`'[hmm ]'`, `'[alarm]'`, `'[INFO ]'`, `'[DEBUG]'`; +`'[hmm ]'`, `'[alarm]'`, `'[INFO ]'`, `'[DEBUG]'`. Also you can specify `color` +argument - then level names will be colored in output. It is turned off as +default. If you want to use color you prefer then you need to surround level +names with `\e[m` and `\e[0m`; - `%date{$yyyy-$yy-$MM-$MMM-$dd $hh:$mm:$ss:$mss $z}` - current date and time. String in curly brackets is used as subpattern. @@ -398,9 +402,25 @@ at the same log call line; `callframe().code.name` in log call block; Note that using `%framefile`, `%frameline` or `%framename` in the pattern will -slow your program because it requires several `callframe` calls on each +slow your program because it requires several `callframe()` calls on each resultative log call; +### Async writing + +`LogP6` provides writer and handle implementation for asynchronous writing. + +You can use `LogP6::Handle::Async.new(IO::Handle :$delegate!, Scheduler :$scheduler = $*SCHEDULER)` +as handle which will schedule `WRITE` method call of `delegate` handle. + +If is it not enough to wrap a handle then you can wrap whole writer. Use +`LogP6::WriterConf::Async.new(LogP6::WriterConf :$delegate!, Scheduler :$scheduler = $*SCHEDULER), :$name, Bool :$need-callframe)` +as writer configuration of another configuration. Final writer will schedule +`write` method call of `delegate` created writer with copy of current +`logger context`. If you miss a `:name` parameter then `delegate`'s name will +be used. Pass boolean parameter `need-callframe` if you plan to use callframe +information in wrapped writer. Note that using callframe will slow your program +because it requires several `callframe()` calls on each resultative log call. + ### Writer factory subroutines `LogP6` module has the following subs for manage writers configurations: @@ -1030,10 +1050,8 @@ subroutines logic then make a special sub for retrieve logger like # ROADMAP -- Make IO::Handle for write log in databases; - Make IO::Handle rollover support - change log file after some period of time or after file size limit are reached; -- Add Writer for asynchronous writing; - Add a `Cro::Transform` for using `LogP6` in `cro` applications. # AUTHOR diff --git a/lib/LogP6/WriterConf/Async.pm6 b/lib/LogP6/WriterConf/Async.pm6 index a690358..8e00fcd 100644 --- a/lib/LogP6/WriterConf/Async.pm6 +++ b/lib/LogP6/WriterConf/Async.pm6 @@ -14,7 +14,7 @@ class LogP6::WriterConf::Async does LogP6::WriterConf { } method clone-with-name($name --> LogP6::WriterConf:D) { - return self.close: :$name; + return self.clone(:$name); } method make-writer(*%defaults --> LogP6::Writer:D) { From b7190fc33932d894a9a75ae2f4e45c79220e440b Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sun, 14 Apr 2019 12:03:18 +0700 Subject: [PATCH 04/26] Move Filter::Std and Writer::Std construction to its Confs --- lib/LogP6/Filter/Std.pm6 | 33 --------------------------------- lib/LogP6/FilterConf/Std.pm6 | 33 ++++++++++++++++++++++++++++++++- lib/LogP6/Writer/Std.pm6 | 11 ----------- lib/LogP6/WriterConf/Std.pm6 | 7 ++++++- 4 files changed, 38 insertions(+), 46 deletions(-) diff --git a/lib/LogP6/Filter/Std.pm6 b/lib/LogP6/Filter/Std.pm6 index cbc95d7..6f4ccf2 100644 --- a/lib/LogP6/Filter/Std.pm6 +++ b/lib/LogP6/Filter/Std.pm6 @@ -7,23 +7,6 @@ class LogP6::Filter::Std does LogP6::Filter { has List:D $.before-check is required; has List:D $.after-check is required; - only method new(LogP6::FilterConf:D $conf, *%defaults) { - my $first-level-check = $conf.first-level-check // - %defaults; - my $level = $conf.level // %defaults; - my &level-check := chose-level-check($level); - my $before = ($conf.before-check // ()).Array; - $before = $first-level-check - ?? [&level-check].push(|$before) - !! $before.push(&level-check); - $before = $before.list; - self.bless( - reactive-level => $first-level-check ?? $level !! LogP6::Level::trace, - before-check => $before, - after-check => $conf.after-check // (), - ); - } - method reactive-level() { $!reactive-level; } @@ -40,20 +23,4 @@ class LogP6::Filter::Std does LogP6::Filter { return unless $check($context); } } - - sub chose-level-check($need-level) { - given $need-level { - when LogP6::Level::trace { return &trace-level-check } - when LogP6::Level::debug { return &debug-level-check } - when LogP6::Level::info { return &info-level-check } - when LogP6::Level::warn { return &warn-level-check } - when LogP6::Level::error { return &error-level-check } - } - } - - sub trace-level-check($context) { LogP6::Level::trace <= $context.level } - sub debug-level-check($context) { LogP6::Level::debug <= $context.level } - sub info-level-check($context) { LogP6::Level::info <= $context.level } - sub warn-level-check($context) { LogP6::Level::warn <= $context.level } - sub error-level-check($context) { LogP6::Level::error <= $context.level } } diff --git a/lib/LogP6/FilterConf/Std.pm6 b/lib/LogP6/FilterConf/Std.pm6 index 8cf3f2f..a25e4ea 100644 --- a/lib/LogP6/FilterConf/Std.pm6 +++ b/lib/LogP6/FilterConf/Std.pm6 @@ -17,6 +17,37 @@ class LogP6::FilterConf::Std does LogP6::FilterConf { } method make-filter(*%defaults --> LogP6::Filter:D) { - LogP6::Filter::Std.new(self, |%defaults); + my $first-level-check = $!first-level-check // + %defaults; + my $level = $!level // %defaults; + my $reactive-level = $first-level-check ?? $level !! LogP6::Level::trace; + my &level-check := chose-level-check($level); + my $before = ($!before-check // ()).Array; + $before = $first-level-check + ?? [&level-check].push(|$before) + !! $before.push(&level-check); + $before = $before.list; + + LogP6::Filter::Std.new( + :$reactive-level, + before-check => $before, + after-check => $!after-check // (), + ); + } + + sub chose-level-check($need-level) { + given $need-level { + when LogP6::Level::trace { return &trace-level-check } + when LogP6::Level::debug { return &debug-level-check } + when LogP6::Level::info { return &info-level-check } + when LogP6::Level::warn { return &warn-level-check } + when LogP6::Level::error { return &error-level-check } + } } + + sub trace-level-check($context) { LogP6::Level::trace <= $context.level } + sub debug-level-check($context) { LogP6::Level::debug <= $context.level } + sub info-level-check($context) { LogP6::Level::info <= $context.level } + sub warn-level-check($context) { LogP6::Level::warn <= $context.level } + sub error-level-check($context) { LogP6::Level::error <= $context.level } } diff --git a/lib/LogP6/Writer/Std.pm6 b/lib/LogP6/Writer/Std.pm6 index d1f3cc0..8bbcfdd 100644 --- a/lib/LogP6/Writer/Std.pm6 +++ b/lib/LogP6/Writer/Std.pm6 @@ -8,17 +8,6 @@ class LogP6::Writer::Std does LogP6::Writer { has @!pieces; - only method new(LogP6::WriterConf:D $conf, *%defaults) { - my $auto-exeptions = $conf.auto-exceptions // - %defaults; - my $pattern = $conf.pattern // %defaults; - $pattern ~= %defaults if $auto-exeptions; - self.bless( - pattern => $pattern, - handle => $conf.handle // %defaults - ); - } - submethod TWEAK() { @!pieces := Grammar.parse($!pattern, actions => Actions).made; } diff --git a/lib/LogP6/WriterConf/Std.pm6 b/lib/LogP6/WriterConf/Std.pm6 index ae9b6d8..66fbd0d 100644 --- a/lib/LogP6/WriterConf/Std.pm6 +++ b/lib/LogP6/WriterConf/Std.pm6 @@ -24,7 +24,12 @@ class LogP6::WriterConf::Std does LogP6::WriterConf { } method make-writer(*%defaults --> LogP6::Writer:D) { - LogP6::Writer::Std.new(self, |%defaults); + my $auto-ex = $!auto-exceptions // %defaults; + my $pattern = $!pattern // %defaults; + $pattern ~= %defaults if $auto-ex; + my $handle = $!handle // %defaults; + + LogP6::Writer::Std.new(:$pattern, :$handle); } method close(--> Nil) { From 85f64fba775e765f5e73ebba17074c57d0f56ed1 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sun, 14 Apr 2019 14:22:53 +0700 Subject: [PATCH 05/26] Improve exception behaviour Do not die in case of any sort of corrupt config file. --- lib/LogP6.pm6 | 42 ++++++++++++++----- lib/LogP6/Cliche.pm6 | 4 +- lib/LogP6/ConfigFile.pm6 | 4 +- lib/LogP6/Exceptions.pm6 | 8 ++++ lib/LogP6/FilterConf/Std.pm6 | 2 +- lib/LogP6/LoggerPure.pm6 | 17 ++++++++ lib/LogP6/Wrapper/SyncTime.pm6 | 2 + t/00-bad-config-file.t | 23 ++++++++++ t/00-config-file.t | 8 +++- t/00-factories.t | 8 +++- t/resource/00-config-file/log-p6-corrupt.json | 3 ++ .../00-config-file/log-p6-wrong-config.json | 12 ++++++ .../00-config-file/log-p6-wrong-syntax.json | 15 +++++++ 13 files changed, 127 insertions(+), 21 deletions(-) create mode 100644 t/00-bad-config-file.t create mode 100644 t/resource/00-config-file/log-p6-corrupt.json create mode 100644 t/resource/00-config-file/log-p6-wrong-config.json create mode 100644 t/resource/00-config-file/log-p6-wrong-syntax.json diff --git a/lib/LogP6.pm6 b/lib/LogP6.pm6 index 0a2a45d..3124720 100644 --- a/lib/LogP6.pm6 +++ b/lib/LogP6.pm6 @@ -22,6 +22,8 @@ use LogP6::ThreadLocal; use LogP6::ConfigFile; use LogP6::LogGetter; +use LogP6::Exceptions; + our $trace is export(:configure) = Level::trace; our $debug is export(:configure) = Level::debug; our $info is export(:configure) = Level::info; @@ -66,24 +68,39 @@ sub try-initialize() { sub init-from-file($config-path) is export(:configure) { $lock.protect({ + my $prev-initialized = ⚛$initialized; $initialized ⚛= 1; init-getter( get-wrap => sub ($t) { get-logger($t) }, get-pure => sub ($t) { get-logger-pure($t) }); - wipe-log-config(); - - return without $config-path; + without $config-path { + wipe-log-config(); + return; + } unless $config-path.trim.Bool { - die "log-p6 config-path is blank"; + logp6-error('log-p6 config-path is blank'); + wipe-log-config(); + return; } if !$config-path.IO.e { - die "log-p6 config '$config-path' is not exist"; + logp6-error("log-p6 config '$config-path' is not exist"); + wipe-log-config(); + return; } - my $config = parse-config($config-path); + my $config; + try { + $config = parse-config($config-path); + CATCH { default { + logp6-error($_); + wipe-log-config() if $prev-initialized == 0; + return; + }} + } + wipe-log-config(); set-default-pattern($_) with $config.default-pattern; set-default-auto-exceptions($_) with $config.default-auto-exceptions; set-default-handle($_) with $config.default-handle; @@ -94,6 +111,7 @@ sub init-from-file($config-path) is export(:configure) { writer($_) for $config.writers; filter($_) for $config.filters; cliche($_) for $config.cliches; + CATCH { default { logp6-error($_); .resume } } }); } @@ -116,7 +134,7 @@ sub clean-all-settings() { $loggers = %(); $default-pattern = '[%date{$hh:$mm:$ss}][%level] %msg'; - die "wrong default lib pattern <$($default-pattern)>" + die "wrong default pattern <$($default-pattern)>" unless Grammar.parse($default-pattern); $default-auto-exceptions = True; $default-handle = $*OUT; @@ -641,7 +659,7 @@ sub check-pattern(Str $pattern) { sub find-cliche-with(Str:D $name!, Str:D $type where * ~~ any('writer', 'filter') --> List:D ) { - @cliches.grep(*.has($name, $type)).list; + @cliches.grep(*.has($name, $type)).List; } multi sub update-loggers(Positional:D $cliches) { @@ -670,10 +688,10 @@ sub change-cliche($old-cliche, $new-cliche) { } sub create-logger($trait, $cliche) { - my $grooves = (0...^$cliche.writers.elems).list.map(-> $i { ( + my $grooves = (0...^$cliche.writers.elems).List.map(-> $i { ( get-writer($cliche.writers[$i]).make-writer(|writer-defaults($cliche)), get-filter($cliche.filters[$i]).make-filter(|filter-defaults($cliche)) - ) }).list; + ) }).List; return $grooves.elems > 0 ?? LogP6::LoggerPure.new(:$trait, :$grooves) !! LogP6::LoggerMute.new(:$trait); @@ -717,8 +735,9 @@ sub get-logger(Str:D $trait --> Logger:D) is export(:MANDATORY) { try-initialize(); return $_ with $loggers{$trait}; create-and-store-logger($trait); - $loggers{$trait} + return $loggers{$trait} }); + CATCH { default { logp6-error($_); return LogP6::LoggerMute.new(:$trait) } } } sub get-logger-pure(Str:D $trait --> Logger:D) is export(:configure) { @@ -730,6 +749,7 @@ sub get-logger-pure(Str:D $trait --> Logger:D) is export(:configure) { return $_ with atomic-fetch($loggers-pure){$trait}; create-and-store-logger($trait); }); + CATCH { default { logp6-error($_); return LogP6::LoggerMute.new(:$trait) } } } sub remove-logger(Str:D $trait --> Logger) is export(:configure) { diff --git a/lib/LogP6/Cliche.pm6 b/lib/LogP6/Cliche.pm6 index 17c9129..56328b0 100644 --- a/lib/LogP6/Cliche.pm6 +++ b/lib/LogP6/Cliche.pm6 @@ -30,9 +30,9 @@ class LogP6::Cliche { ) { my $new-writers = $!writers; my $new-filters = $!filters; - $new-writers = $new-writers.map(-> $w { $w eq $old ?? $new !! $w }).list + $new-writers = $new-writers.map(-> $w { $w eq $old ?? $new !! $w }).List if $type eq 'writer'; - $new-filters = $new-filters.map(-> $f { $f eq $old ?? $new !! $f }).list + $new-filters = $new-filters.map(-> $f { $f eq $old ?? $new !! $f }).List if $type eq 'filter'; self.clone(writers => $new-writers, filters => $new-filters); } diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index 53e603d..63be135 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -60,7 +60,7 @@ sub parse-config(IO() $file-path) is export { sub list(\json, &each) { return () without json; - return json.map({each($_)}).list; + return json.map({each($_)}).eager.List; } sub writer(\json) { @@ -80,7 +80,6 @@ sub writer(\json) { die "Wrong writer type $_"; } } - json.Str; } sub filter(\json) { @@ -101,7 +100,6 @@ sub filter(\json) { die "Wrong filter type $_"; } } - json.Str; } sub bool(\json) { diff --git a/lib/LogP6/Exceptions.pm6 b/lib/LogP6/Exceptions.pm6 index 18dd9d1..5dadd2e 100644 --- a/lib/LogP6/Exceptions.pm6 +++ b/lib/LogP6/Exceptions.pm6 @@ -4,3 +4,11 @@ class X::LogP6::PatternIsNotValid is Exception { "Wrong writer pattern format: <$!pattern>"; } } + +multi sub logp6-error(Exception:D $x) is export { + $*ERR.print('LogP6 error: ', $x.^name, ': ', $x.message, "\n", $x.backtrace); +} + +multi sub logp6-error(Str:D $x) is export { + $*ERR.say('LogP6 error: ', $x); +} \ No newline at end of file diff --git a/lib/LogP6/FilterConf/Std.pm6 b/lib/LogP6/FilterConf/Std.pm6 index a25e4ea..b8d9abe 100644 --- a/lib/LogP6/FilterConf/Std.pm6 +++ b/lib/LogP6/FilterConf/Std.pm6 @@ -26,7 +26,7 @@ class LogP6::FilterConf::Std does LogP6::FilterConf { $before = $first-level-check ?? [&level-check].push(|$before) !! $before.push(&level-check); - $before = $before.list; + $before = $before.List; LogP6::Filter::Std.new( :$reactive-level, diff --git a/lib/LogP6/LoggerPure.pm6 b/lib/LogP6/LoggerPure.pm6 index 944d76d..0faeaf7 100644 --- a/lib/LogP6/LoggerPure.pm6 +++ b/lib/LogP6/LoggerPure.pm6 @@ -1,6 +1,7 @@ use LogP6::Level; use LogP6::Logger; use LogP6::FilterConf::Std; +use LogP6::Exceptions; class LogP6::LoggerPure does LogP6::Logger { has Str:D $.trait is required; @@ -14,53 +15,68 @@ class LogP6::LoggerPure does LogP6::Logger { @$!grooves.map(-> $g {$g[1].reactive-level // LogP6::Level::trace}).min; } + method trait() { + $!trait; + } + method ndc-push($obj) { get-context.ndc-push: $obj; + CATCH { default { logp6-error($_) } } } method ndc-pop() { get-context.ndc-pop; + CATCH { default { logp6-error($_) } } } method ndc-clean() { get-context.ndc-clean; + CATCH { default { logp6-error($_) } } } method mdc-put($key, $obj) { get-context.mdc-put: $key, $obj; + CATCH { default { logp6-error($_) } } } method mdc-remove($key) { get-context.mdc-remove: $key; + CATCH { default { logp6-error($_) } } } method mdc-clean() { get-context.mdc-clean; + CATCH { default { logp6-error($_) } } } method trace(*@args, :$x) { return if $!reactive-level > LogP6::Level::trace; self!log(LogP6::Level::trace, @args, :$x); + CATCH { default { logp6-error($_) } } } method debug(*@args, :$x) { return if $!reactive-level > LogP6::Level::debug; self!log(LogP6::Level::debug, @args, :$x); + CATCH { default { logp6-error($_) } } } method info(*@args, :$x) { return if $!reactive-level > LogP6::Level::info; self!log(LogP6::Level::info, @args, :$x); + CATCH { default { logp6-error($_) } } } method warn(*@args, :$x) { return if $!reactive-level > LogP6::Level::warn; self!log(LogP6::Level::warn, @args, :$x); + CATCH { default { logp6-error($_) } } } method error(*@args, :$x) { return if $!reactive-level > LogP6::Level::error; self!log(LogP6::Level::error, @args, :$x); + CATCH { default { logp6-error($_) } } } submethod !log($level, @args, :$x) { @@ -87,6 +103,7 @@ class LogP6::LoggerPure does LogP6::Logger { class LogP6::LoggerMute does LogP6::Logger { has Str:D $.trait is required; + method trait() { $!trait } method ndc-push($obj) {} method ndc-pop() {} method ndc-clean() {} diff --git a/lib/LogP6/Wrapper/SyncTime.pm6 b/lib/LogP6/Wrapper/SyncTime.pm6 index 89f048e..c3ad0e3 100644 --- a/lib/LogP6/Wrapper/SyncTime.pm6 +++ b/lib/LogP6/Wrapper/SyncTime.pm6 @@ -1,6 +1,7 @@ use LogP6::Wrapper; use LogP6::Wrapper::SyncAbstract; use LogP6::LogGetter; +use LogP6::Exceptions; #| Wrapper for synchronize a logger each X seconds itself. class LogP6::Wrapper::SyncTime is LogP6::Wrapper::SyncAbstract { @@ -20,6 +21,7 @@ class LogP6::Wrapper::SyncTime is LogP6::Wrapper::SyncAbstract { self.update-aggr; self.put-sync-obj($now); } + CATCH { default { logp6-error($_) } } } } diff --git a/t/00-bad-config-file.t b/t/00-bad-config-file.t new file mode 100644 index 0000000..764b882 --- /dev/null +++ b/t/00-bad-config-file.t @@ -0,0 +1,23 @@ +use Test; + +use lib 'lib'; +use LogP6 :configure; +use LogP6::ConfigFile; +use lib './t/resource/Helpers'; +use IOString; + +plan 3; + +$*ERR = IOString.new; + +lives-ok { + init-from-file('./t/resource/00-config-file/log-p6-wrong-syntax.json') }, + 'cannot init from wrong syntax file'; +lives-ok { + init-from-file('./t/resource/00-config-file/log-p6-corrupt.json') }, + 'cannot init from corrupt file'; +lives-ok { + init-from-file('./t/resource/00-config-file/log-p6-wrong-config.json') }, + 'cannot init from corrupt file'; + +done-testing; diff --git a/t/00-config-file.t b/t/00-config-file.t index 6d80350..b99e04b 100644 --- a/t/00-config-file.t +++ b/t/00-config-file.t @@ -3,13 +3,17 @@ use Test; use lib 'lib'; use LogP6 :configure; use LogP6::ConfigFile; +use lib './t/resource/Helpers'; +use IOString; plan 8; +$*ERR = IOString.new; + subtest { plan 1; - dies-ok { init-from-file('./t/resource/00-config-file/log-p6-empty.json') }, + lives-ok { init-from-file('./t/resource/00-config-file/log-p6-empty.json') }, 'cannot init from empty file'; }, 'empty file'; @@ -17,7 +21,7 @@ subtest { subtest { plan 2; - dies-ok { init-from-file('./t/resource/00-config-file/not-exist.json') }, + lives-ok { init-from-file('./t/resource/00-config-file/not-exist.json') }, 'cannot init from miss file'; lives-ok { init-from-file(Any) }, 'can init empty argument'; diff --git a/t/00-factories.t b/t/00-factories.t index f0a9315..1827b5a 100644 --- a/t/00-factories.t +++ b/t/00-factories.t @@ -2,9 +2,13 @@ use Test; use lib 'lib'; use LogP6 :configure; +use lib './t/resource/Helpers'; +use IOString; plan 5; +$*ERR = IOString.new; + sub some-sub1($context) { True; } @@ -303,6 +307,7 @@ subtest { use lib './t/resource/Helpers'; use LogP6::LoggerPure; + use LogP6::LoggerPure; use LogP6::Wrapper::SyncTime; use IOString; @@ -368,8 +373,7 @@ subtest { remove-logger('not-log'); remove-logger('any'); remove-logger('any2'); - cliche(:name(''), :matcher('hahaha'), grooves => ('', ''), :replace); - dies-ok { get-logger('not-log') }, 'default cliche is changed'; + does-ok get-logger('not-log'), LogP6::LoggerMute, 'default cliche is changed'; }, 'logger'; diff --git a/t/resource/00-config-file/log-p6-corrupt.json b/t/resource/00-config-file/log-p6-corrupt.json new file mode 100644 index 0000000..fd89171 --- /dev/null +++ b/t/resource/00-config-file/log-p6-corrupt.json @@ -0,0 +1,3 @@ +{ + "write +} diff --git a/t/resource/00-config-file/log-p6-wrong-config.json b/t/resource/00-config-file/log-p6-wrong-config.json new file mode 100644 index 0000000..d074093 --- /dev/null +++ b/t/resource/00-config-file/log-p6-wrong-config.json @@ -0,0 +1,12 @@ +{ + "writers": [ + { + "type": "std", + "name": "w1" + }, + { + "type": "std", + "name": "w1" + } + ] +} diff --git a/t/resource/00-config-file/log-p6-wrong-syntax.json b/t/resource/00-config-file/log-p6-wrong-syntax.json new file mode 100644 index 0000000..38d6aac --- /dev/null +++ b/t/resource/00-config-file/log-p6-wrong-syntax.json @@ -0,0 +1,15 @@ +{ + "writers": [ + { + "type": "STD", + "name": "w1", + "pattern": "%msg", + "handle": { + "type": "file", + "path": "./t/resource/00-config-file/handle1.after", + "append": false + }, + "auto-exceptions": true + } + ] +} From 59008c882d266c39c1c2d0efc3a341e668fe9158 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Mon, 15 Apr 2019 21:28:05 +0700 Subject: [PATCH 06/26] Add parameters for %trait placeholder --- TODO.md | 4 +- lib/LogP6/WriterConf/Pattern.pm6 | 80 ++++++++++++++++++++++++++++++-- t/00-grammar.t | 33 ++++++++++++- 3 files changed, 111 insertions(+), 6 deletions(-) diff --git a/TODO.md b/TODO.md index 348e946..a9e4fc1 100644 --- a/TODO.md +++ b/TODO.md @@ -10,8 +10,8 @@ High priority: 1. ~~add `length=0` in `%level` for exact length. make it default~~ Medium priority: -1. use a better exceptions instead of 'die' -1. add params for %trait in pattern (short, long variant) +1. ~~use a better exceptions instead of 'die'~~ +1. ~~add params for %trait in pattern (short, long variant)~~ 1. try make entities be really immutable (filters, writes, loggers) 1. ~~add database writer~~ diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index cc4cc27..fcb2494 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -10,6 +10,49 @@ class Trait does PatternPart { method show($context) { $context.trait } } +role TraitParam does PatternPart { + # not need eviction. if we use trait ones then we use it all program lifetime + has %!cache = %(); + + method calculate($trait) { ... } + + method show($context) { + my $trait = $context.trait; + %!cache{$trait} //= self.calculate($trait); + } +} + +class TraitShort does TraitParam { + has $.separator; + has $.minus; + has $.length; + has $.abreviature; + + method calculate($trait) { + my $parts = $trait.split('::').List; + my $elems = $parts.elems; + if $!length >= $elems || $!length == 0 { + return $parts.join($!separator); + } + my $middle = $!minus ?? $!length !! $elems - $!length; + return $parts.kv.map(-> $i, $p { + if $i < $middle { + $!abreviature ?? substr($p, 0, $!abreviature) !! Any; + } else { + $p; + } + }).grep(*.defined).join($!separator); + } +} + +class TraitSprintf does TraitParam { + has $.placeholder; + + method calculate($trait) { + sprintf($!placeholder, $trait); + } +} + class Tid does PatternPart { method show($context) { $context.tid } } @@ -167,7 +210,12 @@ grammar Grammar is export { proto token item { * } # %trait - logger name (trait) - token item:sym { '%trait' } + token item:sym { '%trait'? } + token trait-params { \{ \} } + proto rule trait-param { * } + rule trait-param:sym + { 'short' '=' '[' $=<-[\]]>+ ']' } + rule trait-param:sym { 'sprintf' '=' } # %tid - thread id token item:sym { '%tid' } # %tname - thread name @@ -221,12 +269,28 @@ grammar Grammar is export { token item:sym { '%framename' } token word { $=<-[\s}]>+ } + token minus { '-' } token num { $=\d+ } + token fract { '.' } + token real-num { ??} } class Actions is export { method TOP($/) { make $>>.made.List } - method item:sym($/) { make Trait } + method item:sym($/) { + with $ { + make $.made + } else { + make Trait + } + } + method trait-params($/) { make $.made } + method trait-param:sym($/) { + make TraitShort.new(:separator($.Str), |$.made); + } + method trait-param:sym($/) { + make TraitSprintf.new(:placeholder($.Str)) + } method item:sym($/) { make Tid } method item:sym($/) { make Tname } method item:sym($/) { make Msg } @@ -283,9 +347,19 @@ class Actions is export { method level-param:sym($/) { make Level::info.Int.Str => $.Str } method level-param:sym($/) { make Level::warn.Int.Str => $.Str } method level-param:sym($/) { make Level::error.Int.Str => $.Str } - method level-param:sym($/) { make 'length' => $.Str } + method level-param:sym($/) { make 'length' => $.made.Str } method level-param:sym($/) { make 'color' => True } method item:sym($/) { make FrameFile } method item:sym($/) { make FrameLine } method item:sym($/) { make FrameName } + method minus($/) { make True } + method num($/) { make $.Int } + method fract($/) { make $.made } + method real-num($/) { + make %( + :length($.made), + :abreviature($.made // 0), + :minus($.made // False) + ); + } } \ No newline at end of file diff --git a/t/00-grammar.t b/t/00-grammar.t index e76564a..2f8b656 100644 --- a/t/00-grammar.t +++ b/t/00-grammar.t @@ -7,7 +7,7 @@ use LogP6::WriterConf::Pattern; use LogP6::Context; use IOString; -plan 15; +plan 42; my LogP6::Context $context .= new; $context.level-set($info); @@ -79,4 +79,35 @@ sub info($frame) { sub foo() { info(callframe) } foo; +$context.trait-set('LogP6::Writer::Async::Std'); +is parse-process('%trait{short=[.]5.4}'), 'LogP6.Writer.Async.Std', '[.]5.4'; +is parse-process('%trait{short=[.]5}' ), 'LogP6.Writer.Async.Std', '[.]5'; +is parse-process('%trait{short=[.]4.2}'), 'LogP6.Writer.Async.Std', '[.]4.4'; +is parse-process('%trait{short=[.]4}' ), 'LogP6.Writer.Async.Std', '[.]4'; +is parse-process('%trait{short=[.]3.2}'), 'Lo.Writer.Async.Std', '[.]3.2'; +is parse-process('%trait{short=[.]3}' ), 'Writer.Async.Std', '[.]3'; +is parse-process('%trait{short=[.]2.3}'), 'Log.Wri.Async.Std', '[.]2.3'; +is parse-process('%trait{short=[.]2}' ), 'Async.Std', '[.]2'; +is parse-process('%trait{short=[.]1.1}'), 'L.W.A.Std', '[.]1.1'; +is parse-process('%trait{short=[.]1}' ), 'Std', '[.]1'; +is parse-process('%trait{short=[.]0.4}'), 'LogP6.Writer.Async.Std', '[.]0.4'; +is parse-process('%trait{short=[.]0}' ), 'LogP6.Writer.Async.Std', '[.]0'; +is parse-process('%trait{short=[.]-0.2}'),'LogP6.Writer.Async.Std', '[.]-0.2'; +is parse-process('%trait{short=[.]-0}' ), 'LogP6.Writer.Async.Std', '[.]-0'; +is parse-process('%trait{short=[.]-1.4}'),'LogP.Writer.Async.Std', '[.]-1.4'; +is parse-process('%trait{short=[.]-1}' ), 'Writer.Async.Std', '[.]-1'; +is parse-process('%trait{short=[.]-2.0}'),'Async.Std', '[.]-2.0'; +is parse-process('%trait{short=[.]-2}' ), 'Async.Std', '[.]-2'; +is parse-process('%trait{short=[.]-3.1}'),'L.W.A.Std', '[.]-3.1'; +is parse-process('%trait{short=[.]-3}' ), 'Std', '[.]-3'; +is parse-process('%trait{short=[.]-4.0}'),'LogP6.Writer.Async.Std', '[.]-4.0'; +is parse-process('%trait{short=[.]-4}' ), 'LogP6.Writer.Async.Std', '[.]-4'; +is parse-process('%trait{short=[.]-5.2}'), 'LogP6.Writer.Async.Std', '[.]-5.2'; +is parse-process('%trait{short=[.]-5}' ), 'LogP6.Writer.Async.Std', '[.]-5'; + +$context.trait-set('LogP6::Writer::Async'); +is parse-process('%trait{sprintf=%.2s}'), 'Lo', '%.2s'; +is parse-process('%trait{sprintf=%21s}'), ' LogP6::Writer::Async', '%21s'; +is parse-process('%trait{sprintf=%-21s}'), 'LogP6::Writer::Async ', '%-21s'; + done-testing; From f7c1e506cd98f42b535eb5ad396060a54a706ba5 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 17 Apr 2019 00:16:35 +0700 Subject: [PATCH 07/26] Add %color placeholder --- lib/LogP6/WriterConf/Pattern.pm6 | 104 +++++++++++++++++++++++++++---- t/00-grammar.t | 8 +++ 2 files changed, 100 insertions(+), 12 deletions(-) diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index fcb2494..af485ec 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -176,13 +176,15 @@ $lnames[LogP6::Level::error.Int] = 'ERROR'; $lnames .= List; my $color = []; -$color[LogP6::Level::trace.Int] = "\e[33m"; # yellow -$color[LogP6::Level::debug.Int] = "\e[32m"; # green; -$color[LogP6::Level::info.Int] = "\e[34m"; # blue; -$color[LogP6::Level::warn.Int] = "\e[35m"; # magenta; -$color[LogP6::Level::error.Int] = "\e[31m"; # red; +$color[LogP6::Level::trace.Int] = "33"; # yellow +$color[LogP6::Level::debug.Int] = "32"; # green; +$color[LogP6::Level::info.Int] = "34"; # blue; +$color[LogP6::Level::warn.Int] = "35"; # magenta; +$color[LogP6::Level::error.Int] = "31"; # red; $color .= List; +my $code = %(:33yellow, :32green, :34blue, :35magenta, :31red); + class LevelName does PatternPart { has $.levels; @@ -191,7 +193,7 @@ class LevelName does PatternPart { my $length = $conf // 0; my $colored = $conf // False; for 1..5 -> $i { - $levels[$i] = $conf{$i.Str} // $levels[$i]; + $levels[$i] = $conf{$i} // $levels[$i]; $levels[$i] = sprintf('%-*.*s', $length, $length, $levels[$i]) if $length > 0; $levels[$i] = $color[$i] ~ $levels[$i] ~ "\e[0m" if $colored; @@ -205,6 +207,33 @@ class LevelName does PatternPart { } } +class Color { ... } +class ColorReset { ... } + +role ColorFactory { + method create($conf) { + return ColorReset if $conf{'reset'}; + my $colors = $color.clone.Array; + for 1..5 -> $i { + $colors[$i] = $conf{$i} // $colors[$i]; + $colors[$i] = "\e[" ~ $colors[$i] ~ 'm'; + } + Color.new(colors => $colors.List); + } +} + +class ColorReset does ColorFactory does PatternPart { + method show($context) { "\e[0m" } +} + +class Color does ColorFactory does PatternPart { + has $.colors; + + method show($context) { + $!colors[$context.level]; + } +} + grammar Grammar is export { token TOP { * } @@ -267,6 +296,23 @@ grammar Grammar is export { token item:sym { '%frameline' } # %framename - frame code name token item:sym { '%framename' } + # %color{TRACE=yellow DEBUG=green INFO=blue WARN=magenta ERROR=red} + # $color{reset} + token item:sym { '%color'? } + token color-params { \{ \} } + proto token color-param { * } + token color-param:sym { + } + token color-param:sym { 'reset' } + proto rule color-level-param { * } + rule color-level-param:sym { 'TRACE' '=' } + rule color-level-param:sym { 'DEBUG' '=' } + rule color-level-param:sym { 'INFO' '=' } + rule color-level-param:sym { 'WARN' '=' } + rule color-level-param:sym { 'ERROR' '=' } + proto token color { * } + token color:sym + { $=('black' | 'white' | 'yellow' | 'green' | 'blue' | 'magenta' | 'red') } + token color:sym { (';')* } token word { $=<-[\s}]>+ } token minus { '-' } @@ -276,7 +322,19 @@ grammar Grammar is export { } class Actions is export { - method TOP($/) { make $>>.made.List } + method TOP($/) { + my $items = $>>.made.List; + my $first = $items.reverse.first(* ~~ ColorFactory); + with $first { + if $first ~~ ColorReset { + make $items; + } else { + make (|$items, ColorReset).List; + } + } else { + make $items; + } + } method item:sym($/) { with $ { make $.made @@ -342,16 +400,38 @@ class Actions is export { } } method level-params($/) { make $>>.made.hash } - method level-param:sym($/) { make Level::trace.Int.Str => $.Str } - method level-param:sym($/) { make Level::debug.Int.Str => $.Str } - method level-param:sym($/) { make Level::info.Int.Str => $.Str } - method level-param:sym($/) { make Level::warn.Int.Str => $.Str } - method level-param:sym($/) { make Level::error.Int.Str => $.Str } + method level-param:sym($/) { make Level::trace.Int => $.Str } + method level-param:sym($/) { make Level::debug.Int => $.Str } + method level-param:sym($/) { make Level::info.Int => $.Str } + method level-param:sym($/) { make Level::warn.Int => $.Str } + method level-param:sym($/) { make Level::error.Int => $.Str } method level-param:sym($/) { make 'length' => $.made.Str } method level-param:sym($/) { make 'color' => True } method item:sym($/) { make FrameFile } method item:sym($/) { make FrameLine } method item:sym($/) { make FrameName } + method item:sym($/) { + with $ { + make ColorFactory.create($.made); + } else { + make ColorFactory.create(%()); + } + } + method color-params($/) { make $.made } + method color-param:sym($/) { make $>>.made.hash } + method color-param:sym($/) { make 'reset' => True } + method color-level-param:sym($/) + { make Level::trace.Int => $.made } + method color-level-param:sym($/) + { make Level::debug.Int => $.made } + method color-level-param:sym($/) + { make Level::info.Int => $.made } + method color-level-param:sym($/) + { make Level::warn.Int => $.made } + method color-level-param:sym($/) + { make Level::error.Int => $.made } + method color:sym($/) { make $code{$.Str} } + method color:sym($/) { make $/.Str } method minus($/) { make True } method num($/) { make $.Int } method fract($/) { make $.made } diff --git a/t/00-grammar.t b/t/00-grammar.t index 2f8b656..9154d3d 100644 --- a/t/00-grammar.t +++ b/t/00-grammar.t @@ -110,4 +110,12 @@ is parse-process('%trait{sprintf=%.2s}'), 'Lo', '%.2s'; is parse-process('%trait{sprintf=%21s}'), ' LogP6::Writer::Async', '%21s'; is parse-process('%trait{sprintf=%-21s}'), 'LogP6::Writer::Async ', '%-21s'; +$context.level-set($warn); +$context.msg-set('msg'); +is parse-process('%msg [%color%level%color{reset}]'), "msg [\e[35mWARN\e[0m]"; +is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), "msg [\e[34mWARN\e[0m]"; +$context.level-set($error); +is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), "msg [\e[31mERROR\e[0m]"; +is parse-process('%msg [%color%level]'), "msg [\e[31mERROR]\e[0m"; + done-testing; From fcaf46ebdd4667d89188e1abcdde01670c342b62 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 17 Apr 2019 00:20:29 +0700 Subject: [PATCH 08/26] Remove 'color' from %level placeholder --- lib/LogP6/WriterConf/Pattern.pm6 | 4 ---- t/00-grammar.t | 19 +++++++++++-------- 2 files changed, 11 insertions(+), 12 deletions(-) diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index af485ec..7220484 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -191,12 +191,10 @@ class LevelName does PatternPart { method new($conf) { my $levels = $lnames.clone.Array; my $length = $conf // 0; - my $colored = $conf // False; for 1..5 -> $i { $levels[$i] = $conf{$i} // $levels[$i]; $levels[$i] = sprintf('%-*.*s', $length, $length, $levels[$i]) if $length > 0; - $levels[$i] = $color[$i] ~ $levels[$i] ~ "\e[0m" if $colored; } self.bless(levels => $levels.List); @@ -289,7 +287,6 @@ grammar Grammar is export { rule level-param:sym { 'WARN' '=' } rule level-param:sym { 'ERROR' '=' } rule level-param:sym { 'length' '=' } - rule level-param:sym { 'color' } # %framefile - frame file path token item:sym { '%framefile' } # %frameline - frame file line @@ -406,7 +403,6 @@ class Actions is export { method level-param:sym($/) { make Level::warn.Int => $.Str } method level-param:sym($/) { make Level::error.Int => $.Str } method level-param:sym($/) { make 'length' => $.made.Str } - method level-param:sym($/) { make 'color' => True } method item:sym($/) { make FrameFile } method item:sym($/) { make FrameLine } method item:sym($/) { make FrameName } diff --git a/t/00-grammar.t b/t/00-grammar.t index 9154d3d..7892da0 100644 --- a/t/00-grammar.t +++ b/t/00-grammar.t @@ -7,7 +7,7 @@ use LogP6::WriterConf::Pattern; use LogP6::Context; use IOString; -plan 42; +plan 45; my LogP6::Context $context .= new; $context.level-set($info); @@ -68,8 +68,6 @@ is parse-process($level-line), '', 'level length 0 warn'; $context.level-set($info); is parse-process($level-line), '', 'level length 0 info'; -is parse-process('colored %level{color}'), "colored \e[34mINFO\e[0m", - 'level color'; # %frame* sub info($frame) { @@ -79,6 +77,7 @@ sub info($frame) { sub foo() { info(callframe) } foo; +# %trait $context.trait-set('LogP6::Writer::Async::Std'); is parse-process('%trait{short=[.]5.4}'), 'LogP6.Writer.Async.Std', '[.]5.4'; is parse-process('%trait{short=[.]5}' ), 'LogP6.Writer.Async.Std', '[.]5'; @@ -104,18 +103,22 @@ is parse-process('%trait{short=[.]-4.0}'),'LogP6.Writer.Async.Std', '[.]-4.0'; is parse-process('%trait{short=[.]-4}' ), 'LogP6.Writer.Async.Std', '[.]-4'; is parse-process('%trait{short=[.]-5.2}'), 'LogP6.Writer.Async.Std', '[.]-5.2'; is parse-process('%trait{short=[.]-5}' ), 'LogP6.Writer.Async.Std', '[.]-5'; - $context.trait-set('LogP6::Writer::Async'); is parse-process('%trait{sprintf=%.2s}'), 'Lo', '%.2s'; is parse-process('%trait{sprintf=%21s}'), ' LogP6::Writer::Async', '%21s'; is parse-process('%trait{sprintf=%-21s}'), 'LogP6::Writer::Async ', '%-21s'; +# %color $context.level-set($warn); $context.msg-set('msg'); -is parse-process('%msg [%color%level%color{reset}]'), "msg [\e[35mWARN\e[0m]"; -is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), "msg [\e[34mWARN\e[0m]"; +is parse-process('%msg [%color%level%color{reset}]'), + "msg [\e[35mWARN\e[0m]", 'color empty with reset'; +is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), + "msg [\e[34mWARN\e[0m]", 'color WARN with reset'; $context.level-set($error); -is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), "msg [\e[31mERROR\e[0m]"; -is parse-process('%msg [%color%level]'), "msg [\e[31mERROR]\e[0m"; +is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), + "msg [\e[31mERROR\e[0m]", 'color WARN error with reset'; +is parse-process('%msg [%color%level]'), + "msg [\e[31mERROR]\e[0m", 'color empty without reset'; done-testing; From 95d4695fc5e1d1f19fa98d0213db27f9ab953f20 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 17 Apr 2019 00:25:50 +0700 Subject: [PATCH 09/26] Add %creset placeholder as %color{reset} alternative --- lib/LogP6/WriterConf/Pattern.pm6 | 4 +++- t/00-grammar.t | 4 +++- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index 7220484..b5d674c 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -294,8 +294,9 @@ grammar Grammar is export { # %framename - frame code name token item:sym { '%framename' } # %color{TRACE=yellow DEBUG=green INFO=blue WARN=magenta ERROR=red} - # $color{reset} + # %color{reset} %creset token item:sym { '%color'? } + token item:sym { '%creset' } token color-params { \{ \} } proto token color-param { * } token color-param:sym { + } @@ -413,6 +414,7 @@ class Actions is export { make ColorFactory.create(%()); } } + method item:sym($/) { make ColorReset } method color-params($/) { make $.made } method color-param:sym($/) { make $>>.made.hash } method color-param:sym($/) { make 'reset' => True } diff --git a/t/00-grammar.t b/t/00-grammar.t index 7892da0..2b6f2e4 100644 --- a/t/00-grammar.t +++ b/t/00-grammar.t @@ -7,7 +7,7 @@ use LogP6::WriterConf::Pattern; use LogP6::Context; use IOString; -plan 45; +plan 46; my LogP6::Context $context .= new; $context.level-set($info); @@ -120,5 +120,7 @@ is parse-process('%msg [%color{WARN=34}%level%color{reset}]'), "msg [\e[31mERROR\e[0m]", 'color WARN error with reset'; is parse-process('%msg [%color%level]'), "msg [\e[31mERROR]\e[0m", 'color empty without reset'; +is parse-process('%msg [%color%level%creset]'), + "msg [\e[31mERROR\e[0m]", 'color empty with creset'; done-testing; From 62c3677529af49d89f1114e4e5b989652407df95 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Thu, 18 Apr 2019 01:10:59 +0700 Subject: [PATCH 10/26] Avoid creating logger instance for each trait Do it for each cliche instead. --- lib/LogP6.pm6 | 122 +++++++++++++++++++++++++++++++++++------------ t/00-factories.t | 36 ++++++++++++-- 2 files changed, 123 insertions(+), 35 deletions(-) diff --git a/lib/LogP6.pm6 b/lib/LogP6.pm6 index 3124720..2a1e429 100644 --- a/lib/LogP6.pm6 +++ b/lib/LogP6.pm6 @@ -35,7 +35,9 @@ my atomicint $initialized = 0; my @cliches; my $cliches-names; +my %cliches-to-traits; my %cliches-to-loggers; +my %cliches-to-loggers-pure; my $loggers-pure = %(); my $loggers = $(); @@ -129,7 +131,9 @@ sub create-default-cliche() { sub clean-all-settings() { @cliches = []; $cliches-names = SetHash.new; + %cliches-to-traits = %(); %cliches-to-loggers = %(); + %cliches-to-loggers-pure = %(); $loggers-pure = %(); $loggers = %(); @@ -593,7 +597,6 @@ multi sub cliche(Str:D :$name!, :$remove! where *.so --> LogP6::Cliche) { my $old = @cliches.grep(*.name eq $name).first // LogP6::Cliche; @cliches = @cliches.grep(*.name ne $name).Array; $cliches-names{$name} = False; - %cliches-to-loggers = %(); update-loggers; $old; }); @@ -662,21 +665,6 @@ sub find-cliche-with(Str:D $name!, @cliches.grep(*.has($name, $type)).List; } -multi sub update-loggers(Positional:D $cliches) { - for |$cliches -> $cliche { - for (%cliches-to-loggers{$cliche.name} // SetHash.new).keys -> $trait { - create-and-store-logger($trait); - } - } -} - -multi sub update-loggers() { - my @traits := atomic-fetch($loggers).keys.List; - for @traits -> $trait { - create-and-store-logger($trait); - } -} - sub change-cliche($old-cliche, $new-cliche) { for @cliches.kv -> $i, $cliche { if $cliche.name eq $old-cliche.name { @@ -728,26 +716,21 @@ sub find-cliche-for-trait($trait) { } sub get-logger(Str:D $trait --> Logger:D) is export(:MANDATORY) { - my %logs = atomic-fetch($loggers); - return $_ with %logs{$trait}; + return $_ with atomic-fetch($loggers){$trait}; $lock.protect({ try-initialize(); - return $_ with $loggers{$trait}; - create-and-store-logger($trait); - return $loggers{$trait} + return find-or-create-and-store-logger($trait); }); CATCH { default { logp6-error($_); return LogP6::LoggerMute.new(:$trait) } } } sub get-logger-pure(Str:D $trait --> Logger:D) is export(:configure) { - my %logs = atomic-fetch($loggers-pure); - return $_ with %logs{$trait}; + return $_ with atomic-fetch($loggers-pure){$trait}; $lock.protect({ try-initialize(); - return $_ with atomic-fetch($loggers-pure){$trait}; - create-and-store-logger($trait); + return find-or-create-and-store-logger-pure($trait); }); CATCH { default { logp6-error($_); return LogP6::LoggerMute.new(:$trait) } } } @@ -768,20 +751,99 @@ sub remove-logger(Str:D $trait --> Logger) is export(:configure) { }); } -sub create-and-store-logger($trait) { +multi sub update-loggers(Positional:D $cliches) { + for |$cliches -> $cliche { + %cliches-to-loggers{$cliche.name}:delete; + %cliches-to-loggers-pure{$cliche.name}:delete; + } + for |$cliches -> $cliche { + my %copy = %cliches-to-traits{$cliche.name} // SetHash.new; + %cliches-to-traits{$cliche.name} = SetHash.new; + for %copy.keys -> $trait { + logger-map-del($trait); + logger-pure-map-del($trait); + find-or-create-and-store-logger($trait); + } + } +} + +multi sub update-loggers() { + %cliches-to-traits = %(); + %cliches-to-loggers = %(); + %cliches-to-loggers-pure = %(); + logger-map-clean(); + my @traits := logger-pure-map-clean(); + for @traits -> $trait { + find-or-create-and-store-logger($trait); + } +} + +sub find-or-create-and-store-logger-pure($trait) { + return $_ with atomic-fetch($loggers-pure){$trait}; + my $cliche = find-cliche-for-trait($trait); + with %cliches-to-loggers-pure{$cliche.name} { + logger-pure-map-put($trait, $_); + return $_; + } + create-and-store-loggers($trait, $cliche, :pure); +} + +sub find-or-create-and-store-logger($trait) { + return $_ with atomic-fetch($loggers){$trait}; my $cliche = find-cliche-for-trait($trait); + with %cliches-to-loggers{$cliche.name} { + logger-map-put($trait, $_); + return $_; + } + create-and-store-loggers($trait, $cliche, :!pure); +} + +sub create-and-store-loggers($trait, $cliche, :$pure) { my $logger-pure = create-logger($trait, $cliche); + my $logger = wrap-logger($logger-pure, $cliche); + logger-map-put($trait, $logger); + logger-pure-map-put($trait, $logger-pure); + (%cliches-to-traits{$cliche.name} //= SetHash.new){$trait} = True; + %cliches-to-loggers{$cliche.name} = $logger; + %cliches-to-loggers-pure{$cliche.name} = $logger-pure; - my %new-loggers = atomic-fetch($loggers).clone; + return $pure ?? $logger-pure !! $logger; +} + +sub logger-pure-map-put($trait, $logger-pure) { my %new-loggers-pure = atomic-fetch($loggers-pure).clone; - %new-loggers{$trait} = wrap-logger($logger-pure, $cliche); %new-loggers-pure{$trait} = $logger-pure; - (%cliches-to-loggers{$cliche.name} //= SetHash.new){$trait} = True; + atomic-assign($loggers-pure, %new-loggers-pure); +} +sub logger-map-put($trait, $logger) { + my %new-loggers = atomic-fetch($loggers).clone; + %new-loggers{$trait} = $logger; atomic-assign($loggers, %new-loggers); +} + +sub logger-pure-map-del($trait) { + my %new-loggers-pure = atomic-fetch($loggers-pure).clone; + %new-loggers-pure{$trait}:delete; atomic-assign($loggers-pure, %new-loggers-pure); +} + +sub logger-map-del($trait) { + my %new-loggers = atomic-fetch($loggers).clone; + %new-loggers{$trait}:delete; + atomic-assign($loggers, %new-loggers); +} + +sub logger-pure-map-clean() { + my $keys = atomic-fetch($loggers-pure).keys.List; + atomic-assign($loggers-pure, %()); + return $keys; +} - return $logger-pure; +sub logger-map-clean() { + my $keys = atomic-fetch($loggers).keys.List; + atomic-assign($loggers, %()); + return $keys; } END { diff --git a/t/00-factories.t b/t/00-factories.t index 1827b5a..7fcb763 100644 --- a/t/00-factories.t +++ b/t/00-factories.t @@ -5,7 +5,7 @@ use LogP6 :configure; use lib './t/resource/Helpers'; use IOString; -plan 5; +plan 6; $*ERR = IOString.new; @@ -126,7 +126,6 @@ my $pattern1 = '[%date{$hh:$mm:$ss}][%level{length=5}] %msg'; my $pattern2 = '%level| %msg'; subtest { - plan 45; my $w-wo-name = writer(:pattern($pattern1), :auto-exceptions, @@ -205,7 +204,6 @@ subtest { }, 'writer'; subtest { - plan 54; use LogP6::Wrapper::Transparent; @@ -302,7 +300,6 @@ subtest { }, 'cliche'; subtest { - plan 18; use lib './t/resource/Helpers'; @@ -323,7 +320,7 @@ subtest { my $log = get-logger('log'); ok $log, 'log cliche'; isnt $log, $any-log, 'default and log are not the same logs'; - isnt get-logger('any2'), $any-log, 'default and another are not the same'; + is get-logger('any2'), $any-log, 'same cliche - same logger'; is get-logger('any'), $any-log, 'same trait - same logger'; is get-logger('log'), $log, 'same trait - same logger again'; @@ -377,4 +374,33 @@ subtest { }, 'logger'; +subtest { + plan 7; + + cliche(:name, :matcher, grooves => + (writer(:name), filter(:name))); + cliche(:name, :matcher(/cliche2/), grooves => + (writer(:name), filter(:name))); + + my $first-log1 = get-logger('cliche1'); + my $first-log2 = get-logger('cliche2'); + isnt $first-log1, $first-log2, 'loggers are different for different cliches'; + + filter(:name, :update); + my $second-log1 = get-logger('cliche1'); + my $second-log2 = get-logger('cliche2'); + isnt $second-log1, $first-log1, 'new logger after update cliche\'s filter'; + is $second-log2, $first-log2, 'same logger after update another cliche'; + isnt $second-log1, $second-log2, 'different logger after update filter'; + + cliche(:name, :matcher(/cliche/), + grooves => ('writer2', 'filter2'), :replace); + my $third-log1 = get-logger('cliche1'); + my $third-log2 = get-logger('cliche2'); + isnt $third-log1, $second-log1, 'new logger after update another cliche'; + isnt $third-log2, $second-log2, 'new logger after update cliche'; + is $third-log1, $third-log2, 'same loggers for updated cliche'; + +}, 'create same logger'; + done-testing; From 483ef44e562b2dbf67eee4064b76b574162e24b0 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Thu, 18 Apr 2019 12:56:19 +0700 Subject: [PATCH 11/26] Remove unnecessary module declaration --- lib/LogP6/ConfigFile.pm6 | 2 -- lib/LogP6/LogGetter.pm6 | 6 ++---- lib/LogP6/WriterConf/Pattern.pm6 | 2 -- 3 files changed, 2 insertions(+), 8 deletions(-) diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index 63be135..11f2d74 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -1,5 +1,3 @@ -unit module LogP6::ConfigFile; - use JSON::Fast; use LogP6::WriterConf::Std; diff --git a/lib/LogP6/LogGetter.pm6 b/lib/LogP6/LogGetter.pm6 index 0d1747a..a1094bd 100644 --- a/lib/LogP6/LogGetter.pm6 +++ b/lib/LogP6/LogGetter.pm6 @@ -1,8 +1,6 @@ -#|[Module with subs for getting wrapped and pure logger. -#| The modules are created for avoid circular dependencies +#|[Subs for getting wrapped and pure logger. +#| The subs are created for avoid circular dependencies #| (many modules depends on LogP6 module).] -unit module LogP6::LogGetter; - my &my-get-wrap; my &my-get-pure; diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index b5d674c..01d5b16 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -1,5 +1,3 @@ -unit module LogP6::WriterConf::Pattern; - use LogP6::Level; role PatternPart { From 42b5c7e97e7fd4cc318ee436ebb0cc6802e2be39 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Thu, 18 Apr 2019 16:13:29 +0700 Subject: [PATCH 12/26] Reuse handles from previous config file read --- lib/LogP6.pm6 | 5 +- lib/LogP6/ConfigFile.pm6 | 437 +++++++++++++++++++++------------------ t/00-config-file.t | 17 +- 3 files changed, 248 insertions(+), 211 deletions(-) diff --git a/lib/LogP6.pm6 b/lib/LogP6.pm6 index 2a1e429..984ab6f 100644 --- a/lib/LogP6.pm6 +++ b/lib/LogP6.pm6 @@ -33,6 +33,8 @@ our $error is export(:configure) = Level::error; my Lock $lock .= new; my atomicint $initialized = 0; +my LogP6::ConfigFile $config-file; + my @cliches; my $cliches-names; my %cliches-to-traits; @@ -95,7 +97,8 @@ sub init-from-file($config-path) is export(:configure) { my $config; try { - $config = parse-config($config-path); + $config-file //= LogP6::ConfigFile.new; + $config = $config-file.parse-config($config-path); CATCH { default { logp6-error($_); wipe-log-config() if $prev-initialized == 0; diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index 11f2d74..3809064 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -30,244 +30,271 @@ sub default-config-path() is export { } } -sub parse-config(IO() $file-path) is export { - CATCH { - default { - die "Cannot read and create config from file $file-path. Cause " - ~ $_.^name ~ ': ' ~ $_.Str +class LogP6::ConfigFile { + has %!current-cache = %(); + has %!previous-cache = %(); + + method parse-config(LogP6::ConfigFile:D: IO() $file-path) is export { + CATCH { + default { + die "Cannot read and create config from file $file-path. Cause " + ~ $_.^name ~ ': ' ~ $_.gist, "\n"; + } } - } - return LogP6::Config.new unless $file-path.e; - my $file-content = slurp($file-path).trim; - die "config file $file-path is empty" if $file-content.chars == 0; - my \conf = from-json($file-content); + return LogP6::Config.new unless $file-path.e; + my $file-content = slurp($file-path).trim; + die "config file $file-path is empty" if $file-content.chars == 0; + my \conf = from-json($file-content); - return LogP6::Config.new( - writers => list(conf, &writer), - filters => list(conf, &filter), - cliches => list(conf, &cliche), - default-pattern => string-e(conf), - default-auto-exceptions => bool(conf), - default-handle => handle(conf), - default-x-pattern => string-e(conf), - default-level => level(conf), - default-first-level-check => bool(conf), - default-wrapper => wrapper(conf) - ); -} + my $config = LogP6::Config.new( + writers => self!list(conf, 'writer'), + filters => self!list(conf, 'filter'), + cliches => self!list(conf, 'cliche'), + default-pattern => self!string-e(conf), + default-auto-exceptions => self!bool(conf), + default-handle => self!handle(conf), + default-x-pattern => self!string-e(conf), + default-level => self!level(conf), + default-first-level-check => self!bool(conf), + default-wrapper => self!wrapper(conf) + ); -sub list(\json, &each) { - return () without json; - return json.map({each($_)}).eager.List; -} + %!previous-cache = %!current-cache; + %!current-cache = %(); -sub writer(\json) { - given json { - when 'std' { - return LogP6::WriterConf::Std.new( - name => json // Str, - pattern => string-e(json), - handle => handle(json), - auto-exceptions => bool(json) - ); - } - when 'custom' { - return custom(json); - } - default { - die "Wrong writer type $_"; - } + return $config; } -} -sub filter(\json) { - given json { - when 'std' { - return LogP6::FilterConf::Std.new( - name => json // Str, - level => level(json), - first-level-check => json // Bool, - before-check => list(json, &custom), - after-check => list(json, &custom) - ); - } - when 'custom' { - return custom(json); + method !list(\json, $each) { + return () without json; + return json.map({self!"$each"($_)}).eager.List; + } + + method !writer(\json) { + given json { + when 'std' { + return LogP6::WriterConf::Std.new( + name => json // Str, + pattern => self!string-e(json), + handle => self!handle(json), + auto-exceptions => self!bool(json) + ); + } + when 'custom' { + return self!custom(json); + } + default { + die "Wrong writer type $_"; + } } - default { - die "Wrong filter type $_"; + } + + method !filter(\json) { + given json { + when 'std' { + return LogP6::FilterConf::Std.new( + name => json // Str, + level => self!level(json), + first-level-check => json // Bool, + before-check => self!list(json, 'custom'), + after-check => self!list(json, 'custom') + ); + } + when 'custom' { + return self!custom(json); + } + default { + die "Wrong filter type $_"; + } } } -} -sub bool(\json) { - return Bool without json; - return json if json ~~ Bool; - return json eq 'true'; -} + method !bool(\json) { + return Bool without json; + return json if json ~~ Bool; + return json eq 'true'; + } -sub string(\json) { - return json.Str with json; - return Str; -} + method !string(\json) { + return json.Str with json; + return Str; + } -sub string-e(\json) { - return json.Str.trans(['\e', '\a', '\n'] => ["\e", "\a", "\n"]) with json; - return Str; -} + method !string-e(\json) { + return json.Str.trans(['\e', '\a', '\n'] => ["\e", "\a", "\n"]) with json; + return Str; + } -sub cliche(\json) { - my $name = json; - die 'Missing cliche\'s name' without $name; - return LogP6::Cliche.new( - name => $name, - matcher => matcher(json), - default-pattern => string-e(json), - default-auto-exceptions => bool(json), - default-handle => handle(json), - default-x-pattern => string-e(json), - default-level => level(json), - default-first-level-check => bool(json), - grooves => list(json, &string), - wrapper => wrapper(json) - ); -} + method !cliche(\json) { + my $name = json; + die 'Missing cliche\'s name' without $name; + return LogP6::Cliche.new( + name => $name, + matcher => self!matcher(json), + default-pattern => self!string-e(json), + default-auto-exceptions => self!bool(json), + default-handle => self!handle(json), + default-x-pattern => self!string-e(json), + default-level => self!level(json), + default-first-level-check => self!bool(json), + grooves => self!list(json, 'string'), + wrapper => self!wrapper(json) + ); + } -sub handle(\json) { - return IO::Handle without json; - given json { - when 'std' { - my $path = json; - given $path { - when 'out' { - return $*OUT; - } - when 'err' { - return $*ERR; - } - default { - die "Wrong std handle path $path"; + method !handle(\json) { + return IO::Handle without json; + given json { + when 'std' { + my $path = json; + given $path { + when 'out' { + return $*OUT; + } + when 'err' { + return $*ERR; + } + default { + die "Wrong std handle path $path"; + } } } - } - when 'file' { - my $path = json; - die 'Missing file handle path' without $path; - my $append = json; - my $not-append = ($append eqv False); - return $path.IO.open(:create, append => !$not-append); - } - when 'custom' { - return custom(json); - } - default { - die "Wrong handle type $_";; + when 'file' { + my $path = json; + die 'Missing file handle path' without $path; + return self!produce(:use-cache, json, sub { + my $mode = (json // True) ?? :a !! :w; + return $path.IO.open(|$mode); + }); + } + when 'custom' { + return self!custom(json, :use-cache); + } + default { + die "Wrong handle type $_"; + ; + } } } -} -sub any(\json) { - return list(json, &any) if json ~~ Positional; - if json ~~ Associative { - return custom(json) if json ~~ 'custom'; - return associative(json); + method !any(\json) { + return self!list(json, 'any') if json ~~ Positional; + if json ~~ Associative { + return self!custom(json) if json ~~ 'custom'; + return self!associative(json); + } + return json; } - return json; -} -sub associative(\json) { - return %() without json; - return Any unless json ~~ Associative; - my %result = %(); - return json.kv.map(-> $k, $v { $k => any($v) }).Hash; -} + method !associative(\json) { + return %() without json; + return Any unless json ~~ Associative; + my %result = %(); + return json.kv.map(-> $k, $v { $k => self!any($v) }).Hash; + } -sub custom(\json) { - my \my-require = json; - die 'Missing \'require\' field in custom definition' without my-require; + method !custom(\json, :$use-cache) { + my \my-require = json; + die 'Missing \'require\' field in custom definition' without my-require; - my \fqn-method = json; - my \fqn-class = json; - die "Missing both 'fqn-method' and 'fqn-class' fields in custom definition" - if !fqn-method.defined && !fqn-class.defined; - die "Defined both 'fqn-method' and 'fqn-class' fields in custom definition" - if fqn-method.defined && fqn-class.defined; + my \fqn-method = json; + my \fqn-class = json; + die "Missing both 'fqn-method' and 'fqn-class' fields in custom definition" + if !fqn-method.defined && !fqn-class.defined; + die "Defined both 'fqn-method' and 'fqn-class' fields in custom definition" + if fqn-method.defined && fqn-class.defined; - my \args = associative(json); - die "'args' field are not Associative in cusom definition" - unless args ~~ Associative; + my \args = self!associative(json); + die "'args' field are not Associative in cusom definition" + unless args ~~ Associative; - require ::(my-require); - with fqn-method { - my &method = ::(fqn-method); - return method(|args); - } - with fqn-class { - my $class-name = ::(fqn-class); - return $class-name.new(|args); + return self!produce(:$use-cache, json, sub { + require ::(my-require); + with fqn-method { + my &method = ::(fqn-method); + return method(|args); + } + with fqn-class { + my $class-name = ::(fqn-class); + return $class-name.new(|args); + } + }); } -} -sub level(\json) { - return LogP6::Level without json; - given json { - when 'trace' { return LogP6::Level::trace; } - when 'debug' { return LogP6::Level::debug; } - when 'info' { return LogP6::Level::info; } - when 'warn' { return LogP6::Level::warn; } - when 'error' { return LogP6::Level::error; } - default { die 'wrong level value ' ~ json; } + method !level(\json) { + return LogP6::Level without json; + given json { + when 'trace' { return LogP6::Level::trace; } + when 'debug' { return LogP6::Level::debug; } + when 'info' { return LogP6::Level::info; } + when 'warn' { return LogP6::Level::warn; } + when 'error' { return LogP6::Level::error; } + default { die 'wrong level value ' ~ json; } + } } -} -sub matcher(\json) { - die 'Missing cliche\'s matcher' without json; - my $matcher = json.Str; - given $matcher { - when /^ \/ .+ \/ $/ { - my $substr = $matcher.substr(1, *-1); - return / <$substr> /; - } - default { - return $matcher; + method !matcher(\json) { + die 'Missing cliche\'s matcher' without json; + my $matcher = json.Str; + given $matcher { + when /^ \/ .+ \/ $/ { + my $substr = $matcher.substr(1, *-1); + return / <$substr> /; + } + default { + return $matcher; + } } } -} -sub wrapper(\json) { - return LogP6::Wrapper without json; - given json { - when 'time' { - die "Missing 'seconds' field in time wrapper-factory" - without json; - return custom(%( - :type, - :require, - :fqn-class, - args => %( - seconds => json, - config-path => json - ) - )); - } - when 'each' { - return custom(%( - :type, - :require, - :fqn-class, - args => %( - config-path => json - ) - )); - } - when 'transparent' { - return LogP6::Wrapper::Transparent::Wrapper.new; - } - when 'custom' { - return custom(json); + method !wrapper(\json) { + return LogP6::Wrapper without json; + given json { + when 'time' { + die "Missing 'seconds' field in time wrapper-factory" + without json; + return self!custom(%( + :type, + :require, + :fqn-class, + args => %( + seconds => json, + config-path => json + ) + )); + } + when 'each' { + return self!custom(%( + :type, + :require, + :fqn-class, + args => %( + config-path => json + ) + )); + } + when 'transparent' { + return LogP6::Wrapper::Transparent::Wrapper.new; + } + when 'custom' { + return self!custom(json); + } + defined { + die 'wrong wrapper type value ' ~ json; + } } - defined { - die 'wrong wrapper type value ' ~ json; + } + + method !produce(\json, &factory, :$use-cache) { + return factory() unless $use-cache; + my $key = to-json(json, :sorted-keys); + with %!previous-cache{$key} { + %!current-cache{$key} = $_; + return $_; } + return $_ with %!current-cache{$key}; + my $result = factory(); + %!current-cache{$key} = $result; + return $result; } } \ No newline at end of file diff --git a/t/00-config-file.t b/t/00-config-file.t index b99e04b..f198d42 100644 --- a/t/00-config-file.t +++ b/t/00-config-file.t @@ -9,6 +9,7 @@ use IOString; plan 8; $*ERR = IOString.new; +my LogP6::ConfigFile $config .= new; subtest { plan 1; @@ -28,10 +29,10 @@ subtest { }, 'miss file'; subtest { - plan 18; + plan 20; my ($w, $cn); - $cn = parse-config('./t/resource/00-config-file/log-p6-1.json'); + $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); is $cn.writers.elems, 5, 'parsed 5 writers'; @@ -60,6 +61,12 @@ subtest { nok $w.handle, 'w5 handle'; nok $w.auto-exceptions, 'w5 auto-exceptions'; + my $w0h = $cn.writers[0].handle.WHICH; + my $w3h = $cn.writers[3].handle.WHICH; + $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); + is $w0h, $cn.writers[0].handle.WHICH, 'get file handle from cache'; + is $w3h, $cn.writers[3].handle.WHICH, 'get custom handle from cache'; + }, 'writers'; subtest { @@ -70,7 +77,7 @@ subtest { use Custom; my ($f, $cn); - $cn = parse-config('./t/resource/00-config-file/log-p6-1.json'); + $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); is $cn.filters.elems, 4, 'parsed 4 writers'; @@ -110,7 +117,7 @@ subtest { use LogP6::Wrapper::Transparent; my ($c, $cn); - $cn = parse-config('./t/resource/00-config-file/log-p6-1.json'); + $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); is $cn.cliches.elems, 2, 'parsed 2 cliches'; @@ -147,7 +154,7 @@ subtest { use LogP6::Wrapper::SyncTime; - my $cn = parse-config('./t/resource/00-config-file/log-p6-1.json'); + my $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); is $cn.default-pattern, '%msg', 'default-pattern'; is $cn.default-auto-exceptions, False, 'default-auto-exceptions'; is $cn.default-handle, $*ERR, 'default-handle'; From 4e78317ae039d2f5d0792e3646d75c132995f47e Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Thu, 18 Apr 2019 16:13:51 +0700 Subject: [PATCH 13/26] Speedup async tests --- t/00-handle-async-single.t | 2 +- t/00-handle-async.t | 2 +- t/00-logger-wrapper-sync-time.t | 10 +++++----- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/t/00-handle-async-single.t b/t/00-handle-async-single.t index db05cc4..2e7a037 100644 --- a/t/00-handle-async-single.t +++ b/t/00-handle-async-single.t @@ -12,7 +12,7 @@ my LogP6::Handle::AsyncSingle $async .= new: :$delegate; $async.say('boom'); $async.say('moob'); -sleep 2; +sleep(1); is $delegate.clean, "boom\nmoob\n", 'delegate writes'; diff --git a/t/00-handle-async.t b/t/00-handle-async.t index 21b172b..22ed507 100644 --- a/t/00-handle-async.t +++ b/t/00-handle-async.t @@ -12,7 +12,7 @@ my LogP6::Handle::Async $async .= new: :$delegate; $async.say('boom'); $async.say('moob'); -sleep 2; +sleep(1); is $delegate.clean, "boom\nmoob\n", 'delegate writes'; diff --git a/t/00-logger-wrapper-sync-time.t b/t/00-logger-wrapper-sync-time.t index af247f1..797b3cc 100644 --- a/t/00-logger-wrapper-sync-time.t +++ b/t/00-logger-wrapper-sync-time.t @@ -13,7 +13,7 @@ writer(:name, :handle($h), :pattern<%msg>); filter(:name, :level($info)); set-default-wrapper( - LogP6::Wrapper::SyncTime::Wrapper.new(:2seconds)); + LogP6::Wrapper::SyncTime::Wrapper.new(:1seconds)); subtest { plan 3; @@ -27,13 +27,13 @@ subtest { $cliche = cliche(:name($cliche.name), :matcher($cliche.matcher), :replace); $logger.info('log this yet'); - sleep(3); + sleep(1.2); $logger.info('ignore this'); is $h.clean, "log this yet\n", 'mute logger is turned off'; cliche(:name($cliche.name), :matcher($cliche.matcher), grooves => , :replace); - sleep(3); + sleep(1.2); $logger.info('log this'); is $h.clean.trim, 'log this', 'mute logger worked again'; }, 'mute sync'; @@ -50,12 +50,12 @@ subtest { filter(:name, :level($error), :replace); $logger.info('log this yet'); - sleep(3); + sleep(1); $logger.info('ignore this'); is $h.clean, "log this yet\n", 'general logger is changed level'; filter(:name, :level($info), :replace); - sleep(3); + sleep(1); $logger.info('log this'); is $h.clean.trim, 'log this', 'general logger worked again'; }, 'general sync'; From bf946138d9c30e0806084dac9b5f3878a971eb79 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Thu, 18 Apr 2019 16:14:10 +0700 Subject: [PATCH 14/26] Small README change --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 9e9d304..d30694e 100644 --- a/README.md +++ b/README.md @@ -450,8 +450,8 @@ array. Only `std` (for standard configuration) and `custom` types are supported. In case of standard configuration all field are optional excepts `name`. Handle can be: -- `file` type for output into file. You can specify `path` and `append` -arguments; +- `file` type for output into file. You can specify `path` and `append` (`True` +by default) arguments; - `std` type for output into `$*OUT` or `$*ERR`. You can specify `path` as `out` or `err`. - `custom` type. From 85feaef3030c6bf216d5bb72f2ff34e70aad7cf3 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Fri, 19 Apr 2019 14:46:15 +0700 Subject: [PATCH 15/26] Add possibility to backup and restore NDC and MDC values --- TODO.md | 4 ++-- lib/LogP6/Context.pm6 | 13 ++++++++++- lib/LogP6/Logger.pm6 | 4 ++++ lib/LogP6/LoggerPure.pm6 | 36 ++++++++++++++++++++--------- lib/LogP6/Wrapper/SyncAbstract.pm6 | 12 ++++++---- t/00-logger-wrapper-sync-abstract.t | 8 +++++++ t/00-logger.t | 29 ++++++++++++++++++++++- 7 files changed, 86 insertions(+), 20 deletions(-) diff --git a/TODO.md b/TODO.md index a9e4fc1..e3a3ef5 100644 --- a/TODO.md +++ b/TODO.md @@ -12,14 +12,14 @@ High priority: Medium priority: 1. ~~use a better exceptions instead of 'die'~~ 1. ~~add params for %trait in pattern (short, long variant)~~ -1. try make entities be really immutable (filters, writes, loggers) 1. ~~add database writer~~ Low priority: 1. add 'turn off/on cliche' factory method 1. add trace-some methods in logger (like 'returns value', 'enter method with') -1. add backup/restore ndc and mdc +1. try make entities be really immutable (filters, writes, loggers) +1. ~~add backup/restore ndc and mdc~~ Archive: diff --git a/lib/LogP6/Context.pm6 b/lib/LogP6/Context.pm6 index abcf375..86f75cd 100644 --- a/lib/LogP6/Context.pm6 +++ b/lib/LogP6/Context.pm6 @@ -27,7 +27,7 @@ submethod BUILD() { } submethod TWEAK( - :$msg, :$date, :$level, :$x, :$trait, :@ndc, :%mdc, :$callframe + :$msg, :$date, :$level, :$x, :$trait, :@ndc, :%mdc, :$callframe ) { $!msg = $msg; $!date = $date; @@ -147,6 +147,17 @@ method mdc-clean() { %!mdc = %(); } +#| Get copy of NDC and MDC +method dc-get() { + %('ndc' => @!ndc.clone, 'mdc' => %!mdc.clone); +} + +#| Restore values of NDC and MDC from its copy +method dc-restore($dc) { + @!ndc := $dc // []; + %!mdc = $dc // %(); +} + #|[Gets current DataTime.now value. The value will be cache until the date will #| be set to undefined value by .date-set(), .date-clean() or clean() methods. #| Normally the date value are reset before each logging without addition user's diff --git a/lib/LogP6/Logger.pm6 b/lib/LogP6/Logger.pm6 index e7b9ea1..9c79497 100644 --- a/lib/LogP6/Logger.pm6 +++ b/lib/LogP6/Logger.pm6 @@ -18,6 +18,10 @@ role LogP6::Logger { method mdc-remove($key) { ... } #| Cleans MDC method mdc-clean() { ... } + #| Get copy of NDC and MDC + method dc-copy() { ... } + #| Restore values of NDC and MDC from its copy + method dc-restore($dc) { ... } #|[Writes log with trace importance level. #| @args - data for logging. If the array has more then one element then the #| first element is used as format for sprintf sub and the rest element as diff --git a/lib/LogP6/LoggerPure.pm6 b/lib/LogP6/LoggerPure.pm6 index 0faeaf7..1c7240b 100644 --- a/lib/LogP6/LoggerPure.pm6 +++ b/lib/LogP6/LoggerPure.pm6 @@ -20,63 +20,75 @@ class LogP6::LoggerPure does LogP6::Logger { } method ndc-push($obj) { - get-context.ndc-push: $obj; CATCH { default { logp6-error($_) } } + get-context.ndc-push: $obj; } method ndc-pop() { - get-context.ndc-pop; - CATCH { default { logp6-error($_) } } + try { + CATCH { default { logp6-error($_) } } + return get-context.ndc-pop; + } } method ndc-clean() { - get-context.ndc-clean; CATCH { default { logp6-error($_) } } + get-context.ndc-clean; } method mdc-put($key, $obj) { - get-context.mdc-put: $key, $obj; CATCH { default { logp6-error($_) } } + get-context.mdc-put: $key, $obj; } method mdc-remove($key) { - get-context.mdc-remove: $key; CATCH { default { logp6-error($_) } } + get-context.mdc-remove: $key; } method mdc-clean() { + CATCH { default { logp6-error($_) } } get-context.mdc-clean; + } + + method dc-copy() { CATCH { default { logp6-error($_) } } + get-context.dc-get; + } + + method dc-restore($dc) { + CATCH { default { logp6-error($_) } } + get-context.dc-restore($dc); } method trace(*@args, :$x) { + CATCH { default { logp6-error($_) } } return if $!reactive-level > LogP6::Level::trace; self!log(LogP6::Level::trace, @args, :$x); - CATCH { default { logp6-error($_) } } } method debug(*@args, :$x) { + CATCH { default { logp6-error($_) } } return if $!reactive-level > LogP6::Level::debug; self!log(LogP6::Level::debug, @args, :$x); - CATCH { default { logp6-error($_) } } } method info(*@args, :$x) { + CATCH { default { logp6-error($_) } } return if $!reactive-level > LogP6::Level::info; self!log(LogP6::Level::info, @args, :$x); - CATCH { default { logp6-error($_) } } } method warn(*@args, :$x) { + CATCH { default { logp6-error($_) } } return if $!reactive-level > LogP6::Level::warn; self!log(LogP6::Level::warn, @args, :$x); - CATCH { default { logp6-error($_) } } } method error(*@args, :$x) { + CATCH { default { logp6-error($_) } } return if $!reactive-level > LogP6::Level::error; self!log(LogP6::Level::error, @args, :$x); - CATCH { default { logp6-error($_) } } } submethod !log($level, @args, :$x) { @@ -110,6 +122,8 @@ class LogP6::LoggerMute does LogP6::Logger { method mdc-put($key, $obj) {} method mdc-remove($key) {} method mdc-clean() {} + method dc-copy() { Nil } + method dc-restore($dc) {} method trace(*@args, :$x) { get-context().clean } method debug(*@args, :$x) { get-context().clean } method info(*@args, :$x) { get-context().clean } diff --git a/lib/LogP6/Wrapper/SyncAbstract.pm6 b/lib/LogP6/Wrapper/SyncAbstract.pm6 index a97c776..6c10561 100644 --- a/lib/LogP6/Wrapper/SyncAbstract.pm6 +++ b/lib/LogP6/Wrapper/SyncAbstract.pm6 @@ -40,11 +40,13 @@ class LogP6::Wrapper::SyncAbstract does LogP6::Logger { method mdc-put($key, $obj) { $!aggr.mdc-put($key, $obj) } method mdc-remove($key) { $!aggr.mdc-remove($key) } method mdc-clean() { $!aggr.mdc-clean() } - method trace(*@args, :$x) { self.sync(get-context); $!aggr.trace(|@args, :$x)} - method debug(*@args, :$x) { self.sync(get-context); $!aggr.debug(|@args, :$x)} - method info(*@args, :$x) { self.sync(get-context); $!aggr.info(|@args, :$x)} - method warn(*@args, :$x) { self.sync(get-context); $!aggr.warn(|@args, :$x)} - method error(*@args, :$x) { self.sync(get-context); $!aggr.error(|@args, :$x)} + method dc-copy() { $!aggr.dc-copy } + method dc-restore($dc) { $!aggr.dc-restore($dc) } + method trace(*@args, :$x) { self.sync(get-context); $!aggr.trace(|@args, :$x)} + method debug(*@args, :$x) { self.sync(get-context); $!aggr.debug(|@args, :$x)} + method info(*@args, :$x) { self.sync(get-context); $!aggr.info(|@args, :$x)} + method warn(*@args, :$x) { self.sync(get-context); $!aggr.warn(|@args, :$x)} + method error(*@args, :$x) { self.sync(get-context); $!aggr.error(|@args, :$x)} } #|[Wrapper logic for synchronize a configuration and a logger. diff --git a/t/00-logger-wrapper-sync-abstract.t b/t/00-logger-wrapper-sync-abstract.t index bdff653..cfc6cae 100644 --- a/t/00-logger-wrapper-sync-abstract.t +++ b/t/00-logger-wrapper-sync-abstract.t @@ -41,6 +41,14 @@ class MockLogger does LogP6::Logger { @!calls.push('mdc-clean'); } + method dc-copy() { + @!calls.push('dc-copy'); + } + + method dc-restore($dc) { + @!calls.push('dc-restore' ~ $dc); + } + method trace(*@args, :$x) { @!calls.push('trace' ~ @args.join('') ~ ((defined $x) ?? $x.message !! '')); } diff --git a/t/00-logger.t b/t/00-logger.t index 9497da2..c939853 100644 --- a/t/00-logger.t +++ b/t/00-logger.t @@ -7,7 +7,7 @@ use LogP6::LoggerPure; use IOString; use LogP6 :configure; -plan 7; +plan 8; my ($h1, $h2, $h3) = (IOString.new xx 3).list; sub clean-io() { $_.clean for ($h1, $h2, $h3) } @@ -203,4 +203,31 @@ subtest { isnt $l1, $l2, 'frames are different'; }, 'different frame'; +subtest { + plan 5; + + my LogP6::Logger $log = get-logger(''); + + $log.ndc-clean; + $log.mdc-clean; + $log.mdc-put('k', 'v'); + $log.mdc-put('v', 'k'); + my $dc = $log.dc-copy; + $log.mdc-clean; + nok $log.mdc-remove('k'), 'mdc is empty'; + $log.dc-restore($dc); + is $log.mdc-remove('k'), 'v', 'mdc restore k'; + is $log.mdc-remove('v'), 'k', 'mdc restore v'; + + $log.mdc-clean; + $log.ndc-push: 'k'; + $log.ndc-push: 'v'; + $dc = $log.dc-copy; + $log.ndc-clean; + $log.dc-restore($dc); + is $log.ndc-pop, 'v', 'ndc restore v'; + is $log.ndc-pop, 'k', 'ndc restore k'; + +}, 'dc-copy'; + done-testing; From 052488a114592c21c4f332d144318da5945667e9 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Fri, 19 Apr 2019 15:45:39 +0700 Subject: [PATCH 16/26] Add documentation for %color placeholder --- README.md | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index d30694e..19d8d30 100644 --- a/README.md +++ b/README.md @@ -376,16 +376,24 @@ subpattern. `$msg` - optional exception message, `$name` - optional exception name, `$trace` - optional exception stacktrace. For example, `'%x{($name "$msg") Trace: $trace}'` can be converted into `'(X::AdHoc "test exception") Trace: ...'`; -- `%level{WARN=W DEBUG=D ERROR=E TRACE=T INFO=I length=2 color}` - log importance +- `%level{WARN=W DEBUG=D ERROR=E TRACE=T INFO=I length=2}` - log importance level. By default logger will use level name in upper case but you can specify synonyms for all or part of them in curly brackets in format `=`. You can specify a fixed length of log level name. Default length is 0 - write level as is. For example `'[%level{WARN=hmm ERROR=alarm length=5}]'` can be converted into -`'[hmm ]'`, `'[alarm]'`, `'[INFO ]'`, `'[DEBUG]'`. Also you can specify `color` -argument - then level names will be colored in output. It is turned off as -default. If you want to use color you prefer then you need to surround level -names with `\e[m` and `\e[0m`; +`'[hmm ]'`, `'[alarm]'`, `'[INFO ]'`, `'[DEBUG]'`; +- `%color{TRACE=yellow DEBUG=green INFO=blue WARN=magenta ERROR=red}` - colorize +log string after that placeholder. You can specify color for any log level. +Level you not specified color will be use its default color (as in example +above). For example, `%color{ERROR=green}` means +`%color{TRACE=yellow DEBUG=green INFO=blue WARN=magenta ERROR=green}`. You can +use `yellow`, `green`, `blue`, `magenta`, `green` color names or color code +(more [information](https://misc.flogisoft.com/bash/tip_colors_and_formatting). +For example `%color{TRACE=35 DEBUG=30;48;5;82 INFO=green}`. You can use `%color` +placeholder several times; +- `%color{reset}` or `%creset` - reset log string colorizing after that +placeholder; - `%date{$yyyy-$yy-$MM-$MMM-$dd $hh:$mm:$ss:$mss $z}` - current date and time. String in curly brackets is used as subpattern. From a4edae67bf176642be2a15ea6cf756027ef63d78 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Fri, 19 Apr 2019 17:05:54 +0700 Subject: [PATCH 17/26] Update documentation for %trait placeholder --- README.md | 16 +++++++++++++--- lib/LogP6/WriterConf/Pattern.pm6 | 2 +- 2 files changed, 14 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 19d8d30..a212431 100644 --- a/README.md +++ b/README.md @@ -365,7 +365,17 @@ placeholder name. Pattern can has the following placeholders: -- `%trait` - for name of logger trait; +- `%trait`, `%trait{short=[package-delimeter]number}`, `%trait{sprintf=pattern}` - +for name of logger trait. Additionally you can specify one of two options of +trait representation. `sprintf` option is useful for traits like `database`, +`audit` or so, when you want to represent all traits with the same length. For +example, `[%trait{sprintf=%s7}]` can be converted into `[ audit]`. `short` +option is useful for traits like `Module::Packge1::Package2::Class`. You can +specify package delimiter (instead of `::`) and how many packages will be +displayed. For example, `%trait{short=[.]2` can be converted into +`Package2.Class`, and `%trait{short=[.]2.4` - into `Modu.Pack.Package2.Class`. +If `number` is integer then only `number` right elements will be displayed. If +`number` is real then left elements will be cut to fractional symbols; - `%tid` - for current `Thread` id; - `%tname` - for current `Thread` name; - `%msg` - for log message; @@ -410,7 +420,7 @@ at the same log call line; `callframe().code.name` in log call block; Note that using `%framefile`, `%frameline` or `%framename` in the pattern will -slow your program because it requires several `callframe()` calls on each +slow your logging because it requires several `callframe()` calls on each resultative log call; ### Async writing @@ -426,7 +436,7 @@ as writer configuration of another configuration. Final writer will schedule `write` method call of `delegate` created writer with copy of current `logger context`. If you miss a `:name` parameter then `delegate`'s name will be used. Pass boolean parameter `need-callframe` if you plan to use callframe -information in wrapped writer. Note that using callframe will slow your program +information in wrapped writer. Note that using callframe will slow your logging because it requires several `callframe()` calls on each resultative log call. ### Writer factory subroutines diff --git a/lib/LogP6/WriterConf/Pattern.pm6 b/lib/LogP6/WriterConf/Pattern.pm6 index 01d5b16..835a831 100644 --- a/lib/LogP6/WriterConf/Pattern.pm6 +++ b/lib/LogP6/WriterConf/Pattern.pm6 @@ -234,7 +234,7 @@ grammar Grammar is export { token TOP { * } proto token item { * } - # %trait - logger name (trait) + # %trait{short=[delimiter]number printf=%6s} - logger name (trait) token item:sym { '%trait'? } token trait-params { \{ \} } proto rule trait-param { * } From 25d3081fdb6755fa72be3b56c089978a37e27891 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Fri, 19 Apr 2019 17:09:48 +0700 Subject: [PATCH 18/26] Add documentation for dc- logger methods --- README.md | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index a212431..9765692 100644 --- a/README.md +++ b/README.md @@ -211,8 +211,11 @@ current date-time and so from the context. Logger has the following methods: - `trait()` - returns logger trait; -- `ndc-push($obj)`, `ndc-pop()`, `ndc-clean()` - work with NDC; -- `mdc-put($key, $obj)`, `mdc-remove($key)`, `mdc-clean()` - work with MDC; +- `ndc-push($obj)`, `ndc-pop()`, `ndc-clean()` - work with `NDC`; +- `mdc-put($key, $obj)`, `mdc-remove($key)`, `mdc-clean()` - work with `MDC`; +- `dc-copy()`, `dc-restore($dc-copy)` - make copy of `NDC` and `MDC` and restore +them from copy. The methods are useful when you want to share NDC and MDC values +across multiple threads. - `trace(*@args, :$x)`, `debug(*@args, :$x)`, `info(*@args, :$x)`, `warn(*@args, :$x)`, `error(*@args, :$x)` - logging the arguments with specified importance log level. `:$x` is an optional exception argument. `@args` - data From b25cca8911cd29d0bba93b5e8e849d489ff29109 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sat, 20 Apr 2019 16:46:42 +0700 Subject: [PATCH 19/26] Revert logic of 62c36775 (Avoid creating logger instance for each trait) --- lib/LogP6.pm6 | 20 -------------------- t/00-factories.t | 33 ++------------------------------- 2 files changed, 2 insertions(+), 51 deletions(-) diff --git a/lib/LogP6.pm6 b/lib/LogP6.pm6 index 984ab6f..f53c293 100644 --- a/lib/LogP6.pm6 +++ b/lib/LogP6.pm6 @@ -38,8 +38,6 @@ my LogP6::ConfigFile $config-file; my @cliches; my $cliches-names; my %cliches-to-traits; -my %cliches-to-loggers; -my %cliches-to-loggers-pure; my $loggers-pure = %(); my $loggers = $(); @@ -135,8 +133,6 @@ sub clean-all-settings() { @cliches = []; $cliches-names = SetHash.new; %cliches-to-traits = %(); - %cliches-to-loggers = %(); - %cliches-to-loggers-pure = %(); $loggers-pure = %(); $loggers = %(); @@ -755,10 +751,6 @@ sub remove-logger(Str:D $trait --> Logger) is export(:configure) { } multi sub update-loggers(Positional:D $cliches) { - for |$cliches -> $cliche { - %cliches-to-loggers{$cliche.name}:delete; - %cliches-to-loggers-pure{$cliche.name}:delete; - } for |$cliches -> $cliche { my %copy = %cliches-to-traits{$cliche.name} // SetHash.new; %cliches-to-traits{$cliche.name} = SetHash.new; @@ -772,8 +764,6 @@ multi sub update-loggers(Positional:D $cliches) { multi sub update-loggers() { %cliches-to-traits = %(); - %cliches-to-loggers = %(); - %cliches-to-loggers-pure = %(); logger-map-clean(); my @traits := logger-pure-map-clean(); for @traits -> $trait { @@ -784,20 +774,12 @@ multi sub update-loggers() { sub find-or-create-and-store-logger-pure($trait) { return $_ with atomic-fetch($loggers-pure){$trait}; my $cliche = find-cliche-for-trait($trait); - with %cliches-to-loggers-pure{$cliche.name} { - logger-pure-map-put($trait, $_); - return $_; - } create-and-store-loggers($trait, $cliche, :pure); } sub find-or-create-and-store-logger($trait) { return $_ with atomic-fetch($loggers){$trait}; my $cliche = find-cliche-for-trait($trait); - with %cliches-to-loggers{$cliche.name} { - logger-map-put($trait, $_); - return $_; - } create-and-store-loggers($trait, $cliche, :!pure); } @@ -807,8 +789,6 @@ sub create-and-store-loggers($trait, $cliche, :$pure) { logger-map-put($trait, $logger); logger-pure-map-put($trait, $logger-pure); (%cliches-to-traits{$cliche.name} //= SetHash.new){$trait} = True; - %cliches-to-loggers{$cliche.name} = $logger; - %cliches-to-loggers-pure{$cliche.name} = $logger-pure; return $pure ?? $logger-pure !! $logger; } diff --git a/t/00-factories.t b/t/00-factories.t index 7fcb763..b2f7c70 100644 --- a/t/00-factories.t +++ b/t/00-factories.t @@ -5,7 +5,7 @@ use LogP6 :configure; use lib './t/resource/Helpers'; use IOString; -plan 6; +plan 5; $*ERR = IOString.new; @@ -320,7 +320,7 @@ subtest { my $log = get-logger('log'); ok $log, 'log cliche'; isnt $log, $any-log, 'default and log are not the same logs'; - is get-logger('any2'), $any-log, 'same cliche - same logger'; + isnt get-logger('any2'), $any-log, 'different trait - same logger'; is get-logger('any'), $any-log, 'same trait - same logger'; is get-logger('log'), $log, 'same trait - same logger again'; @@ -374,33 +374,4 @@ subtest { }, 'logger'; -subtest { - plan 7; - - cliche(:name, :matcher, grooves => - (writer(:name), filter(:name))); - cliche(:name, :matcher(/cliche2/), grooves => - (writer(:name), filter(:name))); - - my $first-log1 = get-logger('cliche1'); - my $first-log2 = get-logger('cliche2'); - isnt $first-log1, $first-log2, 'loggers are different for different cliches'; - - filter(:name, :update); - my $second-log1 = get-logger('cliche1'); - my $second-log2 = get-logger('cliche2'); - isnt $second-log1, $first-log1, 'new logger after update cliche\'s filter'; - is $second-log2, $first-log2, 'same logger after update another cliche'; - isnt $second-log1, $second-log2, 'different logger after update filter'; - - cliche(:name, :matcher(/cliche/), - grooves => ('writer2', 'filter2'), :replace); - my $third-log1 = get-logger('cliche1'); - my $third-log2 = get-logger('cliche2'); - isnt $third-log1, $second-log1, 'new logger after update another cliche'; - isnt $third-log2, $second-log2, 'new logger after update cliche'; - is $third-log1, $third-log2, 'same loggers for updated cliche'; - -}, 'create same logger'; - done-testing; From a521a85294166ab59abbfb3e99d6df11cb4af708 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sat, 20 Apr 2019 16:54:32 +0700 Subject: [PATCH 20/26] Update README for %trait --- README.md | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 9765692..38a528a 100644 --- a/README.md +++ b/README.md @@ -375,9 +375,11 @@ trait representation. `sprintf` option is useful for traits like `database`, example, `[%trait{sprintf=%s7}]` can be converted into `[ audit]`. `short` option is useful for traits like `Module::Packge1::Package2::Class`. You can specify package delimiter (instead of `::`) and how many packages will be -displayed. For example, `%trait{short=[.]2` can be converted into -`Package2.Class`, and `%trait{short=[.]2.4` - into `Modu.Pack.Package2.Class`. -If `number` is integer then only `number` right elements will be displayed. If +displayed. For example, `%trait{short=[.]1` can be converted into +`Class`, `%trait{short=[.]-1` - into `Packge1.Package2.Class` and +`%trait{short=[.]2.4` - into `Modu.Pack.Package2.Class`. If `number` is a +positive integer then only `number` right elements will be displayed. If +`number` is a negative integer then `|number|` left elements will be deleted. If `number` is real then left elements will be cut to fractional symbols; - `%tid` - for current `Thread` id; - `%tname` - for current `Thread` name; From 2171ad3afaae25f03c863e0f9dc780a7dab72691 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sat, 20 Apr 2019 21:23:16 +0700 Subject: [PATCH 21/26] Add out-buffer parameter to std handle in configuration file --- README.md | 4 ++-- lib/LogP6/ConfigFile.pm6 | 5 ++++- t/00-config-file.t | 12 +++++++++--- t/resource/00-config-file/log-p6-1.json | 11 +++++++++++ 4 files changed, 26 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 38a528a..392a114 100644 --- a/README.md +++ b/README.md @@ -473,8 +473,8 @@ array. Only `std` (for standard configuration) and `custom` types are supported. In case of standard configuration all field are optional excepts `name`. Handle can be: -- `file` type for output into file. You can specify `path` and `append` (`True` -by default) arguments; +- `file` type for output into file. You can specify `path`,`append` (`True` +by default) and `out-buffer` arguments; - `std` type for output into `$*OUT` or `$*ERR`. You can specify `path` as `out` or `err`. - `custom` type. diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index 3809064..e6be1c7 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -161,10 +161,13 @@ class LogP6::ConfigFile { } when 'file' { my $path = json; + my $out-buffer = json; die 'Missing file handle path' without $path; return self!produce(:use-cache, json, sub { my $mode = (json // True) ?? :a !! :w; - return $path.IO.open(|$mode); + my $handle = $path.IO.open(|$mode); + $handle.out-buffer = $out-buffer with $out-buffer; + return $handle; }); } when 'custom' { diff --git a/t/00-config-file.t b/t/00-config-file.t index f198d42..188a826 100644 --- a/t/00-config-file.t +++ b/t/00-config-file.t @@ -29,17 +29,20 @@ subtest { }, 'miss file'; subtest { - plan 20; + plan 22; + + CATCH { default {say .gist }} my ($w, $cn); $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); - is $cn.writers.elems, 5, 'parsed 5 writers'; + is $cn.writers.elems, 6, 'parsed 5 writers'; $w = $cn.writers[0]; is $w.name, 'w1', 'w1 name'; is $w.pattern, '%msg', 'w1 pattern'; is $w.handle.Str, './t/resource/00-config-file/handle1.after', 'w1 handle'; + is $w.handle.out-buffer, 0, 'w1 out-buffer false(0)'; ok $w.auto-exceptions, 'w1 auto-exceptions'; $w = $cn.writers[1]; @@ -56,11 +59,14 @@ subtest { nok $w.auto-exceptions, 'w4 auto-exceptions'; $w = $cn.writers[4]; - is $w.name, 'w5', 'w4 name'; + is $w.name, 'w5', 'w5 name'; nok $w.pattern, 'w5 pattern'; nok $w.handle, 'w5 handle'; nok $w.auto-exceptions, 'w5 auto-exceptions'; + $w = $cn.writers[5]; + is $w.handle.out-buffer, 1000, 'w6 handle out-buffer 1000'; + my $w0h = $cn.writers[0].handle.WHICH; my $w3h = $cn.writers[3].handle.WHICH; $cn = $config.parse-config('./t/resource/00-config-file/log-p6-1.json'); diff --git a/t/resource/00-config-file/log-p6-1.json b/t/resource/00-config-file/log-p6-1.json index 8f435dc..f4608a4 100644 --- a/t/resource/00-config-file/log-p6-1.json +++ b/t/resource/00-config-file/log-p6-1.json @@ -21,6 +21,7 @@ "handle": { "type": "file", "path": "./t/resource/00-config-file/handle1.after", + "out-buffer": false, "append": false }, "auto-exceptions": true @@ -62,6 +63,16 @@ "args": { "name": "w5" } + }, + { + "type": "std", + "name": "w6", + "handle": { + "type": "file", + "path": "./t/resource/00-config-file/handle1.after", + "out-buffer": 1000 + }, + "auto-exceptions": true } ], "filters": [ From a3ff50ec85464ecc3572d3dfb3fe200658e8742c Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Sat, 20 Apr 2019 22:16:20 +0700 Subject: [PATCH 22/26] Improve out-buffer parameter in std handle --- lib/LogP6/ConfigFile.pm6 | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index e6be1c7..4430f9b 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -165,8 +165,7 @@ class LogP6::ConfigFile { die 'Missing file handle path' without $path; return self!produce(:use-cache, json, sub { my $mode = (json // True) ?? :a !! :w; - my $handle = $path.IO.open(|$mode); - $handle.out-buffer = $out-buffer with $out-buffer; + my $handle = $path.IO.open(|$mode, :out-buffer($out-buffer // Nil)); return $handle; }); } From 8c0aa3ca081ed46659f70f8390e5ba8f8cbb10b0 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 24 Apr 2019 12:41:46 +0700 Subject: [PATCH 23/26] Fix repository url in META6 and README --- META6.json | 2 +- README.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/META6.json b/META6.json index 26516af..e4a106b 100644 --- a/META6.json +++ b/META6.json @@ -13,7 +13,7 @@ "LOGGER" ], "license": "Artistic-2.0", - "source-url": "https://github.com/atroxaper/p6-LogP6", + "source-url": "git://github.com/atroxaper/p6-LogP6.git", "provides": { "LogP6::Wrapper::SyncAbstract": "lib/LogP6/Wrapper/SyncAbstract.pm6", "LogP6::LoggerPure": "lib/LogP6/LoggerPure.pm6", diff --git a/README.md b/README.md index 392a114..9502232 100644 --- a/README.md +++ b/README.md @@ -1081,7 +1081,7 @@ or after file size limit are reached; Mikhail Khorkov -Source can be located at: https://github.com/atroxaper/p6-LogP6. Comments and Pull Requests +Source can be located at: [github](https://github.com/atroxaper/p6-LogP6). Comments and Pull Requests are welcome. # COPYRIGHT AND LICENSE From cfc7e959814f228bc18bff2cf0b7f9ed3742464e Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 24 Apr 2019 13:52:44 +0700 Subject: [PATCH 24/26] Add out-buffer parameter to std handle too in configuration file --- lib/LogP6/ConfigFile.pm6 | 7 +++++-- t/00-config-file.t | 3 ++- t/resource/00-config-file/log-p6-1.json | 3 ++- 3 files changed, 9 insertions(+), 4 deletions(-) diff --git a/lib/LogP6/ConfigFile.pm6 b/lib/LogP6/ConfigFile.pm6 index 4430f9b..c8e7960 100644 --- a/lib/LogP6/ConfigFile.pm6 +++ b/lib/LogP6/ConfigFile.pm6 @@ -147,17 +147,20 @@ class LogP6::ConfigFile { given json { when 'std' { my $path = json; + my $result; given $path { when 'out' { - return $*OUT; + $result = $*OUT; } when 'err' { - return $*ERR; + $result = $*ERR; } default { die "Wrong std handle path $path"; } } + $result.out-buffer = $_ with json; + return $result; } when 'file' { my $path = json; diff --git a/t/00-config-file.t b/t/00-config-file.t index 188a826..60b26f7 100644 --- a/t/00-config-file.t +++ b/t/00-config-file.t @@ -29,7 +29,7 @@ subtest { }, 'miss file'; subtest { - plan 22; + plan 23; CATCH { default {say .gist }} @@ -49,6 +49,7 @@ subtest { is $w.name, 'w2', 'w2 name'; is $w.pattern, '%level | %msg', 'w2 pattern'; is $w.handle, $*OUT, 'w2 handle'; + is $w.handle.out-buffer, 100, 'w2 out-buffer false(0)'; is $cn.writers[2].handle, $*ERR, 'w3 handle'; nok $w.auto-exceptions, 'w2 auto-exceptions'; diff --git a/t/resource/00-config-file/log-p6-1.json b/t/resource/00-config-file/log-p6-1.json index f4608a4..2993cb6 100644 --- a/t/resource/00-config-file/log-p6-1.json +++ b/t/resource/00-config-file/log-p6-1.json @@ -32,7 +32,8 @@ "pattern": "%level | %msg", "handle": { "type": "std", - "path": "out" + "path": "out", + "out-buffer": 100 }, "auto-exceptions": false }, From 6d5f63d02bf7b17384087362854deda54b97dfd8 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 24 Apr 2019 13:54:36 +0700 Subject: [PATCH 25/26] Add Changes info --- Changes | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) create mode 100644 Changes diff --git a/Changes b/Changes new file mode 100644 index 0000000..9f692c2 --- /dev/null +++ b/Changes @@ -0,0 +1,20 @@ +Version history: + +1.5.1 2019-04-24 'the new small features' + - add asynchronous implementation of handle and writer + - add parameters for %trait placeholder + - add %color placeholder + - add reusing handles from previous config file read + - add possibility to backup and restore NDC and MDC values + - add out-buffer parameter to std and file handle in configuration file + - improve internal exceptions behaviour + +1.4.2 2019-04-10 'better config file and zero length' + - add length=0 support in %level placeholder + - add support of using 'custom' arguments in custom's args in configuration file + +1.4.1 2019-04-08 'callframe in writer pattern' + - add callframe support in writer pattern + +1.3.1 2019-04-07 'The first public release' + - Production ready log library \ No newline at end of file From 30b9488e2dee9311355cc7c7540e05f1277c9038 Mon Sep 17 00:00:00 2001 From: Mikhail Khorkov Date: Wed, 24 Apr 2019 15:11:50 +0700 Subject: [PATCH 26/26] Bump version --- META6.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/META6.json b/META6.json index e4a106b..f85eab8 100644 --- a/META6.json +++ b/META6.json @@ -6,7 +6,7 @@ ], "description": "Full customisable logging library inspired by idea of separate logging and its configuration.", "resources": [], - "version": "1.4.2", + "version": "1.5.1", "tags": [ "LOG", "LOGGING",