1 |
dpavlin |
14 |
package NMEA; |
2 |
|
|
|
3 |
|
|
use strict; |
4 |
|
|
use warnings; |
5 |
|
|
|
6 |
|
|
use Data::Dump qw/dump/; |
7 |
|
|
|
8 |
|
|
sub hhmm { |
9 |
|
|
my $a = shift; |
10 |
|
|
if ( $a =~ m/^(\d+)(\d\d\.\d\d+)$/ ) { |
11 |
|
|
return sprintf("%1.6f", $1 + ( $2 / 60 )); |
12 |
|
|
} else { |
13 |
|
|
warn "## skipped $a\n"; |
14 |
|
|
return; |
15 |
|
|
} |
16 |
|
|
} |
17 |
|
|
|
18 |
|
|
my $got_it; |
19 |
dpavlin |
26 |
my $stats; |
20 |
dpavlin |
14 |
|
21 |
dpavlin |
27 |
sub error { |
22 |
|
|
my ( $self, $message, $line ) = @_; |
23 |
|
|
push @{ $stats->{errors}->{$message} }, $line; |
24 |
|
|
} |
25 |
|
|
|
26 |
|
|
sub stats { |
27 |
|
|
my $self = shift; |
28 |
|
|
return $stats; |
29 |
|
|
} |
30 |
|
|
|
31 |
dpavlin |
14 |
sub line { |
32 |
|
|
my $self = shift; |
33 |
|
|
my $l = shift || return; |
34 |
|
|
|
35 |
|
|
if ( m/\$GPRMC/ ) { |
36 |
|
|
chomp; |
37 |
|
|
my @a = split(/,/,$_); |
38 |
|
|
|
39 |
dpavlin |
27 |
if ( $#a != 12 ) { |
40 |
|
|
$self->error( 'invalid line length' => $_ ); |
41 |
|
|
return; |
42 |
|
|
} |
43 |
dpavlin |
14 |
|
44 |
dpavlin |
16 |
# warn "## [$#a] ", join(' ', map { "$_:$a[$_]" } ( 0 .. $#a )), " from $_\n"; |
45 |
dpavlin |
14 |
|
46 |
|
|
# is valid? |
47 |
dpavlin |
27 |
if ( $a[2] ne 'A' ) { |
48 |
|
|
$self->error( 'not valid' => $_ ); |
49 |
|
|
return; |
50 |
|
|
} |
51 |
dpavlin |
14 |
|
52 |
|
|
my $hash; |
53 |
|
|
my $i = 1; # skip GPRMC; |
54 |
|
|
$hash->{$_} = $a[$i++] foreach ( qw/ |
55 |
|
|
time validity lat_hhmm lat_ns lon_hhmm lon_ew speed course date var var_ew |
56 |
|
|
/ ); |
57 |
|
|
|
58 |
|
|
$hash->{lat} = hhmm( ( $hash->{lat_ns} eq 'S' ? -1 : 1 ) * $hash->{lat_hhmm} ) || return; |
59 |
dpavlin |
16 |
$hash->{lon} = hhmm( ( $hash->{lon_ew} eq 'W' ? -1 : 1 ) * $hash->{lon_hhmm} ) || return; |
60 |
dpavlin |
14 |
|
61 |
dpavlin |
27 |
if ( $got_it->{ $hash->{lat} . ' ' . $hash->{lon} }++ ) { |
62 |
|
|
$self->error( 'duplicate line' => $_ ); |
63 |
|
|
return; |
64 |
|
|
} |
65 |
dpavlin |
14 |
|
66 |
dpavlin |
29 |
foreach my $n ( qw/lat lon time speed/ ) { |
67 |
|
|
$stats->{'min_'.$n} ||= $hash->{$n}; |
68 |
|
|
$stats->{'max_'.$n} ||= $hash->{$n}; |
69 |
|
|
$stats->{'min_'.$n} = $hash->{$n} if $hash->{$n} < $stats->{'min_'.$n}; |
70 |
|
|
$stats->{'max_'.$n} = $hash->{$n} if $hash->{$n} > $stats->{'max_'.$n}; |
71 |
|
|
} |
72 |
dpavlin |
27 |
|
73 |
|
|
$hash->{number} = $stats->{total}++; |
74 |
|
|
|
75 |
dpavlin |
16 |
# warn "##>>>> ",dump( $hash ); |
76 |
dpavlin |
14 |
return $hash; |
77 |
|
|
} |
78 |
|
|
|
79 |
|
|
return; |
80 |
|
|
} |
81 |
|
|
|
82 |
|
|
1; |