line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
|
2
|
|
|
|
|
|
|
=head1 NAME |
3
|
|
|
|
|
|
|
|
4
|
|
|
|
|
|
|
XBase::Memo - Generic support for various memo formats |
5
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
=cut |
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
package XBase::Memo; |
9
|
|
|
|
|
|
|
|
10
|
11
|
|
|
11
|
|
44
|
use strict; |
|
11
|
|
|
|
|
11
|
|
|
11
|
|
|
|
|
237
|
|
11
|
11
|
|
|
11
|
|
3456
|
use XBase::Base; |
|
11
|
|
|
|
|
18
|
|
|
11
|
|
|
|
|
267
|
|
12
|
|
|
|
|
|
|
|
13
|
11
|
|
|
11
|
|
48
|
use vars qw( $VERSION @ISA ); |
|
11
|
|
|
|
|
58
|
|
|
11
|
|
|
|
|
5894
|
|
14
|
|
|
|
|
|
|
@ISA = qw( XBase::Base ); |
15
|
|
|
|
|
|
|
$VERSION = '1.02'; |
16
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
# Read header is called from open to fill the object structures |
18
|
|
|
|
|
|
|
sub read_header { |
19
|
7
|
|
|
7
|
0
|
9
|
my $self = shift; |
20
|
7
|
|
|
|
|
12
|
my %options = @_; |
21
|
|
|
|
|
|
|
|
22
|
7
|
|
|
|
|
8
|
my $header; |
23
|
7
|
|
|
|
|
25
|
my $read_h_len = $self->read($header, 512); |
24
|
7
|
50
|
33
|
|
|
22
|
unless ($read_h_len == 512 or $read_h_len == 24) { |
25
|
0
|
|
|
|
|
0
|
$self->Error("Error reading header of $self->{'filename'}: $!\n"); |
26
|
0
|
|
|
|
|
0
|
return; |
27
|
|
|
|
|
|
|
}; |
28
|
|
|
|
|
|
|
|
29
|
7
|
|
|
|
|
7
|
my ($next_for_append, $block_size, $version); |
30
|
7
|
|
|
|
|
12
|
my $filename = $self->{'filename'}; |
31
|
7
|
100
|
|
|
|
35
|
if ($filename =~ /\.fpt$/i) { |
|
|
50
|
|
|
|
|
|
32
|
1
|
|
|
|
|
4
|
($next_for_append, $block_size) = unpack 'N@6n', $header; |
33
|
1
|
|
|
|
|
2
|
$version = 5; |
34
|
1
|
|
|
|
|
3
|
bless $self, 'XBase::Memo::Fox'; |
35
|
|
|
|
|
|
|
} elsif ($filename =~ /\.smt$/i) { |
36
|
0
|
|
|
|
|
0
|
($next_for_append, $block_size) = unpack 'VV', $header; |
37
|
0
|
|
|
|
|
0
|
bless $self, 'XBase::Memo::Apollo'; |
38
|
|
|
|
|
|
|
} else { |
39
|
6
|
|
|
|
|
19
|
($next_for_append, $version, $block_size) |
40
|
|
|
|
|
|
|
= unpack 'V @16C @20v', $header; |
41
|
6
|
|
|
|
|
29
|
my $dbf_version = $options{'dbf_version'}; |
42
|
6
|
50
|
|
|
|
13
|
$dbf_version = 15 unless defined $dbf_version; |
43
|
6
|
50
|
33
|
|
|
17
|
if ((($dbf_version & 15) == 3) or $version == 3) { |
44
|
6
|
|
|
|
|
4
|
$block_size = 512; |
45
|
6
|
|
|
|
|
6
|
$version = 3; |
46
|
6
|
|
|
|
|
17
|
bless $self, 'XBase::Memo::dBaseIII'; |
47
|
|
|
|
|
|
|
} else { |
48
|
0
|
|
|
|
|
0
|
$version = 4; |
49
|
0
|
|
|
|
|
0
|
bless $self, 'XBase::Memo::dBaseIV'; |
50
|
|
|
|
|
|
|
} |
51
|
|
|
|
|
|
|
} |
52
|
|
|
|
|
|
|
|
53
|
7
|
50
|
|
|
|
16
|
$block_size = 512 if int($block_size) == 0; |
54
|
|
|
|
|
|
|
|
55
|
7
|
|
|
|
|
95
|
$next_for_append = int ((((-s $self->{'filename'}) - 1) / $block_size) + 1); |
56
|
|
|
|
|
|
|
|
57
|
7
|
|
|
|
|
11
|
@{$self}{ qw( next_for_append header_len record_len version ) } |
|
7
|
|
|
|
|
25
|
|
58
|
|
|
|
|
|
|
= ( $next_for_append, $block_size, $block_size, $version ); |
59
|
|
|
|
|
|
|
|
60
|
7
|
|
|
|
|
13
|
$self->{'memosep'} = $options{'memosep'}; |
61
|
|
|
|
|
|
|
|
62
|
7
|
|
|
|
|
41
|
1; |
63
|
|
|
|
|
|
|
} |
64
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
sub write_record { |
66
|
6
|
|
|
6
|
1
|
24
|
my ($self, $num) = (shift, shift); |
67
|
6
|
|
|
|
|
12
|
my $length = length(join '', @_); |
68
|
6
|
|
|
|
|
9
|
my $record_len = $self->{'record_len'}; |
69
|
6
|
|
|
|
|
14
|
my $num_of_blocks = int (($length + $record_len - 1) / $record_len); |
70
|
6
|
|
|
|
|
17
|
$self->SUPER::write_record($num, @_); |
71
|
6
|
50
|
33
|
|
|
19
|
if ($num < 0 or $num > $self->last_record()) { |
72
|
6
|
|
|
|
|
14
|
my $packed = pack "V", $num + $num_of_blocks + 1; |
73
|
6
|
100
|
|
|
|
16
|
if (ref $self eq 'XBase::Memo::Fox') { |
74
|
2
|
|
|
|
|
3
|
$packed = pack "N", $num + $num_of_blocks + 1; |
75
|
|
|
|
|
|
|
} |
76
|
6
|
|
|
|
|
14
|
$self->SUPER::write_to(0, $packed); |
77
|
6
|
|
|
|
|
8
|
$self->{'next_for_append'} = $num + $num_of_blocks + 1; |
78
|
|
|
|
|
|
|
} |
79
|
6
|
|
|
|
|
6
|
$num; |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
sub last_record { |
83
|
54
|
|
|
54
|
0
|
117
|
shift->{'next_for_append'} - 2; |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
sub create { |
87
|
1
|
|
|
1
|
0
|
1
|
my $self = shift; |
88
|
1
|
|
|
|
|
3
|
my %options = @_; |
89
|
1
|
50
|
|
|
|
4
|
$self->create_file($options{'name'}) or return; |
90
|
1
|
|
|
|
|
2
|
my $version = $options{'version'}; |
91
|
1
|
50
|
|
|
|
6
|
if ($options{'name'} =~ /\.smt$/i) { |
|
|
50
|
|
|
|
|
|
92
|
|
|
|
|
|
|
# For xmt file (whatever they are) |
93
|
0
|
0
|
|
|
|
0
|
$self->write_to(0, pack 'VV a504', 1, 512, '') or return; |
94
|
|
|
|
|
|
|
} elsif ($version == 5) { |
95
|
|
|
|
|
|
|
# Fox fpt file |
96
|
0
|
0
|
|
|
|
0
|
$self->write_to(0, pack 'N a2 n a504', 1, '', 512, '') or return; |
97
|
|
|
|
|
|
|
} else { |
98
|
1
|
50
|
|
|
|
2
|
$version = 3 unless defined $version; |
99
|
1
|
50
|
|
|
|
2
|
$version = 0 if $version == 4; |
100
|
|
|
|
|
|
|
$self->write_to(0, pack 'VVa8Ca3va490', 1, 0, |
101
|
1
|
50
|
|
|
|
8
|
$options{'dbf_filename'}, $version, '', 512, '') |
102
|
|
|
|
|
|
|
or return; |
103
|
|
|
|
|
|
|
} |
104
|
1
|
|
|
|
|
3
|
$self->close(); |
105
|
1
|
|
|
|
|
4
|
return $self; |
106
|
|
|
|
|
|
|
} |
107
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
# ################################ |
110
|
|
|
|
|
|
|
# dBase III+ specific memo methods |
111
|
|
|
|
|
|
|
|
112
|
|
|
|
|
|
|
package XBase::Memo::dBaseIII; |
113
|
|
|
|
|
|
|
|
114
|
11
|
|
|
11
|
|
51
|
use XBase::Base; |
|
11
|
|
|
|
|
11
|
|
|
11
|
|
|
|
|
230
|
|
115
|
11
|
|
|
11
|
|
33
|
use vars qw( @ISA ); |
|
11
|
|
|
|
|
11
|
|
|
11
|
|
|
|
|
3031
|
|
116
|
|
|
|
|
|
|
@ISA = qw( XBase::Memo ); |
117
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
sub read_record { |
119
|
10
|
|
|
10
|
|
14
|
my ($self, $num) = @_; |
120
|
10
|
|
|
|
|
9
|
my $result = ''; |
121
|
10
|
|
|
|
|
16
|
my $last = $self->last_record(); |
122
|
10
|
100
|
|
|
|
20
|
if (not defined $self->{'memosep'}) { |
123
|
2
|
|
|
|
|
4
|
$self->{'memosep'} = "\x1a\x1a"; |
124
|
2
|
50
|
|
|
|
6
|
if (not defined $self->read_record($last)) { |
125
|
0
|
|
|
|
|
0
|
$self->{'memosep'} = "\x1a"; |
126
|
0
|
0
|
|
|
|
0
|
if (not defined $self->read_record($last)) { |
127
|
0
|
|
|
|
|
0
|
$self->{'memosep'} = "\x1a\x1a"; |
128
|
|
|
|
|
|
|
} |
129
|
|
|
|
|
|
|
} |
130
|
|
|
|
|
|
|
} |
131
|
|
|
|
|
|
|
|
132
|
10
|
|
|
|
|
16
|
while ($num <= $last) { |
133
|
10
|
50
|
|
|
|
22
|
my $buffer = $self->SUPER::read_record($num, -1) or return; |
134
|
10
|
|
|
|
|
15
|
my $index = index($buffer, $self->{'memosep'}); |
135
|
10
|
50
|
|
|
|
20
|
if ($index >= 0) { |
136
|
10
|
|
|
|
|
44
|
return $result . substr($buffer, 0, $index); |
137
|
|
|
|
|
|
|
} |
138
|
0
|
|
|
|
|
0
|
$result .= $buffer; |
139
|
0
|
|
|
|
|
0
|
$num++; |
140
|
|
|
|
|
|
|
} |
141
|
0
|
|
|
|
|
0
|
return; |
142
|
|
|
|
|
|
|
} |
143
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
sub write_record { |
145
|
4
|
|
|
4
|
|
6
|
my ($self, $num) = (shift, shift); |
146
|
4
|
|
|
|
|
2
|
my $type = shift; |
147
|
4
|
|
|
|
|
10
|
my $data = join "", @_, "\x1a\x1a"; |
148
|
4
|
50
|
33
|
|
|
13
|
if ($num >= 0 and $num <= $self->last_record()) { |
149
|
0
|
|
|
|
|
0
|
my $buffer = $self->read_record($num); |
150
|
0
|
0
|
|
|
|
0
|
if (defined $buffer) { |
151
|
0
|
|
|
|
|
0
|
my $length = length $buffer; |
152
|
0
|
|
|
|
|
0
|
my $record_len = $self->{'record_len'}; |
153
|
0
|
|
|
|
|
0
|
my $space_in_blocks = |
154
|
|
|
|
|
|
|
int (($length + $record_len - 3) / $record_len); |
155
|
0
|
|
|
|
|
0
|
my $len_in_blocks = |
156
|
|
|
|
|
|
|
int ((length($data) + $record_len - 1) / $record_len); |
157
|
0
|
0
|
|
|
|
0
|
if ($len_in_blocks > $space_in_blocks) { |
158
|
0
|
|
|
|
|
0
|
$num = $self->last_record() + 1; |
159
|
|
|
|
|
|
|
} |
160
|
|
|
|
|
|
|
} |
161
|
|
|
|
|
|
|
} else { |
162
|
4
|
|
|
|
|
13
|
$num = $self->last_record() + 1; |
163
|
|
|
|
|
|
|
} |
164
|
4
|
|
|
|
|
11
|
$self->SUPER::write_record($num, $data); |
165
|
4
|
|
|
|
|
7
|
$num; |
166
|
|
|
|
|
|
|
} |
167
|
|
|
|
|
|
|
|
168
|
|
|
|
|
|
|
# ################################ |
169
|
|
|
|
|
|
|
# dBase IV specific memo methods |
170
|
|
|
|
|
|
|
|
171
|
|
|
|
|
|
|
package XBase::Memo::dBaseIV; |
172
|
|
|
|
|
|
|
|
173
|
11
|
|
|
11
|
|
42
|
use XBase::Base; |
|
11
|
|
|
|
|
15
|
|
|
11
|
|
|
|
|
180
|
|
174
|
11
|
|
|
11
|
|
34
|
use vars qw( @ISA ); |
|
11
|
|
|
|
|
15
|
|
|
11
|
|
|
|
|
4098
|
|
175
|
|
|
|
|
|
|
@ISA = qw( XBase::Memo ); |
176
|
|
|
|
|
|
|
|
177
|
|
|
|
|
|
|
sub read_record { |
178
|
4
|
|
|
4
|
|
5
|
my ($self, $num) = @_; |
179
|
4
|
|
|
|
|
5
|
my $result = ''; |
180
|
4
|
|
|
|
|
4
|
my $last = $self->last_record; |
181
|
|
|
|
|
|
|
|
182
|
4
|
|
|
|
|
9
|
my $buffer = $self->SUPER::read_record($num, -1); |
183
|
4
|
50
|
|
|
|
7
|
if (not defined $buffer) { return; } |
|
0
|
|
|
|
|
0
|
|
184
|
4
|
|
|
|
|
3
|
my $unpackstr; |
185
|
4
|
50
|
|
|
|
6
|
if (ref $self eq 'XBase::Memo::Fox') { |
186
|
4
|
|
|
|
|
4
|
$unpackstr = 'NN'; |
187
|
|
|
|
|
|
|
} else { |
188
|
0
|
|
|
|
|
0
|
$unpackstr = 'VV'; |
189
|
0
|
0
|
|
|
|
0
|
return unless substr($buffer, 0, 4) eq "\xff\xff\x08\x00"; |
190
|
|
|
|
|
|
|
} |
191
|
4
|
|
|
|
|
5
|
my ($unused_id, $length) = unpack $unpackstr, $buffer; |
192
|
4
|
50
|
|
|
|
8
|
$length += 8 if ref $self eq 'XBase::Memo::Fox'; |
193
|
|
|
|
|
|
|
|
194
|
4
|
|
|
|
|
5
|
my $block_size = $self->{'record_len'}; |
195
|
4
|
50
|
|
|
|
5
|
if ($length < $block_size) { |
196
|
4
|
|
|
|
|
17
|
return substr $buffer, 8, $length - 8; |
197
|
|
|
|
|
|
|
} |
198
|
0
|
|
|
|
|
0
|
my $rest_length = $length - $block_size; |
199
|
0
|
|
|
|
|
0
|
my $rest_data = $self->SUPER::read_record($num + 1, $rest_length); |
200
|
0
|
0
|
|
|
|
0
|
if (not defined $rest_data) { return; } |
|
0
|
|
|
|
|
0
|
|
201
|
0
|
|
|
|
|
0
|
return substr($buffer, 8) . $rest_data; |
202
|
|
|
|
|
|
|
} |
203
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
sub write_record { |
205
|
2
|
|
|
2
|
|
3
|
my ($self, $num) = (shift, shift); |
206
|
2
|
|
|
|
|
4
|
my $type = shift; |
207
|
2
|
|
|
|
|
4
|
my $data = join "", @_; |
208
|
2
|
|
|
|
|
2
|
my $length = (length $data) + 8; |
209
|
|
|
|
|
|
|
|
210
|
2
|
|
|
|
|
6
|
my $startfield = "\xff\xff\x08\x00" . pack('V', $length); |
211
|
2
|
50
|
|
|
|
5
|
if (ref $self eq 'XBase::Memo::Fox') { |
212
|
2
|
50
|
|
|
|
6
|
if ($type eq 'P') { $startfield = pack 'N', 0; } |
|
0
|
50
|
|
|
|
0
|
|
213
|
2
|
|
|
|
|
3
|
elsif ($type eq 'M') { $startfield = pack 'N', 1; } |
214
|
0
|
|
|
|
|
0
|
else { $startfield = pack 'N', 2; } |
215
|
2
|
|
|
|
|
4
|
$startfield .= pack 'N', ($length - 8); |
216
|
|
|
|
|
|
|
} |
217
|
|
|
|
|
|
|
### $data = $startfield . $data . "\x1a\x1a"; |
218
|
2
|
|
|
|
|
2
|
$data = $startfield . $data; |
219
|
|
|
|
|
|
|
|
220
|
2
|
50
|
33
|
|
|
10
|
if ($num >= 0 and $num <= $self->last_record()) { |
221
|
0
|
|
|
|
|
0
|
my $buffer = $self->read_record($num); |
222
|
0
|
0
|
|
|
|
0
|
if (defined $buffer) { |
223
|
0
|
|
|
|
|
0
|
my $length = (length $buffer) - 8; |
224
|
0
|
|
|
|
|
0
|
my $record_len = $self->{'record_len'}; |
225
|
0
|
|
|
|
|
0
|
my $space_in_blocks = |
226
|
|
|
|
|
|
|
int (($length + $record_len - 11) / $record_len); |
227
|
0
|
|
|
|
|
0
|
my $len_in_blocks = |
228
|
|
|
|
|
|
|
int ((length($data) + $record_len - 1) / $record_len); |
229
|
0
|
0
|
|
|
|
0
|
if ($len_in_blocks > $space_in_blocks) { |
230
|
0
|
|
|
|
|
0
|
$num = $self->last_record() + 1; |
231
|
|
|
|
|
|
|
} |
232
|
|
|
|
|
|
|
} else { |
233
|
0
|
|
|
|
|
0
|
$num = $self->last_record() + 1; |
234
|
|
|
|
|
|
|
} |
235
|
|
|
|
|
|
|
} else { |
236
|
2
|
|
|
|
|
5
|
$num = $self->last_record() + 1; |
237
|
|
|
|
|
|
|
} |
238
|
2
|
|
|
|
|
4
|
my $fill = $self->{'record_len'} - (( length $data ) % $self->{'record_len'}); |
239
|
2
|
|
|
|
|
3
|
$data .= "\000" x $fill; |
240
|
2
|
|
|
|
|
5
|
$self->SUPER::write_record($num, $data); |
241
|
2
|
|
|
|
|
4
|
$num; |
242
|
|
|
|
|
|
|
} |
243
|
|
|
|
|
|
|
|
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
# ####################################### |
246
|
|
|
|
|
|
|
# FoxPro specific memo methods (fpt file) |
247
|
|
|
|
|
|
|
|
248
|
|
|
|
|
|
|
package XBase::Memo::Fox; |
249
|
|
|
|
|
|
|
|
250
|
11
|
|
|
11
|
|
42
|
use XBase::Base; |
|
11
|
|
|
|
|
15
|
|
|
11
|
|
|
|
|
173
|
|
251
|
11
|
|
|
11
|
|
35
|
use vars qw( @ISA ); |
|
11
|
|
|
|
|
10
|
|
|
11
|
|
|
|
|
540
|
|
252
|
|
|
|
|
|
|
@ISA = qw( XBase::Memo::dBaseIV ); |
253
|
|
|
|
|
|
|
|
254
|
|
|
|
|
|
|
# ####################################### |
255
|
|
|
|
|
|
|
# Apollo specific memo methods (smt file) |
256
|
|
|
|
|
|
|
# |
257
|
|
|
|
|
|
|
# This is a real hack! No documentation used but it works for all files |
258
|
|
|
|
|
|
|
# i have tested with. |
259
|
|
|
|
|
|
|
# Dirk Tostmann (tostmann@tiss.com) |
260
|
|
|
|
|
|
|
|
261
|
|
|
|
|
|
|
package XBase::Memo::Apollo; |
262
|
|
|
|
|
|
|
|
263
|
11
|
|
|
11
|
|
35
|
use XBase::Base; |
|
11
|
|
|
|
|
11
|
|
|
11
|
|
|
|
|
156
|
|
264
|
11
|
|
|
11
|
|
31
|
use vars qw( @ISA ); |
|
11
|
|
|
|
|
11
|
|
|
11
|
|
|
|
|
1916
|
|
265
|
|
|
|
|
|
|
@ISA = qw( XBase::Memo::dBaseIV ); |
266
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
sub read_record { |
268
|
0
|
|
|
0
|
|
|
my ($self, $num) = @_; |
269
|
0
|
|
|
|
|
|
my $result = ''; |
270
|
|
|
|
|
|
|
|
271
|
0
|
0
|
|
|
|
|
return if $num =~ /^\s+$/; |
272
|
|
|
|
|
|
|
|
273
|
0
|
|
|
|
|
|
my ($block, $len, $offset) = unpack('vVV', $num); |
274
|
0
|
|
|
|
|
|
$block *= 8; |
275
|
|
|
|
|
|
|
|
276
|
0
|
|
|
|
|
|
$result = $self->read_from($offset * $block, $len); |
277
|
|
|
|
|
|
|
|
278
|
0
|
|
|
|
|
|
$result; |
279
|
|
|
|
|
|
|
} |
280
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
sub write_record { |
282
|
0
|
|
|
0
|
|
|
my ($self, $num, $type) = (shift, shift, shift); |
283
|
0
|
|
|
|
|
|
my $data = join "", @_; |
284
|
0
|
|
|
|
|
|
my $length = length $data; |
285
|
0
|
|
|
|
|
|
$num = $self->SUPER::write_record($self->{'next_for_append'}, $data); |
286
|
0
|
0
|
0
|
|
|
|
if (defined $num and $num) { |
287
|
0
|
|
|
|
|
|
pack 'vVV', $self->{'block_length'} / 8 , $length, $num; |
288
|
|
|
|
|
|
|
} else { |
289
|
0
|
|
|
|
|
|
' ' x 10; |
290
|
|
|
|
|
|
|
} |
291
|
|
|
|
|
|
|
} |
292
|
|
|
|
|
|
|
1; |
293
|
|
|
|
|
|
|
|
294
|
|
|
|
|
|
|
__END__ |