Group
Extension

MOP4Import-Declare/Base/CLI_JSON.pm

#!/usr/bin/env perl
package MOP4Import::Base::CLI_JSON;
use strict;
use warnings;
use Carp ();

use constant DEBUG => $ENV{DEBUG_MOP4IMPORT};
BEGIN {
  print STDERR "Using (file '" . __FILE__ . "')\n"
    , (DEBUG >= 3 ? Carp::longmess("callstack:") : ())
    if DEBUG and DEBUG >= 2
}

use MOP4Import::Base::CLI -as_base
  , [constant => parse_opts__preserve_hyphen => 1]
  , [fields =>
     , ['help' => doc => "show this help message", json_type => 'string']
     , ['quiet' => doc => 'to be (somewhat) quiet', json_type => 'int']
     , ['scalar' => doc => "evaluate methods in scalar context", json_type => 'bool']
     , ['output' => default => 'ndjson'
        , doc => "choose output serializer (ndjson/json/tsv/dump)"
        , json_type => 'string'
      ]
     , ['flatten'
        , json_type => 'bool'
      ]
     , ['undef-as' => default => 'null'
        , doc => "serialize undef as this value. used in tsv output"
        , json_type => 'string'
      ]
     , ['no-exit-code'
        , doc => "exit with 0(EXIT_SUCCESS) even when result was falsy/empty"
        , json_type => 'bool'
      ]
     , ['binary' => default => 0, doc => "keep STDIN/OUT/ERR binary friendly"
        , json_type => 'bool'
      ]
   ];

use JSON::MaybeXS;
use MOP4Import::Base::JSON -as_base;

use MOP4Import::Opts;
use MOP4Import::Util qw/lexpand globref take_locked_opts_of lock_keys_as/;

use File::Spec ();

use open ();

sub cli_precmd {
  (my MY $self) = @_;
  #
  # cli_precmd() may be called from $class->cmd_help.
  #
  unless (ref $self and $self->{binary}) {
    'open'->import(qw/:locale :std/);
  }
}

#
# Replace parse_opts to use parse_json_opts
#
sub cli_parse_opts {
  my ($pack, $list, $result, $opt_alias, $converter, %opts) = @_;

  MOP4Import::Util::parse_json_opts($pack, $list, $result, $opt_alias);
}

sub cli_eval {
  (my MY $self, my ($script, @args)) = @_;
  my $pack = ref $self;
  my $sub = do {
    local $@;
    my $code = eval qq{package $pack; use strict; sub {my \$self = shift; $script\n}};
    die $@ if $@;
    $code;
  };
  $sub->($self, @args);
}

sub cli_invoke {
  (my MY $self, my ($method, @args)) = @_;

  $self->cli_precmd($method);

  my $sub = $self->can($method)
    or Carp::croak "No such method: $method";

  my $list = $self->cli_invoke_sub($sub, $self, @args);

  $self->cli_exit_for_result($list) unless $self->{'no-exit-code'};
}

sub cli_invoke_sub {
  (my MY $self, my ($sub, $receiver, @args)) = @_;

  my @res;
  if ($self->{scalar}) {
    $res[0] = $sub->($receiver, @args);
  } else {
    @res = $sub->($receiver, @args);
  }

  $self->cli_output(\@res) unless $self->{quiet};

  \@res;
}

#
# cli_output($list) -- Output abstraction (yield).
#
sub cli_output :method {
  (my MY $self, my ($list)) = @_;

  unless ($self->{scalar} ? $list->[0] : @$list) {
    return;
  }

  my $emitter = ref $self->{output} eq 'CODE' ? $self->{output} : sub {
    $self->cli_write_fh(\*STDOUT, @_);
  };

  if ($self->{scalar}) {
    $emitter->($_) for @$list;
  } else {
    $emitter->($list);
  }
}

#
# Gather output from cli_output
#
sub cli_capture_output {
  (my MY $self, my ($subOrArrayOrString, @args)) = @_;
  my @result;
  local $self->{output} = sub {
    push @result, \@_;
  };
  $self->cli_apply($subOrArrayOrString, @args);
  @result;
}

