19 |
push @a, scalar localtime() if $a[0] =~ m{^info}; |
push @a, scalar localtime() if $a[0] =~ m{^info}; |
20 |
|
|
21 |
if ( ! defined $meteor_fh ) { |
if ( ! defined $meteor_fh ) { |
22 |
warn "# open connection to $meteor_server"; |
if ( $meteor_fh = |
23 |
$meteor_fh = IO::Socket::INET->new( |
IO::Socket::INET->new( |
24 |
PeerAddr => $meteor_server, |
PeerAddr => $meteor_server, |
25 |
Timeout => 1, |
Timeout => 1, |
26 |
) || warn "can't connect to meteor $meteor_server: $!"; # FIXME warn => die for production |
) |
27 |
$meteor_fh = 0; # don't try again |
) { |
28 |
|
warn "# meteor connected to $meteor_server"; |
29 |
|
} else { |
30 |
|
warn "can't connect to meteor $meteor_server: $!"; |
31 |
|
$meteor_fh = 0; |
32 |
|
} |
33 |
} |
} |
34 |
|
|
35 |
warn ">> meteor ",dump( @a ); |
if ( $meteor_fh ) { |
36 |
print $meteor_fh "ADDMESSAGE test ",join('|',@a),"\n" if $meteor_fh; |
warn ">> meteor ",dump( @a ); |
37 |
|
print $meteor_fh "ADDMESSAGE test ",join('|',@a),"\n" |
38 |
|
} |
39 |
} |
} |
40 |
|
|
41 |
my $debug = 0; |
my $debug = 0; |
48 |
my $handshake = "none"; |
my $handshake = "none"; |
49 |
|
|
50 |
my $program_path = './program/'; |
my $program_path = './program/'; |
51 |
|
my $secure_path = './secure/'; |
52 |
|
|
53 |
my $response = { |
my $response = { |
54 |
'd500090400110a0500027250' => 'version?', |
'd500090400110a0500027250' => 'version?', |
109 |
my $tags_data; |
my $tags_data; |
110 |
my $visible_tags; |
my $visible_tags; |
111 |
|
|
112 |
|
my $item_type = { |
113 |
|
1 => 'Book', |
114 |
|
6 => 'CD/CD ROM', |
115 |
|
2 => 'Magazine', |
116 |
|
13 => 'Book with Audio Tape', |
117 |
|
9 => 'Book with CD/CD ROM', |
118 |
|
0 => 'Other', |
119 |
|
|
120 |
|
5 => 'Video', |
121 |
|
4 => 'Audio Tape', |
122 |
|
3 => 'Bound Journal', |
123 |
|
8 => 'Book with Diskette', |
124 |
|
7 => 'Diskette', |
125 |
|
}; |
126 |
|
|
127 |
|
warn "## known item type: ",dump( $item_type ) if $debug; |
128 |
|
|
129 |
my $port=new Device::SerialPort($device) || die "can't open serial port $device: $!\n"; |
my $port=new Device::SerialPort($device) || die "can't open serial port $device: $!\n"; |
130 |
warn "using $device $handshake $baudrate $databits $parity $stopbits" if $debug; |
warn "using $device $handshake $baudrate $databits $parity $stopbits" if $debug; |
131 |
$handshake=$port->handshake($handshake); |
$handshake=$port->handshake($handshake); |
215 |
meteor( 'write', $tag ); |
meteor( 'write', $tag ); |
216 |
write_tag( $tag ); |
write_tag( $tag ); |
217 |
} |
} |
218 |
|
if ( -e "$secure_path/$tag" ) { |
219 |
|
meteor( 'secure', $tag ); |
220 |
|
secure_tag( $tag ); |
221 |
|
} |
222 |
} |
} |
223 |
|
|
224 |
foreach my $tag ( keys %$last_visible_tags ) { |
foreach my $tag ( keys %$last_visible_tags ) { |
251 |
$tag_data_block->{$tag}->[ $ord ] = $data; |
$tag_data_block->{$tag}->[ $ord ] = $data; |
252 |
} |
} |
253 |
$tags_data->{ $tag } = join('', @{ $tag_data_block->{$tag} }); |
$tags_data->{ $tag } = join('', @{ $tag_data_block->{$tag} }); |
254 |
print "DATA $tag ",dump( $tags_data ), "\n"; |
|
255 |
|
my $item_type_nr = ord(substr( $tags_data->{$tag}, 3, 1 )); |
256 |
|
print "DATA $tag ",dump( $tags_data ), " item type: ", ( $item_type->{ $item_type_nr } || "UNKWOWN '$item_type_nr' in " . dump( $item_type ) ), "\n"; |
257 |
} |
} |
258 |
|
|
259 |
sub read_tag { |
sub read_tag { |
280 |
} |
} |
281 |
); |
); |
282 |
|
|
283 |
|
my $security; |
284 |
|
|
285 |
|
cmd( |
286 |
|
"D6 00 0B 0A $tag 1234", "check security $tag", |
287 |
|
"D6 00 0D 0A 00", sub { |
288 |
|
my $rest = shift; |
289 |
|
my $from_tag; |
290 |
|
( $from_tag, $security ) = ( substr($rest,0,8), substr($rest,8,1) ); |
291 |
|
die "security from other tag: ",as_hex( $from_tag ) if $from_tag ne str2bytes( $tag ); |
292 |
|
$security = as_hex( $security ); |
293 |
|
warn "# SECURITY $tag = $security\n"; |
294 |
|
} |
295 |
|
); |
296 |
|
|
297 |
|
my $data = $tags_data->{$tag} || die "no data for $tag"; |
298 |
|
my ( $u1, $set_item, $u2, $type, $content, $br_lib, $custom ) = unpack('C4Z16Nl>',$data); |
299 |
|
my $set = ( $set_item & 0xf0 ) >> 4; |
300 |
|
my $total = ( $set_item & 0x0f ); |
301 |
|
my $branch = $br_lib >> 20; |
302 |
|
my $library = $br_lib & 0x000fffff; |
303 |
|
print "TAG $tag [$u1] set: $set/$total [$u2] type: $type '$content' library: $library branch: $branch custom: $custom security: $security\n"; |
304 |
|
|
305 |
} |
} |
306 |
|
|
307 |
sub write_tag { |
sub write_tag { |
315 |
|
|
316 |
my $hex_data = unpack('H*', $data) . ' 00' x ( 16 - length($data) ); |
my $hex_data = unpack('H*', $data) . ' 00' x ( 16 - length($data) ); |
317 |
|
|
318 |
print "write_tag $tag = $data ",dump( $hex_data ); |
print "write_tag $tag = ",dump( $data ), " == $hex_data\n"; |
319 |
|
|
320 |
cmd( |
cmd( |
321 |
"D6 00 26 04 $tag 00 06 00 04 11 00 01 $hex_data 00 00 00 00 FD3B", "write $tag", |
"d6 00 26 04 $tag 00 06 00 04 11 00 01 $hex_data 00 00 00 00 fd3b", "write $tag", |
322 |
"D6 00 0D 04 00 $tag 06 AFB1", sub { assert() }, |
"d6 00 0d 04 00 $tag 06 afb1", sub { assert() }, |
323 |
) foreach ( 1 .. 3 ); # XXX 3M software does this three times! |
) foreach ( 1 .. 3 ); # xxx 3m software does this three times! |
324 |
|
|
325 |
my $to = $path; |
my $to = $path; |
326 |
$to .= '.' . time(); |
$to .= '.' . time(); |
331 |
delete $tags_data->{$tag}; # force re-read of tag |
delete $tags_data->{$tag}; # force re-read of tag |
332 |
} |
} |
333 |
|
|
334 |
|
sub secure_tag { |
335 |
|
my ($tag) = @_; |
336 |
|
|
337 |
|
my $path = "$secure_path/$tag"; |
338 |
|
my $data = substr(read_file( $path ),0,2); |
339 |
|
|
340 |
|
cmd( |
341 |
|
"d6 00 0c 09 $tag $data 1234", "secure $tag -> $data", |
342 |
|
"d6 00 0c 09 00 $tag 1234", sub { assert() }, |
343 |
|
); |
344 |
|
|
345 |
|
my $to = $path; |
346 |
|
$to .= '.' . time(); |
347 |
|
|
348 |
|
rename $path, $to; |
349 |
|
print ">> $to\n"; |
350 |
|
} |
351 |
|
|
352 |
exit; |
exit; |
353 |
|
|
354 |
for ( 1 .. 3 ) { |
for ( 1 .. 3 ) { |