NAME

Sub::Spec::Clause::features - Specify subroutine features

VERSION

version 0.11

SYNOPSIS

In your spec:

features => {
    FEATURE => VALUE,
    ...
}

DESCRIPTION

The 'features' clause allows subroutines to express their features. It is extensible so you can specify more features.

Below is the list of defined features. New feature clause may be defined by providing Sub::Spec::Clause::features::check_<CLAUSE>().

reverse => BOOL|HASHREF

Default is false. If set to true, specifies that subroutine supports reverse operation. To reverse, caller can add special argument '-reverse'. For example:

$SPEC{triple} = {args=>{num=>'num*'}, features=>{reverse=>1}};
sub triple {
    my %args = @_;
    my $num  = $args{num};
    [200, "OK", $args{-reverse} ? $num/3 : $num*3];
}

triple(num=>12);              # => 36
triple(num=>12, -reverse=>1); # =>  4

DRAFT: Conditional reversibility: if set to hashref, sub can declare partial reversibility. Caller can call with special argument '-reverse_action' set to 'test' and sub must return a bool result indicating if the particular action is reversible. Example:

$SPEC{multiply_by} = {args     => {num=>'num*', multiplier=>'num*'},
                      features => {reverse=>{...}}};
sub multiply_by {
    my %args = @_;
    my $n    = $args{num};
    my $m    = $args{multiplier};
    if ($args{-reverse_action} && $args{-reverse_action} eq 'test') {
        # we can only reverse if multiplier is not zero
        return [200, "OK", $m == 0 ? 1:0];
    }
    [200, "OK", $args{-reverse} ? $n/$m : $n*$m];
}

undo => BOOL|HASHREF

Default is false. If set to true, specifies that subroutine supports undo operation. Undo is similar to 'reverse' but needs some state to be saved and restored for do/undo operation, while reverse can work solely from the arguments.

Caller must provide one or more special arguments: -undo_action, -undo_info, -redo_info when dealing with do/undo stuffs.

do

To perform normal operation, -undo_action must be set to 'do'. Sub must save undo information, perform action, and return result along with saved undo info in the undo_info metadata, example:

return [200, "OK", $result, {undo_info=>$undo_info}];

Undo info should contain information (or reference to information) to restore to previous state later. This information should be persistent (e.g. in a file/database) when necessary. For example, if undo information is saved in a file, undo_info can contain the filename. If undo information is saved in a memory structure, undo_info can refer to this memory structure, and so on. Undo info should be serializable. Caller should store this undo info in the undo stack (note: undo stack management is the caller's task).

If -undo_action is false/undef, sub must assume caller want to perform action but without saving undo information.

undo

To perform an undo, caller must set -undo_action to 'undo' and pass back the undo info in -undo_info. Sub must restore previous state using undo info (or return 412 if undo info is invalid/unusable). After a successful undo, sub must return 200. Caller should mark that the undo info has already been used (undone). A redo information can be provided by the sub if necessary:

return [200, "OK", undef, {redo_info=>...}];

redo

To redo a previously undone action, caller must set -undo_action to 'redo' and -redo_info to redo information given by the sub when doing undo previously (if any). Sub should return 412 if redo info is invalid/unusable. Sub must redo the action and can optionally return a new undo_info.

return [200, "OK", $result, {undo_info=>...}];

After a successful redo, action can be undone again using the previous (or new) undo_info.

Example (in the example, undo info is only stored in memory):

use Cwd qw(abs_path);
use File::Slurp;

$SPEC{lc_file} = {
    summary  => 'Convert the *content* of file into all-lowercase',
    args     => {path=>'str*'},
    features => {undo=>1},
};
sub lc_file {
    my %args        = @_;
    my $path        = $args{path};
    my $undo_action = $args{-undo_action} // '';
    my $undo_info   = $args{-undo_info};

    $path = abs_path($path)
        or return [500, "Can't get file absolute path"];

    if ($undo_action eq 'undo') {
        write_file $path, $undo_info->{content}; # restore original content
        utime undef, $undo_info->{mtime}, $path; # as well as original mtime
        return [200, "OK"];
    } else {
        my @st = stat($path)
            or return [500, "Can't stat file"];
        my $content = read_file($path);
        my $undo_info = {mtime=>$st[9], content=>$content};
        write_file $path, lc($content);
        return [200, "OK", undef, {undo_info=>$undo_info}];
    }
}

dry_run => 1

Specifies that subroutine supports dry-run (simulation) mode. Example:

use Log::Any '$log';

$SPEC{rmre} = {
    summary  => 'Delete files in curdir matching a regex',
    args     => {re=>'str*'},
    features => {dry_run=>1}
};
sub rmre {
    my %args    = @_;
    my $re      = qr/$args{re}/;
    my $dry_run = $args{-dry_run};

    opendir my($dir), ".";
    while (my $f = readdir($dir)) {
        next unless $f =~ $re;
        $log->info("Deleting $f ...");
        next if $dry_run;
        unlink $f;
    }
    [200, "OK"];
}

pure => 1

Specifies that subroutine is "pure" and has no "side effects" (these are terms from functional programming / computer science). Having a side effect means changing something, somewhere (e.g. setting the value of a global variable, modifies its arguments, writing some data to disk, changing system date/time, etc.) Specifying a function as pure means, among others:

  • the function needs not be involved in undo operation;

  • you can safely include it during dry run;

TODO

transaction/atomicity

i18n (is translatable, supported languages, locales)

progress bar/estimated duration (we can query each sub how long would an action takes beforehand, or caller pas a callback that gets called multiple times to update progress bar)

SEE ALSO

Sub::Spec

AUTHOR

Steven Haryanto <stevenharyanto@gmail.com>

COPYRIGHT AND LICENSE

This software is copyright (c) 2011 by Steven Haryanto.

This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.