| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package App::JobLog::Command::edit; |
|
2
|
|
|
|
|
|
|
$App::JobLog::Command::edit::VERSION = '1.041'; |
|
3
|
|
|
|
|
|
|
# ABSTRACT: edit the log |
|
4
|
|
|
|
|
|
|
|
|
5
|
2
|
|
|
2
|
|
1721
|
use App::JobLog -command; |
|
|
2
|
|
|
|
|
4
|
|
|
|
2
|
|
|
|
|
15
|
|
|
6
|
2
|
|
|
2
|
|
751
|
use Modern::Perl; |
|
|
2
|
|
|
|
|
4
|
|
|
|
2
|
|
|
|
|
14
|
|
|
7
|
2
|
|
|
|
|
13
|
use Class::Autouse qw{ |
|
8
|
|
|
|
|
|
|
App::JobLog::Log |
|
9
|
|
|
|
|
|
|
App::JobLog::Log::Line |
|
10
|
|
|
|
|
|
|
Digest::MD5 |
|
11
|
|
|
|
|
|
|
FileHandle |
|
12
|
2
|
|
|
2
|
|
310
|
}; |
|
|
2
|
|
|
|
|
3
|
|
|
13
|
2
|
|
|
2
|
|
162
|
use autouse 'File::Temp' => qw(tempfile); |
|
|
2
|
|
|
|
|
3
|
|
|
|
2
|
|
|
|
|
11
|
|
|
14
|
2
|
|
|
2
|
|
446
|
use autouse 'File::Copy' => qw(copy); |
|
|
2
|
|
|
|
|
4
|
|
|
|
2
|
|
|
|
|
8
|
|
|
15
|
2
|
|
|
2
|
|
337
|
use autouse 'App::JobLog::Config' => qw(editor log); |
|
|
2
|
|
|
|
|
3
|
|
|
|
2
|
|
|
|
|
18
|
|
|
16
|
2
|
|
|
2
|
|
251
|
use autouse 'Getopt::Long::Descriptive' => qw(prog_name); |
|
|
2
|
|
|
|
|
5
|
|
|
|
2
|
|
|
|
|
7
|
|
|
17
|
2
|
|
|
2
|
|
109
|
use autouse 'App::JobLog::TimeGrammar' => qw(parse); |
|
|
2
|
|
|
|
|
8
|
|
|
|
2
|
|
|
|
|
8
|
|
|
18
|
2
|
|
|
2
|
|
107
|
use autouse 'App::JobLog::Time' => qw(now); |
|
|
2
|
|
|
|
|
4
|
|
|
|
2
|
|
|
|
|
7
|
|
|
19
|
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
sub execute { |
|
21
|
0
|
|
|
0
|
1
|
|
my ( $self, $opt, $args ) = @_; |
|
22
|
0
|
0
|
0
|
|
|
|
if ( $opt->close || $opt->validate ) { |
|
|
|
0
|
|
|
|
|
|
|
23
|
0
|
|
|
|
|
|
eval { |
|
24
|
0
|
|
|
|
|
|
my $log = App::JobLog::Log->new; |
|
25
|
0
|
0
|
|
|
|
|
if ( $opt->close ) { |
|
26
|
0
|
|
|
|
|
|
my $time = join ' ', @$args; |
|
27
|
0
|
|
|
|
|
|
my ($s) = parse($time); |
|
28
|
0
|
0
|
|
|
|
|
$self->usage_error( |
|
29
|
|
|
|
|
|
|
'you may only insert closing times prior to present') |
|
30
|
|
|
|
|
|
|
unless $s < now; |
|
31
|
0
|
|
|
|
|
|
my ( $e, $i ) = $log->find_previous($s); |
|
32
|
0
|
0
|
|
|
|
|
$self->usage_error('log does not contain appropriate event') |
|
33
|
|
|
|
|
|
|
unless $e; |
|
34
|
0
|
0
|
|
|
|
|
$self->usage_error('no open event at this time') |
|
35
|
|
|
|
|
|
|
unless $e->is_open; |
|
36
|
0
|
|
|
|
|
|
$log->insert( $i + 1, |
|
37
|
|
|
|
|
|
|
App::JobLog::Log::Line->new( time => $s, done => 1 ) ); |
|
38
|
|
|
|
|
|
|
} |
|
39
|
0
|
0
|
|
|
|
|
if ( $opt->validate ) { |
|
40
|
0
|
|
|
|
|
|
my $errors = $log->validate; |
|
41
|
0
|
|
|
|
|
|
_error_report($errors); |
|
42
|
|
|
|
|
|
|
} |
|
43
|
|
|
|
|
|
|
}; |
|
44
|
0
|
0
|
|
|
|
|
$self->usage_error($@) if $@; |
|
45
|
|
|
|
|
|
|
} |
|
46
|
|
|
|
|
|
|
elsif ( my $editor = editor ) { |
|
47
|
0
|
0
|
|
|
|
|
if ( my $log = log ) { |
|
48
|
0
|
|
|
|
|
|
my ( $fh, $fn ) = tempfile; |
|
49
|
0
|
|
|
|
|
|
binmode $fh; |
|
50
|
0
|
|
|
|
|
|
copy( $log, $fh ); |
|
51
|
0
|
|
|
|
|
|
$fh->close; |
|
52
|
0
|
|
|
|
|
|
$fh = FileHandle->new($log); |
|
53
|
0
|
|
|
|
|
|
my $md5 = Digest::MD5->new; |
|
54
|
0
|
|
|
|
|
|
my $md51 = $md5->addfile($fh)->hexdigest; |
|
55
|
0
|
|
|
|
|
|
system "$editor $log"; |
|
56
|
0
|
|
|
|
|
|
$fh = FileHandle->new($log); |
|
57
|
0
|
|
|
|
|
|
my $md52 = $md5->reset->addfile($fh)->hexdigest; |
|
58
|
|
|
|
|
|
|
|
|
59
|
0
|
0
|
|
|
|
|
if ( $md51 ne $md52 ) { |
|
60
|
0
|
|
|
|
|
|
$fh = FileHandle->new( "$log.bak", 'w' ); |
|
61
|
0
|
|
|
|
|
|
copy( $fn, $fh ); |
|
62
|
0
|
|
|
|
|
|
$fh->close; |
|
63
|
0
|
|
|
|
|
|
say "saved backup log in $log.bak"; |
|
64
|
0
|
|
|
|
|
|
my $errors = App::JobLog::Log->new->validate; |
|
65
|
0
|
|
|
|
|
|
_error_report($errors); |
|
66
|
|
|
|
|
|
|
} |
|
67
|
|
|
|
|
|
|
else { |
|
68
|
0
|
|
|
|
|
|
unlink $fn; |
|
69
|
|
|
|
|
|
|
} |
|
70
|
|
|
|
|
|
|
} |
|
71
|
|
|
|
|
|
|
else { |
|
72
|
0
|
|
|
|
|
|
say 'nothing in log to edit'; |
|
73
|
|
|
|
|
|
|
} |
|
74
|
|
|
|
|
|
|
} |
|
75
|
|
|
|
|
|
|
else { |
|
76
|
0
|
0
|
|
|
|
|
$self->usage_error('no editor specified') unless $opt->close; |
|
77
|
|
|
|
|
|
|
} |
|
78
|
|
|
|
|
|
|
} |
|
79
|
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
sub usage_desc { |
|
81
|
0
|
|
|
0
|
1
|
|
'%c ' . __PACKAGE__->name . ' [--validate] [-c ]'; |
|
82
|
|
|
|
|
|
|
} |
|
83
|
|
|
|
|
|
|
|
|
84
|
0
|
|
|
0
|
1
|
|
sub abstract { 'open a text editor to edit the log' } |
|
85
|
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
sub full_description { |
|
87
|
0
|
|
|
0
|
0
|
|
<
|
|
88
|
|
|
|
|
|
|
Close an open task or open a text editor to edit the log. |
|
89
|
|
|
|
|
|
|
|
|
90
|
|
|
|
|
|
|
Closing an open task is the only edit you'll commonly have to make (it's |
|
91
|
|
|
|
|
|
|
easy to forget to close the last task of the day). Fortunately, it is the easiest |
|
92
|
|
|
|
|
|
|
edit to perform. You simply type |
|
93
|
|
|
|
|
|
|
|
|
94
|
0
|
|
|
|
|
|
@{[prog_name]} @{[__PACKAGE__->name]} --close yesterday at 8:00 pm |
|
|
0
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
|
|
96
|
0
|
|
|
|
|
|
for example and @{[prog_name]} will insert the appropriate line if it can do so. |
|
97
|
|
|
|
|
|
|
If it can't because there is no open task at the time specified, it will emit a warning |
|
98
|
|
|
|
|
|
|
instead. |
|
99
|
|
|
|
|
|
|
|
|
100
|
0
|
|
|
|
|
|
The date and time parsing is handled by the same code used by the @{[App::JobLog::Command::summary->name]} command, |
|
101
|
|
|
|
|
|
|
so what works for one works for the other. One generally does not specify hours and such |
|
102
|
0
|
|
|
|
|
|
for summaries, but @{[prog_name]} will understand most common natural language time expressions. |
|
103
|
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
If you need to do more extensive editing of the log this command will open a text editor |
|
105
|
|
|
|
|
|
|
for you and confirm the validity of the log after you save, commenting out |
|
106
|
|
|
|
|
|
|
ill-formed lines and printing a warning. This command requires the you |
|
107
|
|
|
|
|
|
|
to have set editor configuration parameter to specify a text. |
|
108
|
|
|
|
|
|
|
The text editor must be invokable like so, |
|
109
|
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
|
|
112
|
|
|
|
|
|
|
That is, you must be able to specify the file to edit as an argument. If the editor |
|
113
|
|
|
|
|
|
|
requires any additional arguments or options you must provide those via the |
|
114
|
|
|
|
|
|
|
environment variable. |
|
115
|
|
|
|
|
|
|
END |
|
116
|
|
|
|
|
|
|
} |
|
117
|
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
sub options { |
|
119
|
|
|
|
|
|
|
return ( |
|
120
|
|
|
|
|
|
|
[ |
|
121
|
0
|
|
|
0
|
0
|
|
'close|close-task|c' => |
|
122
|
|
|
|
|
|
|
'add a "DONE" line to the log at the specified moment' |
|
123
|
|
|
|
|
|
|
], |
|
124
|
|
|
|
|
|
|
[ 'validate|v' => 'check log for errors, commenting out any found' ], |
|
125
|
|
|
|
|
|
|
); |
|
126
|
|
|
|
|
|
|
} |
|
127
|
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
sub _error_report { |
|
129
|
0
|
|
|
0
|
|
|
my $errors = shift; |
|
130
|
|
|
|
|
|
|
|
|
131
|
0
|
0
|
|
|
|
|
if ($errors) { |
|
132
|
0
|
|
|
|
|
|
say "errors found: $errors"; |
|
133
|
0
|
|
|
|
|
|
say 'Error messages have been inserted into the log. Please edit.'; |
|
134
|
|
|
|
|
|
|
} |
|
135
|
|
|
|
|
|
|
else { |
|
136
|
0
|
|
|
|
|
|
say 'log is valid'; |
|
137
|
|
|
|
|
|
|
} |
|
138
|
|
|
|
|
|
|
} |
|
139
|
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
sub validate { |
|
141
|
0
|
|
|
0
|
0
|
|
my ( $self, $opt, $args ) = @_; |
|
142
|
|
|
|
|
|
|
|
|
143
|
0
|
0
|
|
|
|
|
if ( $opt->close ) { |
|
144
|
0
|
0
|
|
|
|
|
$self->usage_error('no time expression provided') unless @$args; |
|
145
|
|
|
|
|
|
|
} |
|
146
|
|
|
|
|
|
|
} |
|
147
|
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
1; |
|
149
|
|
|
|
|
|
|
|
|
150
|
|
|
|
|
|
|
__END__ |