summaryrefslogtreecommitdiff
path: root/bin/cil
diff options
context:
space:
mode:
Diffstat (limited to 'bin/cil')
-rwxr-xr-xbin/cil1004
1 files changed, 50 insertions, 954 deletions
diff --git a/bin/cil b/bin/cil
index 4ff4d32..a2b4131 100755
--- a/bin/cil
+++ b/bin/cil
@@ -21,7 +21,6 @@ use strict;
use warnings;
use Getopt::Mixed "nextOption";
-use Digest::MD5 qw(md5_hex);
use File::Touch;
use File::Glob ':glob';
use File::Basename;
@@ -33,14 +32,11 @@ use CIL;
use CIL::Issue;
use CIL::Comment;
use CIL::Attachment;
+use CIL::Utils;
## ----------------------------------------------------------------------------
# constants
-my $y = 'y';
-
-use constant VERSION => '0.5.1';
-
my @IN_OPTS = (
# strings
'p=s', # p = path
@@ -55,12 +51,19 @@ my @IN_OPTS = (
'created-by>c', # for 'summary', 'list'
'a=s', # a = assigned_to
'assigned-to>a',# for 'summary', 'list'
+ 'r=s', # r = revision
+ 'revision>s', # for all query commands
# booleans
'bare', # for 'init'
'is-open', # for 'summary', 'list'
'is-closed', # for 'summary', 'list'
'is-mine', # for 'summary', 'list'
+ 'mine', # for 'add'
+ 'add', # for 'add', 'comment'
+ 'commit', # for 'add', 'comment'
+ 'batch', # for 'am'
+ 'remove', # for 'label'
'help',
'version',
);
@@ -72,6 +75,11 @@ my %BOOLEAN_ARGS = (
'is-open' => 1,
'is-closed' => 1,
'is-mine' => 1,
+ 'mine' => 1,
+ 'add' => 1,
+ 'commit' => 1,
+ 'batch' => 1,
+ 'remove' => 1,
);
## ----------------------------------------------------------------------------
@@ -82,7 +90,7 @@ my %BOOLEAN_ARGS = (
# do the version and help
if ( exists $args->{version} ) {
- print "cil version ".VERSION."\n";
+ print "cil version " . $CIL::VERSION . "\n";
exit;
}
@@ -95,901 +103,31 @@ my %BOOLEAN_ARGS = (
Getopt::Mixed::abortMsg('specify a command')
if @ARGV == 0;
- my $command = shift @ARGV;
- $command =~ s{-}{_}gxms;
- no strict 'refs';
- if ( not defined &{"cmd_$command"} ) {
- Getopt::Mixed::abortMsg("'$command' is not a valid cil command.");
- }
+ my $command_name = shift @ARGV;
+ my( $command ) = grep { $command_name eq $_->name } CIL->commands
+ or Getopt::Mixed::abortMsg("'$command_name' is not a valid cil command.");
+ # make a new $cil object
my $cil = CIL->new();
- $cil->read_config_user();
- $cil->read_config_file();
-
- &{"cmd_$command"}($cil, $args, @ARGV);
-}
-
-## ----------------------------------------------------------------------------
-# commands
-
-sub cmd_init {
- my ($cil, $args) = @_;
-
- my $path = $args->{p} || '.'; # default path is right here
-
- # error if $path doesn't exist
- unless ( -d $path ) {
- fatal("path '$path' doesn't exist");
- }
-
- # error if issues/ already exists
- my $issues_dir = "$path/issues";
- if ( -d $issues_dir ) {
- fatal("issues directory '$issues_dir' already exists, not initialising issues");
- }
-
- # error if .cil already exists
- my $config = "$path/.cil";
- if ( -f $config ) {
- fatal("config file '$config' already exists, not initialising issues");
- }
-
- # try to create the issues/ dir
- unless ( mkdir $issues_dir ) {
- fatal("Couldn't create '$issues_dir' directory: $!");
- }
-
- # create a .cil file here also
- if ( $args->{bare} ) {
- unless ( touch $config ) {
- rmdir $issues_dir;
- fatal("couldn't create a '$config' file");
- }
- }
- else {
- # write a default .cil file
- write_file($config, <<'CONFIG');
-StatusStrict: 1
-StatusAllowedList: New
-StatusAllowedList: InProgress
-StatusAllowedList: Finished
-StatusOpenList: New
-StatusOpenList: InProgress
-StatusClosedList: Finished
-LabelStrict: 1
-LabelAllowedList: Type-Enhancement
-LabelAllowedList: Type-Defect
-LabelAllowedList: Priority-High
-LabelAllowedList: Priority-Medium
-LabelAllowedList: Priority-Low
-CONFIG
- }
-
- # add a README.txt so people know what this is about
- unless ( -f "$issues_dir/README.txt" ) {
- write_file("$issues_dir/README.txt", <<'README');
-This directory is used by CIL to track issues and feature requests.
-
-The home page for CIL is at http://kapiti.geek.nz/software/cil.html
-README
- }
-
- # $path/issues/ and $path/.cil create correctly
- msg("initialised empty issue list inside '$path/'");
-}
-
-sub cmd_list {
- my ($cil, $args) = @_;
-
- check_paths($cil);
-
- # find all the issues
- my $issues = $cil->get_issues();
- $issues = filter_issues( $cil, $issues, $args );
- if ( @$issues ) {
- foreach my $issue ( sort { $a->Inserted cmp $b->Inserted } @$issues ) {
- separator();
- display_issue_headers($cil, $issue);
- }
- separator();
- }
- else {
- msg('no issues found');
- }
-}
-
-sub cmd_summary {
- my ($cil, $args) = @_;
-
- check_paths($cil);
-
- # find all the issues
- my $issues = $cil->get_issues();
- $issues = filter_issues( $cil, $issues, $args );
- if ( @$issues ) {
- separator();
- foreach my $issue ( @$issues ) {
- display_issue_summary($cil, $issue);
- }
- separator();
- }
- else {
- msg('no issues found');
- }
-}
-
-sub cmd_show {
- my ($cil, undef, $issue_name) = @_;
-
- # firstly, read the issue in
- my $issue = load_issue_fuzzy( $cil, $issue_name );
- display_issue_full($cil, $issue);
-}
-
-sub cmd_status {
- my ($cil, undef, $issue_name, $status) = @_;
-
- unless ( defined $status ) {
- fatal("provide a status to set this issue to");
- }
-
- # firstly, read the issue in
- my $issue = load_issue_fuzzy( $cil, $issue_name );
-
- # set the status for this issue
- $issue->Status( $status );
- $issue->save($cil);
-
- display_issue($cil, $issue);
-}
-
-sub cmd_steal {
- my ($cil, undef, $issue_name) = @_;
-
- # firstly, read the issue in
- my $issue = load_issue_fuzzy( $cil, $issue_name );
-
- # set the AssignedTo for this issue to you (because you're stealing it)
- $issue->AssignedTo( user($cil) );
- $issue->save($cil);
-
- display_issue($cil, $issue);
-}
-
-sub cmd_add {
- my ($cil, undef, $issue_name) = @_;
-
- CIL::Utils->ensure_interactive();
-
- my $user = user($cil);
-
- my $issue = CIL::Issue->new('tmpname');
- $issue->Status('New');
- $issue->CreatedBy( $user );
- $issue->Description("Description ...");
-
- add_issue_loop($cil, undef, $issue);
-}
-
-sub cmd_edit {
- my ($cil, undef, $issue_name) = @_;
-
- my $issue = load_issue_fuzzy( $cil, $issue_name );
-
- CIL::Utils->ensure_interactive();
-
- my $edit = $y;
-
- # keep going until we get a valid issue or we want to quit
- while ( $edit eq $y ) {
- # read in the new issue text
- my $fh = CIL::Utils->solicit( $issue->as_output );
- $issue = CIL::Issue->new_from_fh( $issue->name, $fh );
-
- # check if the issue is valid
- if ( $issue->is_valid($cil) ) {
- $edit = 'n';
- }
- else {
- msg($_) foreach @{ $issue->errors };
- $edit = ans('Would you like to re-edit (y/n): ');
- }
- }
-
- # if the issue is still invalid, they quit without correcting it
- return unless $issue->is_valid( $cil );
-
- # save it
- $issue->save($cil);
- display_issue($cil, $issue);
-}
-
-sub cmd_comment {
- my ($cil, undef, $issue_name) = @_;
-
- my $issue = load_issue_fuzzy( $cil, $issue_name );
-
- CIL::Utils->ensure_interactive();
-
- # create the new comment
- my $comment = CIL::Comment->new('tmpname');
- $comment->Issue( $issue->name );
- $comment->CreatedBy( user($cil) );
- $comment->Description("Description ...");
-
- add_comment_loop($cil, undef, $issue, $comment);
-}
-
-sub cmd_attach {
- my ($cil, undef, $issue_name, $filename) = @_;
-
- my $issue = load_issue_fuzzy( $cil, $issue_name );
-
- # check to see if the file exists
- unless ( -r $filename ) {
- fatal("couldn't read file '$filename'");
- }
-
- my $basename = basename( $filename );
- my $user = user($cil);
-
- my $add_attachment_text = <<"EOF";
-Filename : $basename
-CreatedBy : $user
-
-File goes here ... this will be overwritten.
-EOF
-
- # read in the new issue text
- CIL::Utils->ensure_interactive();
- my $fh = CIL::Utils->solicit( $add_attachment_text );
-
- my $attachment = CIL::Attachment->new_from_fh( 'tmp', $fh );
- unless ( defined $attachment ) {
- fatal("could not create new attachment");
- }
-
- # now add the file itself
- my $contents = read_file( $filename );
- $attachment->set_file_contents( $contents );
-
- # set the size
- my ($size) = (stat($filename))[7];
- $attachment->Size( $size );
-
- # we've got the attachment, so let's name it
- my $unique_str = time . $attachment->Inserted . $attachment->File;
- $attachment->set_name( substr(md5_hex($unique_str), 0, 8) );
-
- # finally, tell it who it's parent is and then save
- $attachment->Issue( $issue->name );
- $attachment->save($cil);
-
- # add the comment to the issue, update it's timestamp and save it out
- $issue->add_attachment( $attachment );
- $issue->save($cil);
- display_issue_full($cil, $issue);
-}
-
-sub cmd_extract {
- my ($cil, $args, $attachment_name) = @_;
-
- my $attachment = load_attachment_fuzzy($cil, $attachment_name);
-
- my $filename = $args->{f} || $attachment->Filename();
- write_file( $filename, $attachment->as_binary );
-}
-
-sub cmd_track {
- my ($cil, $args, $issue_name) = @_;
-
- fatal("the 'VCS' option in your .cil file is not set")
- unless defined $cil->VCS;
-
- fatal("the 'VCS' option currently only supports values of 'Git'")
- unless $cil->VCS eq 'Git';
-
- my $issue = load_issue_fuzzy($cil, $issue_name);
-
- # add the issue to Git
- my $issue_dir = $cil->IssueDir();
- my @files;
- push @files, "$issue_dir/i_" . $issue->name . '.cil';
- push @files, map { "$issue_dir/c_${_}.cil" } @{ $issue->CommentList };
- push @files, map { "$issue_dir/a_${_}.cil" } @{ $issue->AttachmentList };
- msg("git add @files");
-}
-
-sub cmd_fsck {
- my ($cil, $args) = @_;
-
- # this looks at all the issues it can find and checks for:
- # * validity
- # * all the comments are there
- # * all the attachments are there
- # then it checks each individual comment/attachment for:
- # * ToDo: validity
- # * it's parent exists
-
- check_paths($cil);
-
- # find all the issues, comments and attachments
- my $issues = $cil->get_issues();
- my $issue = {};
- foreach my $i ( @$issues ) {
- $issue->{$i->name} = $i;
- }
- my $comments = $cil->get_comments();
- my $comment = {};
- foreach my $c ( @$comments ) {
- $comment->{$c->name} = $c;
- }
- my $attachments = $cil->get_attachments();
- my $attachment = {};
- foreach my $a ( @$attachments ) {
- $attachment->{$a->name} = $a;
- }
-
- # ------
- # issues
- my $errors = {};
- if ( @$issues ) {
- foreach my $i ( sort { $a->Inserted cmp $b->Inserted } @$issues ) {
- my $name = $i->name;
-
- unless ( $i->is_valid($cil) ) {
- foreach ( @{ $i->errors } ) {
- push @{$errors->{$name}}, $_;
- }
- }
-
- # check that all it's comments are there and that they have this parent
- my $comments = $i->CommentList;
- foreach my $c ( @$comments ) {
- # see if this comment exists at all
- if ( exists $comment->{$c} ) {
- # check the parent is this issue
- push @{$errors->{$name}}, "comment '$c' is listed under issue '" . $i->name . "' but does not reciprocate"
- unless $comment->{$c}->Issue eq $i->name;
- }
- else {
- push @{$errors->{$name}}, "comment '$c' listed in issue '" . $i->name . "' does not exist";
- }
- }
-
- # check that all it's attachments are there and that they have this parent
- my $attachments = $i->AttachmentList;
- foreach my $a ( @$attachments ) {
- # see if this attachment exists at all
- if ( exists $attachment->{$a} ) {
- # check the parent is this issue
- push @{$errors->{$name}}, "attachment '$a' is listed under issue '" . $i->name . "' but does not reciprocate"
- unless $attachment->{$a}->Issue eq $i->name;
- }
- else {
- push @{$errors->{$name}}, "attachment '$a' listed in issue '" . $i->name . "' does not exist";
- }
- }
-
- # check that all it's 'DependsOn' are there and that they have this under 'Precedes'
- my $depends_on = $i->DependsOnList;
- foreach my $d ( @$depends_on ) {
- # see if this issue exists at all
- if ( exists $issue->{$d} ) {
- # check the 'Precedes' is this issue
- my %precedes = map { $_ => 1 } @{$issue->{$d}->PrecedesList};
- push @{$errors->{$name}}, "issue '$d' should precede '" . $i->name . "' but doesn't"
- unless exists $precedes{$i->name};
- }
- else {
- push @{$errors->{$name}}, "issue '$d' listed as a dependency of issue '" . $i->name . "' does not exist";
- }
- }
-
- # check that all it's 'Precedes' are there and that they have this under 'DependsOn'
- my $precedes = $i->PrecedesList;
- foreach my $p ( @$precedes ) {
- # see if this issue exists at all
- if ( exists $issue->{$p} ) {
- # check the 'DependsOn' is this issue
- my %depends_on = map { $_ => 1 } @{$issue->{$p}->DependsOnList};
- push @{$errors->{$name}}, "issue '$p' should depend on '" . $i->name . "' but doesn't"
- unless exists $depends_on{$i->name};
- }
- else {
- push @{$errors->{$name}}, "issue '$p' listed as preceding issue '" . $i->name . "' does not exist";
- }
- }
- }
- }
- print_fsck_errors('Issue', $errors);
-
- # --------
- # comments
- $errors = {};
- # loop through all the comments
- if ( @$comments ) {
- # check that their parent issues exist
- foreach my $c ( sort { $a->Inserted cmp $b->Inserted } @$comments ) {
- # check that the parent of each comment exists
- unless ( exists $issue->{$c->Issue} ) {
- push @{$errors->{$c->name}}, "comment '" . $c->name . "' refers to issue '" . $c->Issue . "' but issue does not exist";
- }
- }
- }
- print_fsck_errors('Comment', $errors);
-
- # -----------
- # attachments
- $errors = {};
- # loop through all the attachments
- if ( @$attachments ) {
- # check that their parent issues exist
- foreach my $a ( sort { $a->Inserted cmp $b->Inserted } @$attachments ) {
- # check that the parent of each attachment exists
- unless ( exists $issue->{$a->Issue} ) {
- push @{$errors->{$a->name}}, "attachment '" . $a->name . "' refers to issue '" . $a->Issue . "' but issue does not exist";
- }
- }
- }
- print_fsck_errors('Attachment', $errors);
-
- # ------------
- # nothing left
- separator();
-}
-
-sub cmd_am {
- my ($cil, $args, $email_filename) = @_;
-
- unless ( -f $email_filename ) {
- fatal("couldn't load email '$email_filename'");
- }
-
- my $msg_text = read_file($email_filename);
-
- my $email = Email::Simple->new($msg_text);
- unless ( defined $email ) {
- fatal("email file '$email_filename' didn't look like an email");
- }
-
- # extract some fields
- my $subject = $email->header('Subject');
- my $from = $email->header('From');
- my $date = find_date($email)->datetime;
- my $body = $email->body;
-
- # see if we can find any issue names in either the subject or the body
- my @issue_names;
- foreach my $text ( $subject, $body ) {
- my @new = ( $text =~ /\b\#?([0-9a-f]{8})\b/gxms );
- push @issue_names, @new;
- }
-
- msg("Found possible issue names in email: ", ( join(' ', @issue_names) || '[none]' ));
-
- my %issue;
- foreach ( @issue_names ) {
- my $i = eval { CIL::Issue->new_from_name($cil, $_) };
- next unless defined $i;
-
- $issue{$i->name} = $i;
- }
-
- if ( keys %issue ) {
- msg( "Found actual issues: " . (join(' ', keys %issue)) );
- # create the new comment
- my $comment = CIL::Comment->new('tmpname');
- $comment->Issue( '...' );
- $comment->CreatedBy( $from );
- $comment->Inserted( $date );
- # $comment->Updated( $date );
- $comment->Description( $body );
+ # for all commands (except init), we need to know we can see the proper paths
+ # (ie. issues/)
+ CIL::Utils->check_paths( $cil )
+ unless $command_name eq 'init';
- # display
- display_comment($cil, $comment);
-
- # found at least one issue, so this might be a comment
- my $issue;
- if ( keys %issue == 1 ) {
- $issue = (values %issue)[0];
- }
- else {
- my $ans = ans('To which issue would you like to add this comment: ');
-
- # ToDo: decide whether we let them choose an arbitrary issue, for
- # now quit unless they choose one from the list
- return unless exists $issue{$ans};
-
- # got a valid issue_name, so set the parent name
- $issue = $issue{$ans};
- }
-
- # set the parent issue
- $comment->Issue( $issue->name );
-
- add_comment_loop($cil, undef, $issue, $comment);
- }
- else {
- msg("Couldn't find reference to any issues in the email.");
-
- # no issue found so make up the issue first
- my $issue = CIL::Issue->new('tmpname');
- $issue->Summary( $subject );
- $issue->Status( 'New' );
- $issue->CreatedBy( $from );
- $issue->AssignedTo( user($cil) );
- $issue->Inserted( $date );
- $issue->Updated( $date );
- $issue->Description( $body );
-
- # display
- display_issue_full($cil, $issue);
-
- # then ask if the user would like to add it
- msg("Couldn't find any likely issues, so this might be a new one.");
- my $ans = ans('Would you like to add this message as an issue shown above (y/n): ');
- return unless $ans eq 'y';
-
- add_issue_loop($cil, undef, $issue);
- }
-}
-
-sub cmd_depends_on {
- my ($cil, undef, $issue_name, $depends_name) = @_;
-
- my $issue = load_issue_fuzzy($cil, $issue_name);
- my $depends = load_issue_fuzzy($cil, $depends_name);
-
- $issue->add_depends_on( $depends->name );
- $depends->add_precedes( $issue->name );
-
- $issue->save($cil);
- $depends->save($cil);
-}
-
-sub cmd_precedes {
- my ($cil, undef, $issue_name, $depends_name) = @_;
-
- # switch them round and call 'DependsOn'
- cmd_depends_on($cil, undef, $depends_name, $issue_name);
-}
-
-## ----------------------------------------------------------------------------
-# helpers
-
-sub load_issue_fuzzy {
- my ($cil, $partial_name) = @_;
-
- my $issues = $cil->list_issues_fuzzy( $partial_name );
- unless ( defined $issues ) {
- fatal("Couldn't find any issues using '$partial_name'");
- }
-
- if ( @$issues > 1 ) {
- fatal('found multiple issues which match that name: ' . join(' ', map { $_->{name} } @$issues));
- }
-
- my $issue_name = $issues->[0]->{name};
- my $issue = CIL::Issue->new_from_name($cil, $issue_name);
- unless ( defined $issue ) {
- fatal("Couldn't load issue '$issue_name'");
- }
-
- return $issue;
-}
-
-sub load_comment_fuzzy {
- my ($cil, $partial_name) = @_;
-
- my $comments = $cil->list_comments_fuzzy( $partial_name );
- unless ( defined $comments ) {
- fatal("Couldn't find any comments using '$partial_name'");
- }
-
- if ( @$comments > 1 ) {
- fatal('found multiple comments which match that name: ' . join(' ', map { $_->{name} } @$comments));
- }
-
- my $comment_name = $comments->[0]->{name};
- my $comment = CIL::comment->new_from_name($cil, $comment_name);
- unless ( defined $comment ) {
- fatal("Couldn't load comment '$comment_name'");
- }
-
- return $comment;
-}
-
-sub load_attachment_fuzzy {
- my ($cil, $partial_name) = @_;
-
- my $attachments = $cil->list_attachments_fuzzy( $partial_name );
- unless ( defined $attachments ) {
- fatal("Couldn't find any attachments using '$partial_name'");
- }
-
- if ( @$attachments > 1 ) {
- fatal('found multiple attachments which match that name: ' . join(' ', map { $_->{name} } @$attachments));
- }
-
- my $attachment_name = $attachments->[0]->{name};
- my $attachment = CIL::Attachment->new_from_name($cil, $attachment_name);
- unless ( defined $attachment ) {
- fatal("Couldn't load attachment '$partial_name'");
- }
-
- return $attachment;
-}
-
-sub ans {
- my ($msg) = @_;
- print $msg;
- my $ans = <STDIN>;
- chomp $ans;
- print "\n";
- return $ans;
-}
-
-sub add_issue_loop {
- my ($cil, undef, $issue) = @_;
-
- my $edit = $y;
-
- # keep going until we get a valid issue or we want to quit
- while ( $edit eq $y ) {
- # read in the new issue text
- my $fh = CIL::Utils->solicit( $issue->as_output );
- $issue = CIL::Issue->new_from_fh( 'tmp', $fh );
-
- # check if the issue is valid
- if ( $issue->is_valid($cil) ) {
- $edit = 'n';
- }
- else {
- msg($_) foreach @{ $issue->errors };
- $edit = ans('Would you like to re-edit (y/n): ');
- }
- }
-
- # if the issue is still invalid, they quit without correcting it
- return unless $issue->is_valid( $cil );
-
- # we've got the issue, so let's name it
- my $unique_str = time . $issue->Inserted . $issue->Summary . $issue->Description;
- $issue->set_name( substr(md5_hex($unique_str), 0, 8) );
- $issue->save($cil);
- display_issue($cil, $issue);
-
- return $issue;
-}
-
-sub add_comment_loop {
- my ($cil, undef, $issue, $comment) = @_;
-
- my $edit = $y;
-
- # keep going until we get a valid issue or we want to quit
- while ( $edit eq $y ) {
- # read in the new comment text
- my $fh = CIL::Utils->solicit( $comment->as_output );
- $comment = CIL::Comment->new_from_fh( 'tmp', $fh );
-
- # check if the comment is valid
- if ( $comment->is_valid($cil) ) {
- $edit = 'n';
- }
- else {
- msg($_) foreach @{ $issue->errors };
- $edit = ans('Would you like to re-edit (y/n): ');
- }
- }
-
- # if the comment is still invalid, they quit without correcting it
- return unless $comment->is_valid( $cil );
-
- # we've got the comment, so let's name it
- my $unique_str = time . $comment->Inserted . $issue->Description;
- $comment->set_name( substr(md5_hex($unique_str), 0, 8) );
-
- # finally, save it
- $comment->save($cil);
-
- # add the comment to the issue, update it's timestamp and save it out
- $issue->add_comment( $comment );
- $issue->save($cil);
- display_issue_full($cil, $issue);
-
- return $comment;
-}
-
-sub check_paths {
- my ($cil) = @_;
-
- # make sure an issue directory is available
- unless ( -d $cil->IssueDir ) {
- fatal("couldn't find '" . $cil->IssueDir . "' directory");
- }
-}
-
-sub filter_issues {
- my ($cil, $issues, $args) = @_;
-
- # don't filter if we haven't been given anything
- return $issues unless %$args;
-
- # check that they aren't filtering on both --assigned-to and --is-mine
- if ( defined $args->{a} and defined $args->{'is-mine'} ) {
- fatal("the --assigned-to and --is-mine filters are mutually exclusive");
- }
-
- # take a copy of the whole lot first (so we don't destroy the input list)
- my @new_issues = @$issues;
-
- # firstly, get out the Statuses we want
- if ( defined $args->{s} ) {
- @new_issues = grep { $_->Status eq $args->{s} } @new_issues;
- }
-
- # then see if we want a particular label (could be a bit nicer)
- if ( defined $args->{l} ) {
- my @tmp;
- foreach my $issue ( @new_issues ) {
- push @tmp, $issue
- if grep { $_ eq $args->{l} } @{$issue->LabelList};
- }
- @new_issues = @tmp;
- }
-
- # filter out dependent on open/closed
- if ( defined $args->{'is-open'} ) {
- # just get the open issues
- @new_issues = grep { $_->is_open($cil) } @new_issues;
- }
- if ( defined $args->{'is-closed'} ) {
- # just get the closed issues
- @new_issues = grep { $_->is_closed($cil) } @new_issues;
- }
-
- # filter out 'created by'
- if ( defined $args->{c} ) {
- @new_issues = grep { $args->{c} eq $_->created_by_email } @new_issues;
- }
-
- # filter out 'assigned to'
- $args->{a} = $cil->UserEmail
- if defined $args->{'is-mine'};
- if ( defined $args->{a} ) {
- @new_issues = grep { $args->{a} eq $_->assigned_to_email } @new_issues;
- }
-
- return \@new_issues;
-}
+ $cil->read_config_user();
+ $cil->read_config_file();
-sub print_fsck_errors {
- my ($entity, $errors) = @_;
- return unless keys %$errors;
+ # add any hooks we want
+ # none yet
- separator();
- foreach my $issue_name ( keys %$errors ) {
- title( "$entity $issue_name ");
- foreach my $error ( @{$errors->{$issue_name}} ) {
- msg("* $error");
- }
- }
+ $command->run($cil, $args, @ARGV);
}
## ----------------------------------------------------------------------------
-# input/output
-
-sub display_issue_summary {
- my ($cil, $issue) = @_;
-
- my $msg = $issue->name();
- $msg .= "\t";
- $msg .= $issue->Status();
- $msg .= "\t";
- $msg .= $issue->Summary();
-
- msg($msg);
-}
-
-sub display_issue_headers {
- my ($cil, $issue) = @_;
-
- title( 'Issue ' . $issue->name() );
- field( 'Summary', $issue->Summary() );
- field( 'CreatedBy', $issue->CreatedBy() );
- field( 'AssignedTo', $issue->AssignedTo() );
- field( 'Inserted', $issue->Inserted() );
- field( 'Status', $issue->Status() );
- field( 'Labels', join(' ', @{$issue->LabelList()}) );
- field( 'DependsOn', join(' ', @{$issue->DependsOnList()}) );
- field( 'Precedes', join(' ', @{$issue->PrecedesList()}) );
-}
-
-sub display_issue {
- my ($cil, $issue) = @_;
-
- separator();
- title( 'Issue ' . $issue->name() );
- field( 'Summary', $issue->Summary() );
- field( 'Status', $issue->Status() );
- field( 'CreatedBy', $issue->CreatedBy() );
- field( 'AssignedTo', $issue->AssignedTo() );
- field( 'Label', $_ )
- foreach sort @{$issue->LabelList()};
- field( 'Comment', $_ )
- foreach sort @{$issue->CommentList()};
- field( 'Attachment', $_ )
- foreach sort @{$issue->AttachmentList()};
- field( 'DependsOn', $_ )
- foreach sort @{$issue->DependsOnList()};
- field( 'Precedes', $_ )
- foreach sort @{$issue->PrecedesList()};
- field( 'Inserted', $issue->Inserted() );
- field( 'Updated', $issue->Inserted() );
- text('Description', $issue->Description());
- separator();
-}
+# hooks
-sub display_issue_full {
- my ($cil, $issue) = @_;
-
- separator();
- title( 'Issue ' . $issue->name() );
- field( 'Summary', $issue->Summary() );
- field( 'Status', $issue->Status() );
- field( 'CreatedBy', $issue->CreatedBy() );
- field( 'AssignedTo', $issue->AssignedTo() );
- field( 'Label', $_ )
- foreach sort @{$issue->Label()};
- field( 'DependsOn', $_ )
- foreach sort @{$issue->DependsOnList()};
- field( 'Precedes', $_ )
- foreach sort @{$issue->PrecedesList()};
- field( 'Inserted', $issue->Inserted() );
- field( 'Updated', $issue->Inserted() );
- text('Description', $issue->Description());
-
- my $comments = $cil->get_comments_for( $issue );
- foreach my $comment ( @$comments ) {
- display_comment( $cil, $comment );
- }
-
- my $attachments = $cil->get_attachments_for( $issue );
- foreach my $attachment ( @$attachments ) {
- display_attachment( $cil, $attachment );
- msg();
- }
-
- separator();
-}
-
-sub display_comment {
- my ($cil, $comment) = @_;
-
- title( 'Comment ' . $comment->name() );
- field( 'CreatedBy', $comment->CreatedBy() );
- field( 'Inserted', $comment->Inserted() );
- field( 'Updated', $comment->Inserted() );
- text('Description', $comment->Description());
-}
-
-sub display_attachment {
- my ($cil, $attachment) = @_;
-
- title( 'Attachment ' . $attachment->name() );
- field( 'Filename', $attachment->Filename() );
- field( 'CreatedBy', $attachment->CreatedBy() );
- field( 'Inserted', $attachment->Inserted() );
- field( 'Updated', $attachment->Inserted() );
-}
-
-sub user {
- my ($cil) = @_;
- return $cil->UserName . ' <' . $cil->UserEmail . '>';
-}
+# none yet
## ----------------------------------------------------------------------------
# helper functions for this command line tool
@@ -1020,48 +158,6 @@ sub get_options {
return $args;
}
-sub msg {
- print ( defined $_[0] ? $_[0] : '' );
- print "\n";
-}
-
-sub separator {
- msg('=' x 79);
-}
-
-sub title {
- my ($title) = @_;
- my $msg = "--- $title ";
- $msg .= '-' x (74 - length($title));
- msg($msg);
-}
-
-sub field {
- my ($field, $value) = @_;
- my $msg = "$field";
- $msg .= " " x (12 - length($field));
- msg("$msg: " . (defined $value ? $value : '') );
-}
-
-sub text {
- my ($field, $value) = @_;
- msg "";
- msg($value);
- msg "";
-}
-
-sub err {
- print STDERR ( defined $_[0] ? $_[0] : '' );
- print STDERR "\n";
-}
-
-sub fatal {
- my ($msg) = @_;
- chomp $msg;
- print STDERR $msg, "\n";
- exit 2;
-}
-
## ----------------------------------------------------------------------------
# program info
@@ -1075,13 +171,14 @@ Commands:
summary [FILTERS...]
list [FILTERS...]
show ISSUE
- status ISSUE NEW_STATUS
+ status NEW_STATUS [ISSUES...]
+ label NEW_LABEL [ISSUES...]
steal ISSUE
edit ISSUE
comment ISSUE
attach ISSUE FILENAME
extract ATTACHMENT [--filename=FILENAME]
- am EMAIL.TXT
+ am EMAIL.TXT [--batch]
track ISSUE
fsck
@@ -1093,7 +190,7 @@ Filters:
--assigned-to=?
--is-mine
-See <http://kapiti.geek.nz/software/cil.html> for further information.
+See <http://www.chilts.org/project/cil/> for further information.
Report bugs to <andychilton\@gmail.com>.
END_USAGE
}
@@ -1159,9 +256,13 @@ and Label fields.
Shows each issue with more information. You may filter on both the Status and
Label fields.
-=item add
+=item add [--mine] [summary]
+
+Adds an issues after you have edited the input. Text passed
+after 'add' will be used as the bug summary line.
-Adds an issues after you have edited the input.
+If you use the --mine flag, the AssignedTo field is automatically set to you
+(saves having to copy and paste the CreatedBy field).
=item show ISSUE
@@ -1256,7 +357,7 @@ C<LabelStrict> turned on.
These check both C<StatusOpenList> and C<StatusClosedList> from your C<.cil>
file. If both are specified, you're likely to get no issues unless you
-explicitely defined a status as being in both lists (for whatever reason you
+explicitly defined a status as being in both lists (for whatever reason you
have).
=item --assigned-to=EMAIL_ADDRESS, --is-mine
@@ -1279,11 +380,8 @@ may be declared multiple times:
The C<.cil> file is fairly simple and an example can be seen here:
- VCS: Git
+ UseGit: 1
StatusStrict: 1
- StatusAllowedList: New
- StatusAllowedList: InProgress
- StatusAllowedList: Finished
StatusOpenList: New
StatusOpenList: InProgress
StatusClosedList: Finished
@@ -1296,12 +394,12 @@ The C<.cil> file is fairly simple and an example can be seen here:
=over
-=item VCS
+=item UseGit
-Default: empty, Type: Enum(Git)
+Default: 0, Type: Boolean (0/1)
-Currently this option only supports Git. This enables you to use the 'track'
-command.
+Determines whether to use Git or not. Some features require Git though Cil is
+perfectly usable without.
=item StatusStrict
@@ -1311,13 +409,6 @@ If this is set to a true value then cil checks that the status you enter into
an issue (after adding or editing) is also in the allowed list (see
StatusAllowedList).
-=item StatusAllowedList
-
-Default: empty, Type: List
-
-This list is checked against when adding or editing issues but only if you have
-StatusStrict on.
-
=item StatusOpenList
Default: empty, Type: List
@@ -1330,6 +421,11 @@ Default: empty, Type: List
This list is checked against when filtering with --is-closed.
+=item StatusAllowedList
+
+This list is automatically generated from the StatusOpenList and the
+StatusClosedList. It does not have to appear in the config file.
+
=item LabelStrict
Default: 0, Type: Boolean (0/1)