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: |