sub cli_examine_result {
  (my MY $self, my $list) = @_;
  if ($self->{scalar}) {
    $list->[0];
  } else {
    @$list;
  }
}

#
# exit code handling
#
sub cli_exit_for_result {
  (my MY $self, my $list) = @_;

  exit($self->cli_examine_result($list) ? 0 : 1);
}

#========================================

sub cli_array :method {
  (my MY $self, my @args) = @_;
  \@args;
}

sub cli_list :method {
  (my MY $self, my @args) = @_;
  @args;
}

sub cli_object :method {
  (my MY $self, my %args) = @_;
  \%args;
}

sub cli_identity :method {
  (my MY $self, my ($thing)) = @_;
  $thing;
}

sub cli_map_apply :method {
  (my MY $self, my ($subOrArray, @args)) = @_;
  map {
    $self->cli_apply($subOrArray, $_);
  } @args;
}

sub cli_grep_apply :method {
  (my MY $self, my ($subOrArray, @args)) = @_;
  grep {
    $self->cli_apply($subOrArray, $_);
  } @args;
}

# XXX: How about cli_reduce_apply?

sub cli_apply :method {
  (my MY $self, my ($subOrArrayOrString, @args)) = @_;
  if (not defined $subOrArrayOrString) {
    Carp::croak "undefined sub for cli_apply";
  } elsif (ref $subOrArrayOrString eq 'CODE') {
    $subOrArrayOrString->(@args);
  } elsif (not ref $subOrArrayOrString or ref $subOrArrayOrString eq 'ARRAY') {
    my ($meth, @opts) = lexpand($subOrArrayOrString);
    if (my $sub = $self->can("cmd_$meth")) {
      $sub->($self, @opts, @args);
    } else {
      $self->$meth(@opts, @args);
    }
  } else {
    Carp::croak "Invalid argument for cli_apply: "
      . MOP4Import::Util::terse_dump($subOrArrayOrString);
  }
}

sub cli_precheck_apply {
  (my MY $self, my ($subOrArrayOrString)) = @_;
  if (not defined $subOrArrayOrString) {
    Carp::croak "undefined sub for cli_apply";
  } elsif (ref $subOrArrayOrString eq 'CODE') {
    1;
  } else {
    if (not ref $subOrArrayOrString or ref $subOrArrayOrString eq 'ARRAY') {
      my ($meth, @opts) = lexpand($subOrArrayOrString);
      return if $self->can("cmd_$meth") || $self->can($meth);
    }
    Carp::croak "Invalid argument for cli_apply: "
      . MOP4Import::Util::terse_dump($subOrArrayOrString);
  }
}

use MOP4Import::Types
  cliopts__xargs => [[fields => qw/null slurp single json decode/]];

sub cli_xargs_json :method {
  (my MY $self, my (@args)) = @_;
  my cliopts__xargs $opts = $self->take_locked_opts_of(
    cliopts__xargs, \@args, {0 => 'null', input => 'decode'},
  );
  $opts->{decode} //= (($opts->{json} //=1) ? 'json' : '');
  $self->_cli_xargs($opts, @args);
}

BEGIN {
  my ($packSuffix) = do {
    if ($] >= 5.022) {
      'compat_double_diamond';
    } else {
      'compat_double_diamond_5_20';
    }
  };
  (my $dir = __FILE__) =~ s,/?[^/]+\z,,;

  my $fn = File::Spec->rel2abs($dir) . "/../Util/$packSuffix.pm";
  if (-r __FILE__ and not -r $fn) {
    die "Can't load $fn";
  }
  do $fn;
  my $pack = 'MOP4Import::Util::'.$packSuffix;
  $pack->import;
  print STDERR "compat_diamond is loaded from $fn\n" if DEBUG and DEBUG >= 2;
}

