File Coverage

blib/lib/Liveman.pm
Criterion Covered Total %
statement 167 226 73.8
branch 45 110 40.9
condition 12 59 20.3
subroutine 20 29 68.9
pod 8 8 100.0
total 252 432 58.3


line stmt bran cond sub pod time code
1             package Liveman;
2:
use 5.22.0;
3: use common::sense;
4:
5: our $VERSION = "0.07";
6:
7: use Term::ANSIColor qw/colored/;
8: use File::Slurper qw/read_text write_text/;
9: use Markdown::To::POD qw/markdown_to_pod/;
10:
11:
12: # Конструктор 13: sub new {
14: my $cls = shift;
15: my $self = bless {@_}, $cls;
16: delete $self->{files} if $self->{files} && !scalar @{$self->{files}};
17: $self
18: }
19:
20: # Пакет из пути
21: sub _pkg($) {
22: ( shift =~ s!^lib/(.*)\.\w+$!$1!r ) =~ s!/!::!gr
23: }
24:
25: # Переменная из пакета
26: sub _var($) {
27: '$' . lcfirst( shift =~ s!::(\w)?!_${\lc $1}!gr )
28: }
29:
30: # Для метода для вставки
31: sub _md_method(@) {
32: my ($pkg, $sub, $args, $remark) = @_;
33: my $sub_args = "$sub ($args)";
34: $args = "($args)" if $args;
35:
36: $remark = "." unless defined $remark;
37: my $var = _var $pkg;
38: << "END";
39: ## $sub_args
40:
41: $remark
42:
43: ```perl
44: my $var = $pkg->new;
45: ${var}->$sub$args # -> .3
46: ```
47:
48: END
49: }
50:
51: # Для фичи для вставки
52: sub _md_feature(@) {
53: my ($pkg, $has, $remark) = @_;
54:
55: $remark = "." unless defined $remark;
56: my $var = _var $pkg;
57: << "END";
58: ## $has
59:
60: $remark
61:
62: ```perl
63: my $var = $pkg->new;
64:
65: ${var}->$has\t# -> .5
66: ```
67:
68: END
69: }
70:
71:
72: # Добавить разделы функций в *.md из *.pm
73: sub appends {
74: my ($self) = @_;
75: my $files = $self->{files} // [split /\n/, `find lib -name '*.pm' -a -type f`];
76: $self->append($_) for @$files;
77: $self
78: }
79:
80: # Добавить разделы функций в *.md из *.pm
81: sub append {
82: my ($self, $pm) = @_;
83:
84: my $md = $pm =~ s!(\.\w+)?$!.md!r;
85:
86: die "Not file $pm!" if !-f $pm;
87: $self->mkmd($md) if !-f $md;
88:
89: local $_ = read_text $pm;
90: my %sub; my %has;
91: while(m! (^\# [\ \t]* (?<remark> .*?) [\ \t]* )? \n (
92: sub \s+ (?<sub> (\w+|::)+ ) .*
93: ( \s* my \s* \( \s* (\$self,? \s* )? (?<args>.*?) \s* \) \s* = \s* \@_; )?
94: | has \s+ (?<has> (\w+|'\w+'|"\w+"|\[ \s* ([^\[\]]*?) \s* \])+ )
95: ) !mgxn) {
96: $sub{$+{sub}} = {%+} if exists $+{sub} and "_" ne substr $+{sub}, 0, 1;
97: $has{$+{has}} = {%+} if exists $+{has} and "_" ne substr $+{has}, 0, 1;
98: }
99:
100: return $self if !keys %sub && !keys %has;
101:
102: $_ = read_text $md;
103:
104: my $pkg = _pkg $md;
105:
106: my $added = 0;
107:
108: s{^\#[\ \t]+( (?<is>METHODS|SUBROUTINES) | DESCRIPTION )
109: (^```.*?^```|.)*? (?= ^\#\s)
110: }{
111: my $x = $&; my $is = $+{is};
112: if($is) {
113: while($x =~ /^\#\#[\ \t]+(\w+)/gm) {
114: delete $sub{$1};
115: }
116: }
117: $added += keys %sub;
118: join "", $x, $is? (): "# SUBROUTINES/METHODS\n\n", map { _md_method $pkg, $_, $sub{$_}{args}, $sub{$_}{remark} } sort keys %sub;
119: }emsx or die "Нет секции DESCRIPTION!" if keys %sub;
120:
121: s{^\#[\ \t]+((?<is>FEATURES) | DESCRIPTION)
122: (^```.*?^```|.)*? (?= ^\#\s)
123: }{
124: my $x = $&; my $is = $+{is};
125: if($is) {
126: while($x =~ /^\#\#[\ \t]+([^\n]+?)[\ \t]*/gm) {
127: delete $has{$1};
128: }
129: }
130: $added += keys %has;
131: join "", $x, $is? (): "# FEATURES\n\n", map { _md_feature $pkg, $_, $sub{$_}{remark} } sort keys %has;
132: }emsx or die "Нет секции DESCRIPTION!" if keys %has;
133:
134:
135: if ($added) {
136: write_text $md, $_;
137: print "🔖 $pm ", colored("⊂", "BRIGHT_GREEN"), " $md ", "\n",
138: " ", scalar keys %has? (colored("FEATURES ", "BRIGHT_WHITE"), join(colored(", ", "red"), sort keys %has), "\n"): (),
139: " ", scalar keys %sub? (colored("SUBROUTINES ", "BRIGHT_WHITE"), join(colored(", ", "red"), sort keys %sub), "\n"): (),
140: ;
141: } else {
142: print "🔖 $pm\n";
143: }
144:
145: $self->{count}++;
146: $self->{added} = $added;
147: $self
148: }
149:
150: sub _git_user_name { shift->{_git_user_name} //= _trim(`git config user.name`) }
151: sub _git_user_email { shift->{_git_user_email} //= _trim(`git config user.email`) }
152: sub _year { shift->{_year} //= _trim(`date +%Y`) }
153: sub _license { shift->{_license} //= -r "minil.toml" && read_text("minil.toml") =~ /^\s*license\s*=\s*"([^"\n]*)"/m ? ($1 eq "perl_5"? "Perl5": uc($1) =~ s/_/v/r): "Perl5" }
154: sub _land { shift->{_land} //= `curl "https://ipapi.co/\$(curl https://2ip.ru --connect-timeout 3 --max-time 3 -Ss)/json/" --connect-timeout 3 --max-time 3 -Ss` =~ /country_name": "([^"\n]*)"/ ? ($1 eq "Russia" ? "Rusland" : $1) : 'Rusland' }
155:
156: # Добавить разделы функций в *.md из *.pm
157: sub mkmd {
158: my ($self, $md) = @_;
159:
160: my $pkg = _pkg $md;
161:
162: my $author = $self->_git_user_name;
163: my $email = $self->_git_user_email;
164: my $year = $self->_year;
165: my $license = $self->_license;
166: my $land = $self->_land;
167:
168: write_text $md, << "END";
169: # NAME
170:
171: $pkg -
172:
173: # SYNOPSIS
174:
175: ```perl
176: use $pkg;
177:
178: my ${\_var $pkg} = $pkg->new;
179: ```
180:
181: # DESCRIPION
182:
183: .
184:
185: # SUBROUTINES
186:
187: # INSTALL
188:
189: For install this module in your system run next [command](https://metacpan.org/pod/App::cpm):
190:
191: ```sh
192: sudo cpm install -gvv $pkg
193: ```
194:
195: # AUTHOR
196:
197: $author [$email](mailto:$email)
198:
199: # LICENSE
200:
201: ⚖ **$license**
202:
203: # COPYRIGHT
204:
205: The $pkg module is copyright © $year $author. $land. All rights reserved.
206: END
207: }
208:
209: # Получить путь к тестовому файлу из пути к md-файлу
210: sub test_path {
211: my ($self, $md) = @_;
212: $md =~ s!^lib/(.*)\.md$!
213: join "", "t/", join("/", map {
214: lcfirst($_) =~ s/[A-Z]/"-" . lc $&/gre
215: } split /\//, $1), ".t"
216: !e;
217: $md
218: }
219:
220: # Трансформирует md-файлы
221: sub transforms {
222: my ($self) = @_;
223: my $mds = $self->{files} // [split /\n/, `find lib -name '*.md'`];
224:
225: $self->{count} = 0;
226:
227: if($self->{compile_force}) {
228: $self->transform($_) for @$mds;
229: } else {
230: for my $md (@$mds) {
231: my $test = $self->test_path($md);
232: my $mdmtime = (stat $md)[9];
233: die "Нет файла $md" if !$mdmtime;
234: $self->transform($md, $test) if !-e $test || -e $test && $mdmtime > (stat $test)[9];
235: }
236: }
237:
238: if(-f "minil.toml" && -r "minil.toml") {
239: my $is_copy; my $name;
240: eval {
241: my $minil = read_text("minil.toml");
242: ($name) = $minil =~ /^name = "([\w:-]+)"/m;
243: $name =~ s!(-|::)!/!g;
244: $name = "lib/$name.md";
245: if(-f $name && -r $name) {
246: if(!-e "README.md" || -e "README.md"
247: && (stat $name)[9] > (stat "README.md")[9]) {
248: write_text "README.md", read_text $name;
249: $is_copy = 1;
250: }
251: }
252: };
253: if($@) {warn $@}
254: elsif($is_copy) {
255: print "📘 $name ", colored("↦", "white"), " README.md ", colored("...", "white"), " ", colored("ok", "bright_green"), "\n";
256: }
257: }
258:
259: $self
260: }
261:
262: # Эскейпинг для qr!!
263: sub _qr_esc {
264: $_[0] =~ s/!/\\!/gr
265: }
266:
267: # Эскейпинг для строки в двойных кавычках
268: sub _qq_esc {
269: $_[0] =~ s!"!\\"!gr
270: }
271:
272: # Эскейпинг для строки в одинарных кавычках
273: sub _q_esc {
274: $_[0] =~ s!'!\\'!gr
275: }
276:
277: # Обрезает пробельные символы
278: sub _trim {
279: $_[0] =~ s!^\s*(.*?)\s*\z!$1!sr
280: }
281:
282: # Создаёт путь
283: sub _mkpath {
284: my ($p) = @_;
285: mkdir $`, 0755 while $p =~ /\//g;
286: }
287:
288: # Строка кода для тестирования
289: sub _to_testing {
290: my ($line, %x) = @_;
291:
292: return $x{code} if $x{code} =~ /^\s*#/;
293:
294: my $expected = $x{expected};
295: my $q = _q_esc($line =~ s!\s*$!!r);
296: my $code = _trim($x{code});
297:
298: if(exists $x{is_deeply}) { "::is_deeply scalar do {$code}, scalar do {$expected}, '$q';\n" }
299: elsif(exists $x{is}) { "::is scalar do {$code}, scalar do{$expected}, '$q';\n" }
300: elsif(exists $x{qqis}) { my $ex = _qq_esc($expected); "::is scalar do {$code}, \"$ex\", '$q';\n" }
301: elsif(exists $x{qis}) { my $ex = _q_esc($expected); "::is scalar do {$code}, '$ex', '$q';\n" }
302: elsif(exists $x{like}) { my $ex = _qr_esc($expected); "::like scalar do {$code}, qr!$ex!, '$q';\n" }
303: elsif(exists $x{unlike}) { my $ex = _qr_esc($expected); "::unlike scalar do {$code}, qr!$ex!, '$q';\n" }
304: else { # Что-то ужасное вырвалось на волю!
305: "???"
306: }
307: }
308:
309: # Трансформирует md-файл в тест и документацию
310: sub transform {
311: my ($self, $md, $test) = @_;
312: $test //= $self->test_path($md);
313:
314: print "🔖 $md ", colored("↦", "white"), " $test ", colored("...", "white"), " ";
315:
316: my $markdown = read_text($md);
317:
318: my @pod; my @test; my $title = 'Start'; my $close_subtest; my $use_title = 1;
319:
320: my @text = split /^(```\w*[ \t]*(?:\n|\z))/mo, $markdown;
321:
322: for(my $i=0; $i<@text; $i+=4) {
323: my ($mark, $sec1, $code, $sec2) = @text[$i..$i+4];
324:
325: push @pod, markdown_to_pod($mark);
326: push @test, $mark =~ s/^/# /rmg;
327:
328: last unless defined $sec1;
329: $i--, $sec2 = $code, $code = "" if $code =~ /^```[ \t]*$/;
330:
331: die "=== mark ===\n$mark\n=== sec1 ===\n$sec1\n=== code ===\n$code\n=== sec2 ===\n$sec2\n\nsec2 ne ```" if $sec2 ne "```\n";
332:
333: $title = _trim($1) while $mark =~ /^#+[ \t]+(.*)/gm;
334:
335: push @pod, "\n", ($code =~ s/^/\t/gmr), "\n";
336:
337: my ($infile, $is) = $mark =~ /^(?:File|Файл)[ \t]+(.*?)([\t ]+(?:is|является))?:[\t ]*\n\z/m;
338: if($infile) {
339: my $real_code = $code =~ s/^\\(```\w*[\t ]*$)/$1/mgro;
340: if($is) { # тестируем, что текст совпадает
341: push @test, "\n{ my \$s = '${\_q_esc($infile)}'; open my \$__f__, '<:utf8', \$s or die \"Read \$s: \$!\"; my \$n = join '', <\$__f__>; close \$__f__; ::is \$n, '${\_q_esc($real_code)}', \"File \$s\"; }\n";
342: }
343: else { # записываем тект в файл
344: #push @test, "\n{ my \$s = main::_mkpath_('${\_q_esc($infile)}'); open my \$__f__, '>:utf8', \$s or die \"Read \$s: \$!\"; print \$__f__ '${\_q_esc($real_code)}'; close \$__f__ }\n";
345: push @test, "#\@> $infile\n", $real_code =~ s/^/#>> /rgm, "#\@< EOF\n";
346: }
347: } elsif($sec1 =~ /^```(?:perl)?[ \t]*$/) {
348:
349: if($use_title ne $title) {
350: push @test, "done_testing; }; " if $close_subtest;
351: $close_subtest = 1;
352: push @test, "subtest '${\ _q_esc($title)}' => sub { ";
353: $use_title = $title;
354: }
355:
356: my $test = $code =~ s{^(?<code>.*)#[ \t]*((?<is_deeply>-->|⟶)|(?<is>->|→)|(?<qqis>=>|⇒)|(?<qis>\\>|↦)|(?<like>~>|↬)|(?<unlike><~|↫))\s*(?<expected>.+?)[ \t]*\n}{ _to_testing($&, %+) }grme;
357: push @test, "\n", $test, "\n";
358: }
359: else {
360: push @test, "\n", $code =~ s/^/# /rmg, "\n";
361: }
362: }
363:
364: push @test, "\n\tdone_testing;\n};\n" if $close_subtest;
365: push @test, "\ndone_testing;\n";
366:
367: _mkpath($test);
368: my $mkpath = q{sub _mkpath_ { my ($p) = @_; length($`) && !-e $`? mkdir($`, 0755) || die "mkdir $`: $!": () while $p =~ m!/!g; $p }};
369: my $write_files = q{open my $__f__, "<:utf8", $t or die "Read $t: $!"; read $__f__, $s, -s $__f__; close $__f__; while($s =~ /^#\\@> (.*)\n((#>> .*\n)*)#\\@< EOF\n/gm) { my ($file, $code) = ($1, $2); $code =~ s/^#>> //mg; open my $__f__, ">:utf8", _mkpath_($file) or die "Write $file: $!"; print $__f__ $code; close $__f__; }};
370: #my @symbol = ('a'..'z', 'A'..'Z', '0' .. '9', '-', '_');
371: # "-" . join("", map $symbol[rand(scalar @symbol)], 1..6)
372: my $test_path = join "", "/tmp/.liveman/",
373: `pwd` =~ s/^.*?([^\/]+)\n$/$1/rs,
374: $test =~ s!^t/(.*)\.t$!/$1!r =~ y/\//!/r, "/";
375: my $chdir = "my \$t = `pwd`; chop \$t; \$t .= '/' . __FILE__; my \$s = '${\ _q_esc($test_path)}'; `rm -fr '\$s'` if -e \$s; chdir _mkpath_(\$s) or die \"chdir \$s: \$!\";";
376: # use Carp::Always::Color ::Term;
377: my $die = 'use Scalar::Util qw//; use Carp qw//; $SIG{__DIE__} = sub { my ($s) = @_; if(ref $s) { $s->{STACKTRACE} = Carp::longmess "?" if "HASH" eq Scalar::Util::reftype $s; die $s } else {die Carp::longmess defined($s)? $s: "undef" }};';
378: write_text $test, join "", "use common::sense; use open qw/:std :utf8/; use Test::More 0.98; $mkpath BEGIN { $die $chdir $write_files } ", @test;
379:
380: # Создаём модуль, если его нет
381: my $pm = $md =~ s/\.md$/.pm/r;
382: if(!-e $pm) {
383: my $pkg = ($pm =~ s!^lib/(.*)\.pm$!$1!r) =~ s!/!::!gr;
384: write_text $pm, "package $pkg;\n\n1;";
385: }
386:
387: # Трансформируем модуль (pod и версия):
388: my $pod = join "", @pod;
389: my $module = read_text $pm;
390: $module =~ s!(\s*\n__END__[\t ]*\n.*)?$!\n\n__END__\n\n=encoding utf-8\n\n$pod!sn;
391:
392: # Меняем версию:
393: my ($version) = $markdown =~ /#[ \t]+VERSION\s+([\d\.]+)\s/;
394: $module =~ s!^(our \$VERSION = ")[^"]*(";)!$0.07$version$2!m if defined $version;
395: write_text $pm, $module;
396:
397: $self->{count}++;
398:
399: print colored("ok", "bright_green"), "\n";
400:
401: $self
402: }
403:
404: # Запустить тесты
405: sub tests {
406: my ($self) = @_;
407:
408: my $cover = "/usr/bin/site_perl/cover";
409: $cover = 'cover' if !-e $cover;
410:
411: my $yath = "/usr/bin/site_perl/yath";
412: $yath = 'yath' if !-e $yath;
413:
414: my $options = $self->{options};
415:
416: if($self->{files}) {
417: my @tests = map $self->test_path($_), @{$self->{files}};
418: local $, = " ";
419: $self->{exit_code} = system $self->{prove}
420: ? "prove -Ilib $options @tests"
421: : "$yath test -j4 $options @tests";
422: return $self;
423: }
424:
425: my $perl5opt = $ENV{PERL5OPT};
426:
427: system "$cover -delete";
428: if($self->{prove}) {
429: local $ENV{PERL5OPT} = "$perl5opt -MDevel::Cover";
430: $self->{exit_code} = system "env | grep PERL5OPT; prove -Ilib -r t $options";
431: #$self->{exit_code} = system "prove --exec 'echo `pwd`/lib && perl -MDevel::Cover -I`pwd`/lib' -r t";
432: } else {
433: $self->{exit_code} = system "$yath test -j4 --cover $options";
434: }
435: return $self if $self->{exit_code};
436: system "$cover -report html_basic";
437: system "(opera cover_db/coverage.html || xdg-open cover_db/coverage.html) &> /dev/null" if $self->{open};
438: return $self;
439: }
440:
441: 1;
442:
443: __END__
444:
445: =encoding utf-8
446:
447: =head1 NAME
448:
449: Liveman - markdown compiller to test and pod.
450:
451: =head1 VERSION
452:
453: 0.05
454:
455: =head1 SYNOPSIS
456:
457: File lib/Example.md:
458:
459: Twice two:
460: \```perl
461: 2*2 # -> 2+2
462: \```
463:
464: Test:
465:
466: use Liveman;
467:
468: my $liveman = Liveman->new(prove => 1);
469:
470: # compile lib/Example.md file to t/example.t and added pod to lib/Example.pm
471: $liveman->transform("lib/Example.md");
472:
473: $liveman->{count} # => 1
474: -f "t/example.t" # => 1
475: -f "lib/Example.pm" # => 1
476:
477: # compile all lib/**.md files with a modification time longer than their corresponding test files (t/**.t)
478: $liveman->transforms;
479: $liveman->{count} # => 0
480:
481: # compile without check modification time
482: Liveman->new(compile_force => 1)->transforms->{count} # => 1
483:
484: # start tests with yath
485: my $yath_return_code = $liveman->tests->{exit_code};
486:
487: $yath_return_code # => 0
488: -f "cover_db/coverage.html" # => 1
489:
490: # limit liveman to these files for operations transforms and tests (without cover)
491: my $liveman2 = Liveman->new(files => [], force_compile => 1);
492:
493: =head1 DESCRIPION
494:
495: The problem with modern projects is that the documentation is disconnected from testing.
496: This means that the examples in the documentation may not work, and the documentation itself may lag behind the code.
497:
498: Liveman compile C<lib/**>.md files to C<t/**.t> files
499: and it added pod-documentation to section C<__END__> to C<lib/**.pm> files.
500:
501: Use C<liveman> command for compile the documentation to the tests in catalog of your project and starts the tests:
502:
503: liveman
504:
505: Run it with coverage.
506:
507: Option C<-o> open coverage in browser (coverage file: C<cover_db/coverage.html>).
508:
509: Liveman replace C<our $VERSION = "...";> in C<lib/**.pm> from C<lib/**.md> if it exists in pm and in md.
510:
511: If exists file B<minil.toml>, then Liveman read C<name> from it, and copy file with this name and extension C<.md> to README.md.
512:
513: =head2 TYPES OF TESTS
514:
515: Section codes C<noname> or C<perl> writes as code to C<t/**.t>-file. And comment with arrow translates on test from module C<Test::More>.
516:
517: The test name set as the code-line.
518:
519: =head3 C<is>
520:
521: Compare two expressions for equivalence:
522:
523: "hi!" # -> "hi" . "!"
524: "hi!" # → "hi" . "!"
525:
526: =head3 C<is_deeply>
527:
528: Compare two expressions for structures:
529:
530: "hi!" # --> "hi" . "!"
531: "hi!" # ⟶ "hi" . "!"
532:
533: =head3 C<is> with extrapolate-string
534:
535: Compare expression with extrapolate-string:
536:
537: my $exclamation = "!";
538: "hi!2" # => hi${exclamation}2
539: "hi!2" # ⇒ hi${exclamation}2
540:
541: =head3 C<is> with nonextrapolate-string
542:
543: Compare expression with nonextrapolate-string:
544:
545: 'hi${exclamation}3' # \> hi${exclamation}3
546: 'hi${exclamation}3' # ↦ hi${exclamation}3
547:
548: =head3 C<like>
549:
550: It check a regular expression included in the expression:
551:
552: 'abbc' # ~> b+
553: 'abc' # ↬ b+
554:
555: =head3 C<unlike>
556:
557: It check a regular expression excluded in the expression:
558:
559: 'ac' # <~ b+
560: 'ac' # ↫ b+
561:
562: =head2 EMBEDDING FILES
563:
564: Each test is executed in a temporary directory, which is erased and created when the test is run.
565:
566: This directory format is /tmp/.liveman/I<project>/I<path-to-test>/.
567:
568: Code section in md-file prefixed line B<< File C<path>: >> write to file in rintime testing.
569:
570: Code section in md-file prefixed line B<< File C<path> is: >> will be compared with the file by the method C<Test::More::is>.
571:
572: File experiment/test.txt:
573:
574: hi!
575:
576: File experiment/test.txt is:
577:
578: hi!
579:
580: B<Attention!> An empty string between the prefix and the code is not allowed!
581:
582: Prefixes maybe on russan: C<Файл path:> and C<Файл path является:>.
583:
584: =head1 METHODS
585:
586: =head2 new (files=>[...], open => 1, force_compile => 1)
587:
588: Constructor. Has arguments:
589:
590: =over
591:
592: =item 1. C<files> (array_ref) — list of md-files for methods C<transforms> and C<tests>.
593:
594: =item 2. C<open> (boolean) — open coverage in browser. If is B<opera> browser — open in it. Else — open via C<xdg-open>.
595:
596: =item 3. C<force_compile> (boolean) — do not check the md-files modification time.
597:
598: =back
599:
600: =head2 test_path ($md_path)
601:
602: Get the path to the C<t/**.t>-file from the path to the C<lib/**.md>-file:
603:
604: Liveman->new->test_path("lib/PathFix/RestFix.md") # => t/path-fix/rest-fix.t
605:
606: =head2 transform ($md_path, [$test_path])
607:
608: Compile C<lib/**.md>-file to C<t/**.t>-file.
609:
610: And method C<transform> replace the B<pod>-documentation in section C<__END__> in C<lib/**.pm>-file. And create C<lib/**.pm>-file if it not exists.
611:
612: File lib/Example.pm is:
613:
614: package Example;
615:
616: 1;
617:
618: __END__
619:
620: =encoding utf-8
621:
622: Twice two:
623:
624: 2*2 # -> 2+2
625:
626:
627: =head2 transforms ()
628:
629: Compile C<lib/**.md>-files to C<t/**.t>-files.
630:
631: All if C<< $self-E<gt>{files} >> is empty, or C<< $self-E<gt>{files} >>.
632:
633: =head2 tests ()
634:
635: Tests C<t/**.t>-files.
636:
637: All if C<< $self-E<gt>{files} >> is empty, or C<< $self-E<gt>{files} >> only.
638:
639: =head2 mkmd ($md)
640:
641: It make md-file.
642:
643: =head2 appends ()
644:
645: Append to C<lib/**.md> from C<lib/**.pm> subroutines and features.
646:
647: =head2 append ($path)
648:
649: Append subroutines and features from the module with C<$path> into its documentation in the its sections.
650:
651: File lib/Alt/The/Plan.pm:
652:
653: package Alt::The::Plan;
654:
655: sub planner {
656: my ($self) = @_;
657: }
658:
659: # This is first!
660: sub miting {
661: my ($self, $meet, $man, $woman) = @_;
662: }
663:
664: sub _exquise_me {
665: my ($self, $meet, $man, $woman) = @_;
666: }
667:
668: 1;
669:
670:
671:
672: -e "lib/Alt/The/Plan.md" # -> undef
673:
674: # Set the mocks:
675: *Liveman::_git_user_name = sub {'Yaroslav O. Kosmina'};
676: *Liveman::_git_user_email = sub {'dart@cpan.org'};
677: *Liveman::_year = sub {2023};
678: *Liveman::_license = sub {"Perl5"};
679: *Liveman::_land = sub {"Rusland"};
680:
681: my $liveman = Liveman->new->append("lib/Alt/The/Plan.pm");
682: $liveman->{count} # -> 1
683: $liveman->{added} # -> 2
684:
685: -e "lib/Alt/The/Plan.md" # -> 1
686:
687: # And again:
688: $liveman = Liveman->new->append("lib/Alt/The/Plan.pm");
689: $liveman->{count} # -> 1
690: $liveman->{added} # -> 0
691:
692: File lib/Alt/The/Plan.md is:
693:
694: # NAME
695:
696: Alt::The::Plan -
697:
698: # SYNOPSIS
699:
700: \```perl
701: use Alt::The::Plan;
702:
703: my $alt_the_plan = Alt::The::Plan->new;
704: \```
705:
706: # DESCRIPION
707:
708: .
709:
710: # SUBROUTINES
711:
712: ## miting ($meet, $man, $woman)
713:
714: This is first!
715:
716: \```perl
717: my $alt_the_plan = Alt::The::Plan->new;
718: $alt_the_plan->miting($meet, $man, $woman) # -> .3
719: \```
720:
721: ## planner ()
722:
723: .
724:
725: \```perl
726: my $alt_the_plan = Alt::The::Plan->new;
727: $alt_the_plan->planner # -> .3
728: \```
729:
730: # INSTALL
731:
732: For install this module in your system run next [command](https://metacpan.org/pod/App::cpm):
733:
734: \```sh
735: sudo cpm install -gvv Alt::The::Plan
736: \```
737:
738: # AUTHOR
739:
740: Yaroslav O. Kosmina [dart@cpan.org](mailto:dart@cpan.org)
741:
742: # LICENSE
743:
744: ⚖ **Perl5**
745:
746: # COPYRIGHT
747:
748: The Alt::The::Plan module is copyright © 2023 Yaroslav O. Kosmina. Rusland. All rights reserved.
749:
750: =head1 INSTALL
751:
752: Add to B<cpanfile> in your project:
753:
754: on 'test' => sub {
755: requires 'Liveman',
756: git => 'https://github.com/darviarush/perl-liveman.git',
757: ref => 'master',
758: ;
759: };
760:
761: And run command:
762:
763: $ sudo cpm install -gvv
764:
765: =head1 AUTHOR
766:
767: Yaroslav O. Kosmina LL<mailto:dart@cpan.org>
768:
769: =head1 LICENSE
770:
771: ⚖ B<GPLv3>
772:
773: =head1 COPYRIGHT
774:
775: The Alt::The::Plan module is copyright © 2023 Yaroslav O. Kosmina. Rusland. All rights reserved.
776: