| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
#================================= Base.pm =================================== |
|
2
|
|
|
|
|
|
|
# Filename: Base.pm |
|
3
|
|
|
|
|
|
|
# Description: Generalized hash by full path of file information. |
|
4
|
|
|
|
|
|
|
# Original Author: Dale M. Amon |
|
5
|
|
|
|
|
|
|
# Revised by: $Author: amon $ |
|
6
|
|
|
|
|
|
|
# Date: $Date: 2008-08-28 23:35:28 $ |
|
7
|
|
|
|
|
|
|
# Version: $Revision: 1.10 $ |
|
8
|
|
|
|
|
|
|
# License: LGPL 2.1, Perl Artistic or BSD |
|
9
|
|
|
|
|
|
|
# |
|
10
|
|
|
|
|
|
|
#============================================================================= |
|
11
|
1
|
|
|
1
|
|
2082
|
use strict; |
|
|
1
|
|
|
|
|
3
|
|
|
|
1
|
|
|
|
|
43
|
|
|
12
|
1
|
|
|
1
|
|
974
|
use Fault::Logger; |
|
|
1
|
|
|
|
|
36647
|
|
|
|
1
|
|
|
|
|
37
|
|
|
13
|
1
|
|
|
1
|
|
13
|
use Fault::DebugPrinter; |
|
|
1
|
|
|
|
|
9
|
|
|
|
1
|
|
|
|
|
24
|
|
|
14
|
1
|
|
|
1
|
|
5
|
use File::Spec; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
25
|
|
|
15
|
1
|
|
|
1
|
|
864
|
use FileHash::Entry; |
|
|
1
|
|
|
|
|
4
|
|
|
|
1
|
|
|
|
|
33
|
|
|
16
|
1
|
|
|
1
|
|
10
|
use FileHash::FormatString; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
20
|
|
|
17
|
1
|
|
|
1
|
|
5
|
use Cwd qw(abs_path); |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
66
|
|
|
18
|
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
package FileHash::Base; |
|
20
|
1
|
|
|
1
|
|
5
|
use vars qw{@ISA}; |
|
|
1
|
|
|
|
|
9
|
|
|
|
1
|
|
|
|
|
2835
|
|
|
21
|
|
|
|
|
|
|
@ISA = qw( UNIVERSAL ); |
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
#============================================================================= |
|
24
|
|
|
|
|
|
|
# INTERNAL OPS |
|
25
|
|
|
|
|
|
|
#============================================================================= |
|
26
|
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
sub _leaf { |
|
28
|
0
|
|
|
0
|
|
|
my ($self,$dev,$directory,$file) = @_; |
|
29
|
0
|
|
|
|
|
|
my $entry = FileHash::Entry->alloc; |
|
30
|
0
|
|
|
|
|
|
my $path = File::Spec->catfile($dev,$directory,$file); |
|
31
|
|
|
|
|
|
|
|
|
32
|
0
|
|
|
|
|
|
return $entry->initFromStat ($path); |
|
33
|
|
|
|
|
|
|
} |
|
34
|
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
36
|
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
sub _branch { |
|
38
|
0
|
|
|
0
|
|
|
my ($self,$dev,$directory,$file,$depth) = @_; |
|
39
|
0
|
|
|
|
|
|
my $path = File::Spec->catfile($dev,$directory,$file); |
|
40
|
|
|
|
|
|
|
|
|
41
|
0
|
0
|
0
|
|
|
|
if (-d "$path" and ! -l "$path") { |
|
42
|
0
|
|
|
|
|
|
my ($fh,$new); |
|
43
|
|
|
|
|
|
|
|
|
44
|
0
|
0
|
|
|
|
|
Fault::Logger->assertion_check |
|
45
|
|
|
|
|
|
|
(!(opendir $fh, "$path"),undef,"Can not open '$path': $!") |
|
46
|
|
|
|
|
|
|
or return $self; |
|
47
|
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
# Readdir returns a null list if there are no files. |
|
49
|
0
|
|
|
|
|
|
$depth++; |
|
50
|
0
|
|
|
|
|
|
my @dirlist = readdir ($fh); |
|
51
|
0
|
|
|
|
|
|
closedir $fh; |
|
52
|
|
|
|
|
|
|
|
|
53
|
0
|
|
|
|
|
|
foreach $new (@dirlist) { |
|
54
|
0
|
0
|
|
|
|
|
next if ($new eq "."); |
|
55
|
0
|
0
|
|
|
|
|
next if ($new eq ".."); |
|
56
|
0
|
|
|
|
|
|
$self->_branch ($dev,File::Spec->catfile($directory,$file),$new,$depth); |
|
57
|
|
|
|
|
|
|
} |
|
58
|
0
|
|
|
|
|
|
$depth--; |
|
59
|
|
|
|
|
|
|
} |
|
60
|
|
|
|
|
|
|
else { |
|
61
|
0
|
|
|
|
|
|
my $entry = $self->_leaf ($dev,$directory,$file); |
|
62
|
0
|
0
|
|
|
|
|
defined $entry or return undef; |
|
63
|
|
|
|
|
|
|
|
|
64
|
0
|
|
|
|
|
|
$self->_store ($self->_genKey ($entry),$entry); |
|
65
|
|
|
|
|
|
|
} |
|
66
|
0
|
|
|
|
|
|
return $self; |
|
67
|
|
|
|
|
|
|
} |
|
68
|
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
#============================================================================= |
|
70
|
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
sub _store { |
|
72
|
0
|
|
|
0
|
|
|
my ($self,$key,$val) = @_; |
|
73
|
|
|
|
|
|
|
|
|
74
|
0
|
0
|
|
|
|
|
if (! exists $self->{'filehash'}->{$key}) { |
|
75
|
0
|
|
|
|
|
|
$self->{'filehash'}->{$key} = [$val]; |
|
76
|
0
|
|
|
|
|
|
Fault::DebugPrinter->dbg (2,"NEW KEY <$key>"); |
|
77
|
|
|
|
|
|
|
} |
|
78
|
|
|
|
|
|
|
else { |
|
79
|
0
|
|
|
|
|
|
Fault::DebugPrinter->dbg (2,"DUPLICATE KEY <$key>"); |
|
80
|
0
|
|
|
|
|
|
push @{$self->{'filehash'}->{$key}}, $val; |
|
|
0
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
} |
|
82
|
0
|
|
|
|
|
|
my $path = $val->path; |
|
83
|
0
|
|
|
|
|
|
Fault::DebugPrinter->dbg (3," FILE <$path>"); |
|
84
|
0
|
|
|
|
|
|
return $self; |
|
85
|
|
|
|
|
|
|
} |
|
86
|
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
88
|
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
sub _genKey |
|
90
|
0
|
|
|
0
|
|
|
{Fault::Logger->crash ("Subclass must impliment: _genKey");} |
|
91
|
|
|
|
|
|
|
|
|
92
|
|
|
|
|
|
|
#============================================================================= |
|
93
|
|
|
|
|
|
|
# CLASS METHODS |
|
94
|
|
|
|
|
|
|
#============================================================================= |
|
95
|
|
|
|
|
|
|
|
|
96
|
|
|
|
|
|
|
sub alloc ($) { |
|
97
|
0
|
|
|
0
|
1
|
|
my ($class) = @_; |
|
98
|
0
|
|
|
|
|
|
my $self = bless {}, $class; |
|
99
|
0
|
|
|
|
|
|
$self->{'filehash'} = {}; |
|
100
|
0
|
|
|
|
|
|
return $self; |
|
101
|
|
|
|
|
|
|
} |
|
102
|
|
|
|
|
|
|
|
|
103
|
|
|
|
|
|
|
#============================================================================= |
|
104
|
|
|
|
|
|
|
# INSTANCE METHODS |
|
105
|
|
|
|
|
|
|
#============================================================================= |
|
106
|
|
|
|
|
|
|
# Init methods |
|
107
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
108
|
0
|
|
|
0
|
1
|
|
sub init ($) {shift;} |
|
109
|
0
|
|
|
0
|
1
|
|
sub initFromLines ($$@) {shift->addFromLines (@_);} |
|
110
|
0
|
|
|
0
|
1
|
|
sub initFromFile ($$$) {shift->addFromFile (@_);} |
|
111
|
0
|
|
|
0
|
1
|
|
sub initFromTree ($$) {shift->addFromTree (@_);} |
|
112
|
0
|
|
|
0
|
1
|
|
sub initFromObject ($$) {shift->addFromObject (@_);} |
|
113
|
0
|
|
|
0
|
1
|
|
sub initFromDump ($$) {shift->addFromDump (@_);} |
|
114
|
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
#============================================================================= |
|
116
|
|
|
|
|
|
|
# add methods |
|
117
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
118
|
|
|
|
|
|
|
# The FileHash::Entry object will check for invalid values in @lines. |
|
119
|
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
sub addFromLines ($$@) { |
|
121
|
0
|
|
|
0
|
1
|
|
my ($self,$formatline,@lines) = @_; |
|
122
|
|
|
|
|
|
|
|
|
123
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
124
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_noref ($formatline,"formatline") or return undef; |
|
125
|
|
|
|
|
|
|
} |
|
126
|
|
|
|
|
|
|
|
|
127
|
0
|
|
|
|
|
|
my $fmt = FileHash::FormatString->alloc; |
|
128
|
0
|
0
|
|
|
|
|
$fmt->init ($formatline) or return undef; |
|
129
|
|
|
|
|
|
|
|
|
130
|
0
|
|
|
|
|
|
$self->{'format'} = $fmt; |
|
131
|
0
|
|
|
|
|
|
foreach (@lines) { |
|
132
|
0
|
0
|
|
|
|
|
next if (/^\w*$/); |
|
133
|
|
|
|
|
|
|
|
|
134
|
0
|
|
|
|
|
|
my $entry = FileHash::Entry->alloc; |
|
135
|
0
|
0
|
|
|
|
|
$entry->initFromLine ($self->{'format'},$_) or next; |
|
136
|
|
|
|
|
|
|
|
|
137
|
0
|
|
|
|
|
|
$self->_store ($self->_genKey ($entry),$entry); |
|
138
|
|
|
|
|
|
|
} |
|
139
|
0
|
|
|
|
|
|
return $self; |
|
140
|
|
|
|
|
|
|
} |
|
141
|
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
143
|
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
sub addFromFile ($$$) { |
|
145
|
0
|
|
|
0
|
1
|
|
my ($self,$formatline,$listfname) = @_; |
|
146
|
0
|
|
|
|
|
|
my $fh; |
|
147
|
|
|
|
|
|
|
|
|
148
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
149
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_noref ($formatline,"formatline") or return undef; |
|
150
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_noref ($listfname, "listfname" ) or return undef; |
|
151
|
|
|
|
|
|
|
} |
|
152
|
|
|
|
|
|
|
|
|
153
|
|
|
|
|
|
|
Fault::Logger->assertion_check |
|
154
|
0
|
0
|
|
|
|
|
(!(open $fh, "<$listfname"),undef,"Can not open '$listfname': $!") |
|
155
|
|
|
|
|
|
|
or return undef; |
|
156
|
|
|
|
|
|
|
|
|
157
|
|
|
|
|
|
|
# Readline returns a null list if there are no lines. |
|
158
|
0
|
|
|
|
|
|
my @lines = readline $fh; |
|
159
|
|
|
|
|
|
|
|
|
160
|
0
|
|
|
|
|
|
return $self->addFromLines ($formatline,@lines); |
|
161
|
|
|
|
|
|
|
} |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
164
|
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
sub addFromTree ($$) { |
|
166
|
0
|
|
|
0
|
1
|
|
my ($self,$path) = @_; |
|
167
|
|
|
|
|
|
|
|
|
168
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
169
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_noref ($path,"path") or return undef; |
|
170
|
|
|
|
|
|
|
} |
|
171
|
|
|
|
|
|
|
|
|
172
|
0
|
|
|
|
|
|
my ($dev,$directories,$file) = |
|
173
|
|
|
|
|
|
|
File::Spec->splitpath(Cwd::abs_path($path)); |
|
174
|
|
|
|
|
|
|
|
|
175
|
0
|
|
|
|
|
|
$self->_branch ($dev,$directories,$file, 0); |
|
176
|
0
|
|
|
|
|
|
return $self; |
|
177
|
|
|
|
|
|
|
} |
|
178
|
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
sub addFromObject ($$) { |
|
182
|
0
|
|
|
0
|
1
|
|
my ($self,$old) = @_; |
|
183
|
|
|
|
|
|
|
|
|
184
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
185
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_isa ($old,"FileHash::Base","oldfilehash") |
|
186
|
|
|
|
|
|
|
or return undef; |
|
187
|
|
|
|
|
|
|
} |
|
188
|
|
|
|
|
|
|
|
|
189
|
0
|
|
|
|
|
|
foreach my $i (values %{$old->{'filehash'}}) { |
|
|
0
|
|
|
|
|
|
|
|
190
|
0
|
|
|
|
|
|
foreach my $j (@$i) { |
|
191
|
0
|
|
|
|
|
|
$self->_store ($self->_genKey ($j),$j); |
|
192
|
|
|
|
|
|
|
} |
|
193
|
|
|
|
|
|
|
} |
|
194
|
0
|
|
|
|
|
|
return $self; |
|
195
|
|
|
|
|
|
|
} |
|
196
|
|
|
|
|
|
|
|
|
197
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
198
|
|
|
|
|
|
|
|
|
199
|
|
|
|
|
|
|
sub addFromDump ($$) { |
|
200
|
0
|
|
|
0
|
1
|
|
my ($self,$path) = @_; |
|
201
|
0
|
|
|
|
|
|
my ($fh,$entry,$s,$vers) = undef; |
|
202
|
|
|
|
|
|
|
|
|
203
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
204
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_noref ($path,"path") or return undef; |
|
205
|
|
|
|
|
|
|
} |
|
206
|
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
Fault::Logger->assertion_check |
|
208
|
0
|
0
|
|
|
|
|
(!(open $fh, "<$path"),undef,"Can not open '$path': $!") |
|
209
|
|
|
|
|
|
|
or return undef; |
|
210
|
|
|
|
|
|
|
|
|
211
|
0
|
|
|
|
|
|
$_ = (<$fh>); |
|
212
|
0
|
0
|
|
|
|
|
if (/^Version:/) {($s,$vers) = split;} |
|
|
0
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
else { |
|
214
|
0
|
|
|
|
|
|
Fault::Logger->log |
|
215
|
|
|
|
|
|
|
("No Format version number found in report file header: '$path'"); |
|
216
|
0
|
|
|
|
|
|
close $fh; |
|
217
|
0
|
|
|
|
|
|
return undef; |
|
218
|
|
|
|
|
|
|
} |
|
219
|
|
|
|
|
|
|
|
|
220
|
0
|
|
|
|
|
|
while (1) { |
|
221
|
0
|
|
|
|
|
|
my $entry = FileHash::Entry->alloc; |
|
222
|
0
|
0
|
|
|
|
|
$entry->addFromDump($fh) or last; |
|
223
|
|
|
|
|
|
|
|
|
224
|
0
|
|
|
|
|
|
$self->_store ($self->_genKey ($entry),$entry); |
|
225
|
|
|
|
|
|
|
} |
|
226
|
0
|
|
|
|
|
|
close $fh; |
|
227
|
0
|
|
|
|
|
|
return $self; |
|
228
|
|
|
|
|
|
|
} |
|
229
|
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
#============================================================================= |
|
231
|
|
|
|
|
|
|
# Unary operators |
|
232
|
|
|
|
|
|
|
#============================================================================= |
|
233
|
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
sub identical ($) { |
|
235
|
0
|
|
|
0
|
1
|
|
my ($fha) = @_; |
|
236
|
0
|
|
|
|
|
|
my $class = ref $fha; |
|
237
|
0
|
|
|
|
|
|
my ($j,$k,$v1); |
|
238
|
|
|
|
|
|
|
|
|
239
|
0
|
|
|
|
|
|
my $fhb = $class->alloc; |
|
240
|
0
|
0
|
|
|
|
|
$fhb->init or return undef; |
|
241
|
|
|
|
|
|
|
|
|
242
|
0
|
|
|
|
|
|
my $a = $fha->{'filehash'}; |
|
243
|
0
|
|
|
|
|
|
foreach $v1 (values %$a) { |
|
244
|
0
|
0
|
|
|
|
|
if ($#$v1 > 0) { |
|
245
|
0
|
|
|
|
|
|
foreach $j (@$v1) {$fhb->_store ($fhb->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
246
|
|
|
|
|
|
|
} |
|
247
|
|
|
|
|
|
|
} |
|
248
|
0
|
|
|
|
|
|
return $fhb; |
|
249
|
|
|
|
|
|
|
} |
|
250
|
|
|
|
|
|
|
|
|
251
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
252
|
|
|
|
|
|
|
|
|
253
|
|
|
|
|
|
|
sub unique ($) { |
|
254
|
0
|
|
|
0
|
1
|
|
my ($fha) = @_; |
|
255
|
0
|
|
|
|
|
|
my $class = ref $fha; |
|
256
|
0
|
|
|
|
|
|
my ($j,$k,$v1); |
|
257
|
|
|
|
|
|
|
|
|
258
|
0
|
|
|
|
|
|
my $fhb = $class->alloc; |
|
259
|
0
|
0
|
|
|
|
|
$fhb->init or return undef; |
|
260
|
|
|
|
|
|
|
|
|
261
|
0
|
|
|
|
|
|
my $a = $fha->{'filehash'}; |
|
262
|
0
|
|
|
|
|
|
foreach $v1 (values %$a) { |
|
263
|
0
|
0
|
|
|
|
|
if ($#$v1 == 0) { |
|
264
|
0
|
|
|
|
|
|
foreach $j (@$v1) {$fhb->_store ($fhb->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
265
|
|
|
|
|
|
|
} |
|
266
|
|
|
|
|
|
|
} |
|
267
|
0
|
|
|
|
|
|
return $fhb; |
|
268
|
|
|
|
|
|
|
} |
|
269
|
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
#============================================================================= |
|
271
|
|
|
|
|
|
|
# Binary operators |
|
272
|
|
|
|
|
|
|
#============================================================================= |
|
273
|
|
|
|
|
|
|
|
|
274
|
|
|
|
|
|
|
sub xor ($$) { |
|
275
|
0
|
|
|
0
|
1
|
|
my ($fha,$fhb) = @_; |
|
276
|
0
|
|
|
|
|
|
my $class = ref $fha; |
|
277
|
0
|
|
|
|
|
|
my ($j, $k,$v1,$v2); |
|
278
|
|
|
|
|
|
|
|
|
279
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
280
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_isa ($fhb,$class,"fhb") or return undef; |
|
281
|
|
|
|
|
|
|
} |
|
282
|
|
|
|
|
|
|
|
|
283
|
0
|
|
|
|
|
|
my $fhc = $class->alloc; |
|
284
|
0
|
0
|
|
|
|
|
$fhc->init or return undef; |
|
285
|
|
|
|
|
|
|
|
|
286
|
0
|
|
|
|
|
|
my ($a,$b) = ($fha->{'filehash'},$fhb->{'filehash'}); |
|
287
|
0
|
|
|
|
|
|
while (($k,$v1) = each %$a) { |
|
288
|
0
|
0
|
|
|
|
|
if (!exists $b->{$k}) { |
|
289
|
0
|
|
|
|
|
|
foreach $j (@$v1) {$fhc->_store ($fhc->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
290
|
|
|
|
|
|
|
} |
|
291
|
|
|
|
|
|
|
} |
|
292
|
0
|
|
|
|
|
|
while (($k,$v2) = each %$b) { |
|
293
|
0
|
0
|
|
|
|
|
if (!exists $a->{$k}) { |
|
294
|
0
|
|
|
|
|
|
foreach $j (@$v2) {$fhc->_store ($fhc->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
295
|
|
|
|
|
|
|
} |
|
296
|
|
|
|
|
|
|
} |
|
297
|
0
|
|
|
|
|
|
return $fhc; |
|
298
|
|
|
|
|
|
|
} |
|
299
|
|
|
|
|
|
|
|
|
300
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
301
|
|
|
|
|
|
|
|
|
302
|
|
|
|
|
|
|
sub and ($$) { |
|
303
|
0
|
|
|
0
|
1
|
|
my ($fha,$fhb) = @_; |
|
304
|
0
|
|
|
|
|
|
my $class = ref $fha; |
|
305
|
0
|
|
|
|
|
|
my ($j,$k,$v1,$v2); |
|
306
|
|
|
|
|
|
|
|
|
307
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
308
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_isa ($fhb,$class,"fhb") or return undef; |
|
309
|
|
|
|
|
|
|
} |
|
310
|
|
|
|
|
|
|
|
|
311
|
0
|
|
|
|
|
|
my $fhc = $class->alloc; |
|
312
|
0
|
0
|
|
|
|
|
$fhc->init or return undef; |
|
313
|
|
|
|
|
|
|
|
|
314
|
0
|
|
|
|
|
|
my ($a,$b) = ($fha->{'filehash'},$fhb->{'filehash'}); |
|
315
|
0
|
|
|
|
|
|
while (($k,$v1) = each %$a) { |
|
316
|
0
|
0
|
|
|
|
|
if (exists $b->{$k}) { |
|
317
|
0
|
|
|
|
|
|
my $v2 = $b->{$k}; |
|
318
|
0
|
|
|
|
|
|
foreach $j (@$v1) {$fhc->_store ($fhc->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
319
|
0
|
|
|
|
|
|
foreach $j (@$v2) {$fhc->_store ($fhc->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
320
|
|
|
|
|
|
|
} |
|
321
|
|
|
|
|
|
|
} |
|
322
|
0
|
|
|
|
|
|
return $fhc; |
|
323
|
|
|
|
|
|
|
} |
|
324
|
|
|
|
|
|
|
|
|
325
|
|
|
|
|
|
|
#----------------------------------------------------------------------------- |
|
326
|
|
|
|
|
|
|
|
|
327
|
|
|
|
|
|
|
sub andnot ($$) { |
|
328
|
0
|
|
|
0
|
1
|
|
my ($fha,$fhb) = @_; |
|
329
|
0
|
|
|
|
|
|
my $class = ref $fha; |
|
330
|
0
|
|
|
|
|
|
my ($j,$k,$v1,$v2); |
|
331
|
|
|
|
|
|
|
|
|
332
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
333
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_isa ($fhb,$class,"fhb") or return undef; |
|
334
|
|
|
|
|
|
|
} |
|
335
|
|
|
|
|
|
|
|
|
336
|
0
|
|
|
|
|
|
my $fhc = $class->alloc; |
|
337
|
0
|
0
|
|
|
|
|
$fhc->init or return undef; |
|
338
|
|
|
|
|
|
|
|
|
339
|
0
|
|
|
|
|
|
my ($a,$b) = ($fha->{'filehash'},$fhb->{'filehash'}); |
|
340
|
0
|
|
|
|
|
|
while (($k,$v1) = each %$a) { |
|
341
|
0
|
0
|
|
|
|
|
if (!exists $b->{$k}) { |
|
342
|
0
|
|
|
|
|
|
foreach $j (@$v1) {$fhc->_store ($fhc->_genKey ($j),$j);} |
|
|
0
|
|
|
|
|
|
|
|
343
|
|
|
|
|
|
|
} |
|
344
|
|
|
|
|
|
|
} |
|
345
|
0
|
|
|
|
|
|
return $fhc; |
|
346
|
|
|
|
|
|
|
} |
|
347
|
|
|
|
|
|
|
|
|
348
|
|
|
|
|
|
|
#============================================================================= |
|
349
|
|
|
|
|
|
|
|
|
350
|
|
|
|
|
|
|
sub dump ($$) { |
|
351
|
0
|
|
|
0
|
1
|
|
my ($self,$dumpfile) = @_; |
|
352
|
0
|
|
|
|
|
|
my $fh; |
|
353
|
|
|
|
|
|
|
|
|
354
|
0
|
0
|
|
|
|
|
if ($::DEBUG) { |
|
355
|
0
|
0
|
|
|
|
|
Fault::Logger->arg_check_noref ($dumpfile,"dumpfile") or return undef; |
|
356
|
|
|
|
|
|
|
} |
|
357
|
|
|
|
|
|
|
|
|
358
|
|
|
|
|
|
|
Fault::Logger->assertion_check |
|
359
|
0
|
0
|
|
|
|
|
(!(open $fh, ">$dumpfile"),undef,"Can not open '$dumpfile': $!") |
|
360
|
|
|
|
|
|
|
or return undef; |
|
361
|
|
|
|
|
|
|
|
|
362
|
0
|
0
|
|
|
|
|
if (!printf $fh "Version: " . FileHash::Entry->dumpversion . "\n") { |
|
363
|
0
|
|
|
|
|
|
Fault::Logger->log ("Failed to print dumpfile header: $!"); |
|
364
|
0
|
|
|
|
|
|
close $fh; |
|
365
|
0
|
|
|
|
|
|
return undef; |
|
366
|
|
|
|
|
|
|
} |
|
367
|
|
|
|
|
|
|
|
|
368
|
0
|
|
|
|
|
|
foreach my $i (values %{$self->{'filehash'}}) { |
|
|
0
|
|
|
|
|
|
|
|
369
|
0
|
|
|
|
|
|
foreach my $j (@$i) {$j->fprint ($fh);} |
|
|
0
|
|
|
|
|
|
|
|
370
|
|
|
|
|
|
|
} |
|
371
|
|
|
|
|
|
|
|
|
372
|
0
|
|
|
|
|
|
close $fh; |
|
373
|
0
|
|
|
|
|
|
return $self; |
|
374
|
|
|
|
|
|
|
} |
|
375
|
|
|
|
|
|
|
|
|
376
|
|
|
|
|
|
|
#============================================================================= |
|
377
|
|
|
|
|
|
|
# POD DOCUMENTATION |
|
378
|
|
|
|
|
|
|
#============================================================================= |
|
379
|
|
|
|
|
|
|
# You may extract and format the documention section with the 'perldoc' cmd. |
|
380
|
|
|
|
|
|
|
|
|
381
|
|
|
|
|
|
|
=head1 NAME |
|
382
|
|
|
|
|
|
|
|
|
383
|
|
|
|
|
|
|
FileHash::Base - Abstract superclass for FileHashes. |
|
384
|
|
|
|
|
|
|
|
|
385
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
386
|
|
|
|
|
|
|
|
|
387
|
|
|
|
|
|
|
use FileHash; |
|
388
|
|
|
|
|
|
|
$obj = FileHash::Base->alloc; |
|
389
|
|
|
|
|
|
|
|
|
390
|
|
|
|
|
|
|
$obj = $obj->init; |
|
391
|
|
|
|
|
|
|
$obj = $obj->initFromLines ($formatline,@lines); |
|
392
|
|
|
|
|
|
|
$obj = $obj->initFromFile ($formatline,$datafilepath); |
|
393
|
|
|
|
|
|
|
$obj = $obj->initFromTree ($rootdir); |
|
394
|
|
|
|
|
|
|
$obj = $obj->initFromObject ($obj2); |
|
395
|
|
|
|
|
|
|
$obj = $obj->initFromDump ($path); |
|
396
|
|
|
|
|
|
|
|
|
397
|
|
|
|
|
|
|
$obj = $obj->addFromLines ($formatline,@lines); |
|
398
|
|
|
|
|
|
|
$obj = $obj->addFromFile ($formatline,$datafilepath); |
|
399
|
|
|
|
|
|
|
$obj = $obj->addFromTree ($rootdir); |
|
400
|
|
|
|
|
|
|
$obj = $obj->addFromObject ($obj2); |
|
401
|
|
|
|
|
|
|
$obj = $obj->addFromDump ($path); |
|
402
|
|
|
|
|
|
|
|
|
403
|
|
|
|
|
|
|
$fhb = $fha->identical; |
|
404
|
|
|
|
|
|
|
$fhb = $fha->unique; |
|
405
|
|
|
|
|
|
|
|
|
406
|
|
|
|
|
|
|
$fhc = $fha->and ($fhb); |
|
407
|
|
|
|
|
|
|
$fhc = $fha->andnot ($fhb); |
|
408
|
|
|
|
|
|
|
$fhc = $fha->xor ($fhb); |
|
409
|
|
|
|
|
|
|
|
|
410
|
|
|
|
|
|
|
$obj = $obj->dump ($dumpfile); |
|
411
|
|
|
|
|
|
|
|
|
412
|
|
|
|
|
|
|
=head1 Inheritance |
|
413
|
|
|
|
|
|
|
|
|
414
|
|
|
|
|
|
|
UNIVERSAL |
|
415
|
|
|
|
|
|
|
|
|
416
|
|
|
|
|
|
|
=head1 Description |
|
417
|
|
|
|
|
|
|
|
|
418
|
|
|
|
|
|
|
This is an abstract superclass for containers of lists of file metadata. |
|
419
|
|
|
|
|
|
|
It is not directly useable and will execute a Fault if you attempt it. |
|
420
|
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
FileHash::Name and FileHash::Content inherit most of their behavior from |
|
422
|
|
|
|
|
|
|
here with the exception of hash key selection. |
|
423
|
|
|
|
|
|
|
|
|
424
|
|
|
|
|
|
|
=head1 Examples |
|
425
|
|
|
|
|
|
|
|
|
426
|
|
|
|
|
|
|
See subclasses. |
|
427
|
|
|
|
|
|
|
|
|
428
|
|
|
|
|
|
|
=head1 Class Variables |
|
429
|
|
|
|
|
|
|
|
|
430
|
|
|
|
|
|
|
None. |
|
431
|
|
|
|
|
|
|
|
|
432
|
|
|
|
|
|
|
=head1 Instance Variables |
|
433
|
|
|
|
|
|
|
|
|
434
|
|
|
|
|
|
|
filehash Pointer to a hash of arrays of FileHash::Entry objects |
|
435
|
|
|
|
|
|
|
which contain all the file metadata discovered found |
|
436
|
|
|
|
|
|
|
when the FileHash object was initialized. Entries for files |
|
437
|
|
|
|
|
|
|
with identical keys hash into the same array, making for |
|
438
|
|
|
|
|
|
|
a very efficient sort. |
|
439
|
|
|
|
|
|
|
|
|
440
|
|
|
|
|
|
|
=head1 Class Methods |
|
441
|
|
|
|
|
|
|
|
|
442
|
|
|
|
|
|
|
=over 4 |
|
443
|
|
|
|
|
|
|
|
|
444
|
|
|
|
|
|
|
=item B<$obj = FileHash::Base-Ealloc> |
|
445
|
|
|
|
|
|
|
|
|
446
|
|
|
|
|
|
|
Allocate an empty instance of FileHash::Base. This is for inheritance |
|
447
|
|
|
|
|
|
|
only and should not be used. Subclasses could override but there is |
|
448
|
|
|
|
|
|
|
probably no reason to do so unless they add ivars. None do at present. |
|
449
|
|
|
|
|
|
|
|
|
450
|
|
|
|
|
|
|
=head1 Instance Methods |
|
451
|
|
|
|
|
|
|
|
|
452
|
|
|
|
|
|
|
Unless otherwise specified, instance methods return self on success and |
|
453
|
|
|
|
|
|
|
undef on failure. |
|
454
|
|
|
|
|
|
|
|
|
455
|
|
|
|
|
|
|
=over 4 |
|
456
|
|
|
|
|
|
|
|
|
457
|
|
|
|
|
|
|
=item B<$fhc = $fha-Eand ($fhb)> |
|
458
|
|
|
|
|
|
|
|
|
459
|
|
|
|
|
|
|
Create a file hash containing the groups of files found in both |
|
460
|
|
|
|
|
|
|
filehash a and b. |
|
461
|
|
|
|
|
|
|
|
|
462
|
|
|
|
|
|
|
a and b must be of the same FileHash subclass and the newly created |
|
463
|
|
|
|
|
|
|
c will be off that type also. |
|
464
|
|
|
|
|
|
|
|
|
465
|
|
|
|
|
|
|
=item B<$fhc = $fha-Eandnot ($fhb)> |
|
466
|
|
|
|
|
|
|
|
|
467
|
|
|
|
|
|
|
Create a file hash containing the groups of files found in filehash |
|
468
|
|
|
|
|
|
|
a but not in filehash b. |
|
469
|
|
|
|
|
|
|
|
|
470
|
|
|
|
|
|
|
a and b must be of the same FileHash subclass and the newly created |
|
471
|
|
|
|
|
|
|
c will be off that type also. |
|
472
|
|
|
|
|
|
|
|
|
473
|
|
|
|
|
|
|
If you want not a and b, just reverse the args; not a and not b is |
|
474
|
|
|
|
|
|
|
obviously nonsensical as we are testing keys of a against keys of |
|
475
|
|
|
|
|
|
|
b. |
|
476
|
|
|
|
|
|
|
|
|
477
|
|
|
|
|
|
|
=item B<$obj = $obj-EaddFromDump ($dumpfile)> |
|
478
|
|
|
|
|
|
|
|
|
479
|
|
|
|
|
|
|
Use a dump file to recreate hash entries and add them to a FileHash |
|
480
|
|
|
|
|
|
|
object. |
|
481
|
|
|
|
|
|
|
|
|
482
|
|
|
|
|
|
|
The first line of the file must contain the text: |
|
483
|
|
|
|
|
|
|
|
|
484
|
|
|
|
|
|
|
Version: x.yy |
|
485
|
|
|
|
|
|
|
|
|
486
|
|
|
|
|
|
|
=item B<$obj = $obj-EaddFromFile ($format,$datafilepath)> |
|
487
|
|
|
|
|
|
|
|
|
488
|
|
|
|
|
|
|
Use the format line to create a FileHash::FormatString object. The |
|
489
|
|
|
|
|
|
|
format object is used to parse each of the lines in a file which |
|
490
|
|
|
|
|
|
|
contains lines of text data. Each line in the file is assumed to |
|
491
|
|
|
|
|
|
|
contain data about one file which is to be added to the FileHash. |
|
492
|
|
|
|
|
|
|
|
|
493
|
|
|
|
|
|
|
=item B<$obj = $obj-EaddFromLines ($format,@lines)> |
|
494
|
|
|
|
|
|
|
|
|
495
|
|
|
|
|
|
|
Use the format line to create a FileHash:FormatString object. The |
|
496
|
|
|
|
|
|
|
format object is used to parse each of the lines in a list. Each |
|
497
|
|
|
|
|
|
|
line contains data about one file which is to be added to the FileHash. |
|
498
|
|
|
|
|
|
|
|
|
499
|
|
|
|
|
|
|
=item B<$obj = $obj-EaddFromObject ($obj2)> |
|
500
|
|
|
|
|
|
|
|
|
501
|
|
|
|
|
|
|
Add data to a filehash from another FileHash, $obj2. This is useful for |
|
502
|
|
|
|
|
|
|
merging two objects. The subclasses need not be the same because the |
|
503
|
|
|
|
|
|
|
Entries are inserted by re-hashing into the target object. |
|
504
|
|
|
|
|
|
|
|
|
505
|
|
|
|
|
|
|
my $a = FileHash::Name->alloc; |
|
506
|
|
|
|
|
|
|
$a->initFromTree ("/root"); |
|
507
|
|
|
|
|
|
|
$a->addFromTree ("/home/me"); |
|
508
|
|
|
|
|
|
|
|
|
509
|
|
|
|
|
|
|
=item B<$obj = $obj-EaddFromTree ($rootdir)> |
|
510
|
|
|
|
|
|
|
|
|
511
|
|
|
|
|
|
|
Entries are added to the hash via a recursive descent through a directory |
|
512
|
|
|
|
|
|
|
tree. Each file is a 'leaf node' and is represented by an array record in |
|
513
|
|
|
|
|
|
|
the hash. If two files have the same hash key, the are likely identical |
|
514
|
|
|
|
|
|
|
so the records for them are placed together in an array under that hask key. |
|
515
|
|
|
|
|
|
|
|
|
516
|
|
|
|
|
|
|
=item B<$obj = $obj-Edump ($dumpfile)> |
|
517
|
|
|
|
|
|
|
|
|
518
|
|
|
|
|
|
|
Dump FileHash::Entry objects sequentially, one to a line, to the specified |
|
519
|
|
|
|
|
|
|
filename. |
|
520
|
|
|
|
|
|
|
|
|
521
|
|
|
|
|
|
|
The first line of the file contains the FileHash::Entry dump file format |
|
522
|
|
|
|
|
|
|
version number in this format: |
|
523
|
|
|
|
|
|
|
|
|
524
|
|
|
|
|
|
|
Version: x.yy |
|
525
|
|
|
|
|
|
|
|
|
526
|
|
|
|
|
|
|
=item B<$fhb = $fha-Eidentical> |
|
527
|
|
|
|
|
|
|
|
|
528
|
|
|
|
|
|
|
Return a FileHash containing the contents of hash keys which have more |
|
529
|
|
|
|
|
|
|
than one member. If the keys are md5,length this represents all files |
|
530
|
|
|
|
|
|
|
with the same content; if they keys are name it represents all files |
|
531
|
|
|
|
|
|
|
with the same name. |
|
532
|
|
|
|
|
|
|
|
|
533
|
|
|
|
|
|
|
=item B<$obj = $obj-Einit> |
|
534
|
|
|
|
|
|
|
|
|
535
|
|
|
|
|
|
|
A noop at present. If you need an empty object, use this after alloc |
|
536
|
|
|
|
|
|
|
to make sure that if init is needed in the future, it will be carried |
|
537
|
|
|
|
|
|
|
out. |
|
538
|
|
|
|
|
|
|
|
|
539
|
|
|
|
|
|
|
=item B<$obj = $obj-EinitFromDump ($dumpfile)> |
|
540
|
|
|
|
|
|
|
|
|
541
|
|
|
|
|
|
|
Use a dump file to recreate hash entries in a freshly alloc'd FileHash |
|
542
|
|
|
|
|
|
|
object. |
|
543
|
|
|
|
|
|
|
|
|
544
|
|
|
|
|
|
|
The first line of the file must contain the text: |
|
545
|
|
|
|
|
|
|
|
|
546
|
|
|
|
|
|
|
Version: x.yy |
|
547
|
|
|
|
|
|
|
|
|
548
|
|
|
|
|
|
|
=item B<$obj = $obj-EinitFromFile ($format,$datafilepath)> |
|
549
|
|
|
|
|
|
|
|
|
550
|
|
|
|
|
|
|
Initialize a freshly alloc'd FileHash. It uses the format line |
|
551
|
|
|
|
|
|
|
to init a FileHash::FormatString object. The format object is used to parse |
|
552
|
|
|
|
|
|
|
each of the lines in a file which contains lines of text data. Each line |
|
553
|
|
|
|
|
|
|
in the file is assumed to contain data about one file. |
|
554
|
|
|
|
|
|
|
|
|
555
|
|
|
|
|
|
|
=item B<$obj = $obj-EinitFromLines ($format,@lines)> |
|
556
|
|
|
|
|
|
|
|
|
557
|
|
|
|
|
|
|
Initialize a freshly formatted FileHash. It uses the format line |
|
558
|
|
|
|
|
|
|
to create a FileHash:FormatString object. The format object is used to parse |
|
559
|
|
|
|
|
|
|
each of the lines in a list. Each line contains data about one file. |
|
560
|
|
|
|
|
|
|
|
|
561
|
|
|
|
|
|
|
=item B<$obj = $obj-EinitFromObject ($obj2)> |
|
562
|
|
|
|
|
|
|
|
|
563
|
|
|
|
|
|
|
Initialize the newly alloc'd object using data from another FileHash, $obj2. |
|
564
|
|
|
|
|
|
|
This is useful for changing from hashing by name to hashing by content or |
|
565
|
|
|
|
|
|
|
vice versa: |
|
566
|
|
|
|
|
|
|
|
|
567
|
|
|
|
|
|
|
my $a = FileHash::Name->alloc; |
|
568
|
|
|
|
|
|
|
my $b = FileHash::Content->alloc; |
|
569
|
|
|
|
|
|
|
$a->initFromTree ("/root"); |
|
570
|
|
|
|
|
|
|
$b->initFromObject ($a); |
|
571
|
|
|
|
|
|
|
|
|
572
|
|
|
|
|
|
|
=item B<$obj = $obj-EinitFromTree ($rootdir)> |
|
573
|
|
|
|
|
|
|
|
|
574
|
|
|
|
|
|
|
Initialize a freshly alloc'd FileHash. The hash is filled via a |
|
575
|
|
|
|
|
|
|
recursive descent through a directory tree. Each file is a |
|
576
|
|
|
|
|
|
|
'leaf node' and is represented by an array record in the hash. If |
|
577
|
|
|
|
|
|
|
two files have the same hash key, the are likely identical so the records |
|
578
|
|
|
|
|
|
|
for them are placed together in an array under that hask key. |
|
579
|
|
|
|
|
|
|
|
|
580
|
|
|
|
|
|
|
=item B<$fhb = $fha-Eunique> |
|
581
|
|
|
|
|
|
|
|
|
582
|
|
|
|
|
|
|
Return a FileHash containing the contents of hash keys which have only |
|
583
|
|
|
|
|
|
|
one member. These are files for which no other file has the same content |
|
584
|
|
|
|
|
|
|
if the key is md5,length; or the same name if the key is the name. |
|
585
|
|
|
|
|
|
|
|
|
586
|
|
|
|
|
|
|
=item B<$fhc = $fha-Exor ($fhb)> |
|
587
|
|
|
|
|
|
|
|
|
588
|
|
|
|
|
|
|
Create a filehash c which contains all of the groups of files which |
|
589
|
|
|
|
|
|
|
are only in fha or fhb but not both. |
|
590
|
|
|
|
|
|
|
|
|
591
|
|
|
|
|
|
|
a and b must be of the same FileHash subclass and the newly created |
|
592
|
|
|
|
|
|
|
c will be off that type also. |
|
593
|
|
|
|
|
|
|
|
|
594
|
|
|
|
|
|
|
=back 4 |
|
595
|
|
|
|
|
|
|
|
|
596
|
|
|
|
|
|
|
=head1 Private Class Method |
|
597
|
|
|
|
|
|
|
|
|
598
|
|
|
|
|
|
|
None. |
|
599
|
|
|
|
|
|
|
|
|
600
|
|
|
|
|
|
|
=head1 Private Instance Methods |
|
601
|
|
|
|
|
|
|
|
|
602
|
|
|
|
|
|
|
=item B<$key = $obj-E_genKey($entry)> |
|
603
|
|
|
|
|
|
|
|
|
604
|
|
|
|
|
|
|
Create an appropriate hash key. Each subclass must override this |
|
605
|
|
|
|
|
|
|
stub method as it does nothing except print a warning message |
|
606
|
|
|
|
|
|
|
and crash the program. |
|
607
|
|
|
|
|
|
|
|
|
608
|
|
|
|
|
|
|
=head1 Errors and Warnings |
|
609
|
|
|
|
|
|
|
|
|
610
|
|
|
|
|
|
|
Lots. |
|
611
|
|
|
|
|
|
|
|
|
612
|
|
|
|
|
|
|
=head1 KNOWN BUGS |
|
613
|
|
|
|
|
|
|
|
|
614
|
|
|
|
|
|
|
See TODO. |
|
615
|
|
|
|
|
|
|
|
|
616
|
|
|
|
|
|
|
=head1 SEE ALSO |
|
617
|
|
|
|
|
|
|
|
|
618
|
|
|
|
|
|
|
File::Spec, Cwd, FileHash::Entry, FileHash::FormatString, Fault::Logger. |
|
619
|
|
|
|
|
|
|
|
|
620
|
|
|
|
|
|
|
=head1 AUTHOR |
|
621
|
|
|
|
|
|
|
|
|
622
|
|
|
|
|
|
|
Dale Amon |
|
623
|
|
|
|
|
|
|
|
|
624
|
|
|
|
|
|
|
=cut |
|
625
|
|
|
|
|
|
|
|
|
626
|
|
|
|
|
|
|
#============================================================================= |
|
627
|
|
|
|
|
|
|
# CVS HISTORY |
|
628
|
|
|
|
|
|
|
#============================================================================= |
|
629
|
|
|
|
|
|
|
# $Log: Base.pm,v $ |
|
630
|
|
|
|
|
|
|
# Revision 1.10 2008-08-28 23:35:28 amon |
|
631
|
|
|
|
|
|
|
# perldoc section regularization. |
|
632
|
|
|
|
|
|
|
# |
|
633
|
|
|
|
|
|
|
# Revision 1.9 2008-08-09 20:25:13 amon |
|
634
|
|
|
|
|
|
|
# Documentation error fixed. |
|
635
|
|
|
|
|
|
|
# |
|
636
|
|
|
|
|
|
|
# Revision 1.8 2008-08-09 12:56:01 amon |
|
637
|
|
|
|
|
|
|
# Used wrong method name. Fixed |
|
638
|
|
|
|
|
|
|
# |
|
639
|
|
|
|
|
|
|
# Revision 1.7 2008-08-04 12:12:20 amon |
|
640
|
|
|
|
|
|
|
# Added unary and binary ops; made init methods synonums for add methods. |
|
641
|
|
|
|
|
|
|
# |
|
642
|
|
|
|
|
|
|
# Revision 1.6 2008-07-27 15:16:17 amon |
|
643
|
|
|
|
|
|
|
# Wrote lexical parse for Entry; error checking on eval and other minor issues. |
|
644
|
|
|
|
|
|
|
# |
|
645
|
|
|
|
|
|
|
# Revision 1.5 2008-07-25 14:30:42 amon |
|
646
|
|
|
|
|
|
|
# Documentation improvements and corrections. |
|
647
|
|
|
|
|
|
|
# |
|
648
|
|
|
|
|
|
|
# Revision 1.4 2008-07-24 20:19:43 amon |
|
649
|
|
|
|
|
|
|
# Just in case I missed anything. |
|
650
|
|
|
|
|
|
|
# |
|
651
|
|
|
|
|
|
|
# Revision 1.3 2008-07-24 13:35:26 amon |
|
652
|
|
|
|
|
|
|
# switch to NeXT style alloc/init format for FileHash and Entry classes. |
|
653
|
|
|
|
|
|
|
# |
|
654
|
|
|
|
|
|
|
# Revision 1.2 2008-07-23 21:12:24 amon |
|
655
|
|
|
|
|
|
|
# Moved notes out of file headers; a few doc updates; added assertion checks; |
|
656
|
|
|
|
|
|
|
# minor bug fixes. |
|
657
|
|
|
|
|
|
|
# |
|
658
|
|
|
|
|
|
|
# 20080722 Dale Amon |
|
659
|
|
|
|
|
|
|
# Renamed FileHash.pm to Base.pm so it is FileHash::Base. |
|
660
|
|
|
|
|
|
|
# 20080717 Dale Amon |
|
661
|
|
|
|
|
|
|
# Split FilenameHash, formerly Directory class, into FileHash |
|
662
|
|
|
|
|
|
|
# FileHash::Name and FileHash::Content. |
|
663
|
|
|
|
|
|
|
# 20080625 Dale Amon |
|
664
|
|
|
|
|
|
|
# Created. |
|
665
|
|
|
|
|
|
|
1; |