sub _cli_xargs {
  (my MY $self, my (@args)) = @_;
  my cliopts__xargs $opts = $self->take_locked_opts_of(
    cliopts__xargs, \@args, {0 => 'null'},
  );
  my ($subOrArray, @restPrefix) = @args;
  $self->cli_precheck_apply($subOrArray);

  local $/ = $opts->{null} ? "\0" : "\n";
  local *ARGV;
  if ($opts->{slurp} || $opts->{single}) {
    my @all = $self->cli_slurp_xargs($opts);
    $self->cli_apply(
      $subOrArray, @restPrefix,
      ($opts->{single} ? \@all : @all)
    );
  } else {
    my $decoder = defined $opts->{decode}
      ? $self->cli_decoder_from($opts->{decode}) : undef;
    local $_;
    if (not ref $subOrArray and $self->can("cmd_$subOrArray")) {
      while (defined($_ = $self->cli_compat_diamond)) {
        chomp;
        $self->cli_apply(
          $subOrArray, @restPrefix,
          ($decoder ? $decoder->($_) : $_)
        )
      }
      $self->{'no-exit-code'} = 1;
      ();
    } else {
      my @result;
      while (defined($_ = $self->cli_compat_diamond)) {
        chomp;
        # XXX: yield...
        push @result, $self->cli_apply(
          $subOrArray, @restPrefix,
          ($decoder ? $decoder->($_) : $_)
        )
      }
      @result;
    }
  }
}

sub cli_slurp_xargs_json {
  (my MY $self, my (@args)) = @_;
  my cliopts__xargs $opts = $self->take_locked_opts_of(
    cliopts__xargs, \@args, {0 => 'null'},
  );
  $opts->{decode} //= (($opts->{json} //=1) ? 'json' : '');
  $self->cli_slurp_xargs($opts, @args);
}

sub cli_slurp_xargs {
  (my MY $self, my (@args)) = @_;
  my cliopts__xargs $opts = $self->take_locked_opts_of(
    cliopts__xargs, \@args, {0 => 'null'},
  );

  local @ARGV = @args;
  my $decoder = defined $opts->{decode}
    ? $self->cli_decoder_from($opts->{decode}) : undef;

  map {
    $decoder ? $decoder->($_) : $_
  } $self->cli_compat_diamond
}

sub cli_decoder_from {
  (my MY $self, my ($formatSpec, @rest)) = @_;
  my ($format, @opts) = lexpand($formatSpec);
  my $sub = $self->can("cli_decoder_from__$format")
    or Carp::croak "Unknown decorder is requested: $format";
  $sub->($self, @opts, @rest);
}

#
# pass-through decoder.
#
sub cli_decoder_from__ {
  sub {$_[0]}
}

#
# json decoder
#
sub cli_decoder_from__json {
  (my MY $self, my @opts) = @_;
  my $decoder = $self->cli_json_decoder(qw/allow_nonref/, @opts);
  sub {
    my ($str) = @_;
    Encode::_utf8_off($str);
    $decoder->decode($str);
  }
}

#========================================

sub declare_output_format {
  (my $myPack, my Opts $opts, my ($formatName, $sub)) = m4i_args(@_);
  my $encoderFuncName = "cli_encoder_to__$formatName";
  my $writeFuncName = "cli_write_fh_as_$formatName";
  my $outputFuncName = "cli_output_as_$formatName";
  if (ref $sub eq 'CODE') {
    *{globref($opts->{destpkg}, $writeFuncName)} = $sub;
    *{globref($opts->{destpkg}, $outputFuncName)} = sub {
      shift->$writeFuncName(\*STDOUT, $_[0]);
    };
  } elsif (not defined $sub) {
    if ($opts->{destpkg}->can($writeFuncName)) {
      *{globref($opts->{destpkg}, $outputFuncName)} = sub {
        shift->$writeFuncName(\*STDOUT, $_[0]);
      };
    }
    elsif ($opts->{destpkg}->can($encoderFuncName)) {
      *{globref($opts->{destpkg}, $outputFuncName)} = sub {
        shift->$encoderFuncName(\*STDOUT)->($_[0]);
      };

      unless ($opts->{destpkg}->can($writeFuncName)) {
        *{globref($opts->{destpkg}, $writeFuncName)} = sub {
          shift->$encoderFuncName(shift)->(\@_);
        };
      }
    }
    else {
      Carp::croak "output_format $formatName doesn't have method '$writeFuncName'";
    }
  } else {
    Carp::croak "Invalid argument for output_format: "
      . MOP4Import::Util::terse_dump($sub);
  }
}

# cli_write_fh($outFH, @output) -- Output format abstraction
# In cli_run context, cli_write_fh is called from cli_output($list) and
# used as cli_write_fh(\*STDOUT, $list).
# But in general, cli_write_fh can process multiple arguments at once.
#

sub cli_write_fh {
  (my MY $self, my ($outFH, @args)) = @_;

  my ($outputFmt, @opts) = lexpand($self->{'output'});

  if (my $sub = $self->can("cli_encoder_to__$outputFmt")) {
    my $encoder = $sub->($self, $outFH, @opts);
    $encoder->($_) for ($self->{flatten} ? (map {
      (defined $_ && ref $_ eq 'ARRAY') ? @$_ : $_
    } @args) : @args);
  }
  elsif ($sub = $self->can("cli_write_fh_as_".$outputFmt)) {
    $sub->($self, $outFH, $self->{flatten} ? (map {
      (defined $_ && ref $_ eq 'ARRAY') ? @$_ : $_
    } @args) : @args);
  }
  else {
    Carp::croak("Unknown output format: $self->{'output'}");
  }
}

sub cli_json { JSON() }

sub cli_decode_json {
  (my MY $self, my $string) = @_;
  $self->cli_decoder_from__json->($string);
}

sub cli_encode_json {
  (my MY $self, my ($obj, $json_type)) = @_;
  my $json = $self->SUPER::cli_encode_json($obj, $json_type);
  Encode::_utf8_on($json) unless $self->{binary};
  $json;
}

sub cli_json_encoder {
  (my MY $self) = @_;
  my $js = $self->SUPER::cli_json_encoder;
  $js->utf8 unless $self->{binary};
  $js;
}

sub cli_json_decoder {
  (my MY $self, my @opts) = @_;
  my $js = JSON()->new->relaxed;
  $js->utf8 unless $self->{binary};
  foreach my $opt (@opts) {
    my ($method, @args) = lexpand($opt);
    $js->$method(@args);
  }
  $js;
}

#----------------------------------------

sub cli_encode_as {
  (my MY $self, my ($outputSpec, @items)) = @_;
  my ($outputFmt, $layer, @opts) = lexpand($outputSpec);
  $outputFmt //= '';
  # Allow $layer to be a HASH
  if (defined $layer and ref $layer eq 'HASH') {
    my %opts = %$layer;
    $layer = delete $opts{layer};
    unshift @opts, %opts;
  }
  $layer //= '';
  my $buffer = "";
  {
    open my $outFH, ">$layer", \$buffer;
    if (my $sub = $self->can("cli_encoder_to__$outputFmt")) {
      my $encoder = $sub->($self, $outFH, @opts);
      foreach my $item (@items) {
        $encoder->($item);
      }
    }
    elsif ($sub = $self->can("cli_write_fh_as_$outputFmt")) {
      $sub->($self, $outFH, \@items);
    }
    else {
      Carp::croak "Unknown output format: '$outputFmt'";
    }
  }
  $buffer;
}

MY->declare_output_format(MY, 'ndjson');
sub cli_write_fh_as_ndjson {
  (my MY $self, my ($outFH, @tables)) = @_;
  foreach my $table (@tables) {
    foreach my $item (ref $table eq 'ARRAY' ? @$table : $table) {
      print $outFH ((ref $item ? $self->cli_encode_json($item) : $item // $self->{'undef-as'} // ''), "\n");
    }
  }
}

MY->declare_output_format(MY, 'json');
sub cli_write_fh_as_json {
  (my MY $self, my ($outFH, @args)) = @_;
  foreach my $item (@args) {
    print $outFH $self->cli_encode_json($item), "\n";
  }
}

MY->declare_output_format(MY, 'yaml');
sub cli_write_fh_as_yaml {
  (my MY $self, my ($outFH, @args)) = @_;
  require YAML::Syck;
  print $outFH YAML::Syck::Dump(@args);
}

MY->declare_output_format(MY, 'dump');
sub cli_write_fh_as_dump {
  (my MY $self, my ($outFH, @args)) = @_;
  foreach my $item (@args) {
    my $dumper = Data::Dumper->new($item)
      ->Terse(1)
      ->Sortkeys(1)
      ->Indent(1)
      ->Deparse(1);
    if (my $sub = $dumper->can("Trailingcomma")) {
      $sub->($dumper, 1);
    }

    print $outFH $dumper->Dump;
  }
}

MY->declare_output_format(MY, 'raw');
sub cli_write_fh_as_raw {
  (my MY $self, my ($outFH, @tables)) = @_;
  foreach my $table (@tables) {
    foreach my $item (ref $table eq 'ARRAY' ? @$table : $table) {
      print $outFH $item;
    }
  }
}

MY->declare_output_format(MY, 'tsv');
sub cli_write_fh_as_tsv {
  (my MY $self, my ($outFH, @tables)) = @_;
  foreach my $table (@tables) {
    foreach my $rec (@$table) {
      print $outFH join("\t", map {
        if (not defined $_) {
          $self->{'undef-as'};
        } elsif (ref $_) {
          $self->cli_encode_json($_);
        } else {
          my $cp = $_;
          $cp =~ s/[\t\n]+/ /g;
          $cp
        }
      } ref $rec eq 'ARRAY' ? @$rec : $rec), "\n";
    }
  }
}

#========================================

sub cli_create_from_file :method {
  my ($class, $configFn, @moreOpts) = @_;
  my $realConfigFn = File::Spec->rel2abs($configFn);
  my $oldcwd = $ENV{PWD} || do {require Cwd; Cwd::getcwd()};
  my $realDir = File::Basename::dirname($realConfigFn);
  chdir($realDir)
    or Carp::croak "Can't chdir to $realDir: $!";

  # Read $configFn with scalar context.
  my $opts = $class->cli_read_file($realConfigFn);

  my $object = (ref $class || $class)->new(
    ref $opts eq 'HASH' ? %$opts : @$opts,
    @moreOpts
  );
  chdir($oldcwd)
    or Carp::croak "Can't chdir back to $oldcwd: $!";
  $object;
}

sub cli_read_file :method {
  my ($classOrObj, $fileNameSpec, %moreOpts) = @_;
  my ($fileName, %opts) = lexpand($fileNameSpec);
  my ($ftype) = $fileName =~ m{\.(\w+)$};
  $ftype //= "";

  my $sub = $classOrObj->can("cli_read_file__$ftype")
    or Carp::croak "Unsupported file type '$ftype': $fileName";

  $sub->($classOrObj, $fileName, %opts, %moreOpts);
}

# No filename extension => read entire content except last \n.
sub cli_read_file__ {
  my ($classOrObj, $fileName) = @_;
  open my $fh, '<:utf8', $fileName
    or Carp::croak "Can't open $fileName: $!";
  my $all = do {local $/; <$fh>};
  chomp($all);
  $all;
}

# .txt => array of lines
sub cli_read_file__txt {
  my ($classOrObj, $fileName) = @_;
  my $all = $classOrObj->cli_read_file__($fileName);
  my @list = split "\n", $all;
  wantarray ? @list : \@list;
}

# .yml
*cli_read_file__yaml = *cli_read_file__yml;*cli_read_file__yaml = *cli_read_file__yml;
sub cli_read_file__yml {
  my ($classOrObj, $fileName) = @_;
  require YAML::Syck;
  YAML::Syck::LoadFile($fileName);
}

# .json
sub cli_read_file__json {
  my ($classOrObj, $fileName, %opts) = @_;
  my $allow_comments = delete $opts{allow_comments};
  open my $fh, '<', $fileName
    or Carp::croak "Can't open $fileName: $!";
  my $all = do {local $/; <$fh>};
  unless (defined $all) {
    Carp::croak "Can't read $fileName: $!";
  }
  if ($allow_comments) {
    require MOP4Import::Util::CommentedJson;
    local $@;
    eval {
      $all = MOP4Import::Util::CommentedJson->strip_json_comments($all);
    };
    if ($@) {
      Carp::carp "Can't strip comment in $fileName: $@";
    }
  }
  local $@;
  my @result;
  eval {
    @result = $classOrObj->cli_json_decoder->incr_parse($all);
  };
  if ($@) {
    Carp::croak "decode_json failed in $fileName: $@";
  }
  @result >= 2 ? \@result : $result[0];
}

MY->cli_run(\@ARGV) unless caller;

1;


Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.