- # local processing 1
- my $pcr;
- eval {
- $pcr = Local::pcprot($self, $pcno, @field);
- };
-# dbg("Local::pcprot error $@") if isdbg('local') if $@;
- return if $pcr;
-
- SWITCH: {
- if ($pcno == 10) { # incoming talk
-
- # will we allow it at all?
- if ($censorpc) {
- my @bad;
- if (@bad = BadWords::check($field[3])) {
- dbg("PCPROT: Bad words: @bad, dropped") if isdbg('chanerr');
- return;
- }
- }
-
- # is it for me or one of mine?
- my ($to, $via, $call, $dxchan);
- if ($field[5] gt ' ') {
- $call = $via = $field[2];
- $to = $field[5];
- } else {
- $call = $to = $field[2];
- }
- $dxchan = DXChannel->get($main::myalias) if $call eq $main::mycall;
- $dxchan = DXChannel->get($call) unless $dxchan;
- if ($dxchan && $dxchan->is_user) {
- $field[3] =~ s/\%5E/^/g;
- $dxchan->talk($field[1], $to, $via, $field[3]);
- } else {
- $self->route($field[2], $line); # relay it on its way
- }
- return;
- }
-
- if ($pcno == 11 || $pcno == 26) { # dx spot
-
- # route 'foreign' pc26s
- if ($pcno == 26) {
- if ($field[7] ne $main::mycall) {
- $self->route($field[7], $line);
- return;
- }
- }
-
- # if this is a 'nodx' node then ignore it
- if ($badnode->in($field[7])) {
- dbg("PCPROT: Bad Node, dropped") if isdbg('chanerr');
- return;
- }
-
- # if this is a 'bad spotter' user then ignore it
- if ($badspotter->in($field[6])) {
- dbg("PCPROT: Bad Spotter, dropped") if isdbg('chanerr');
- return;
- }
-
- # convert the date to a unix date
- my $d = cltounix($field[3], $field[4]);
- # bang out (and don't pass on) if date is invalid or the spot is too old (or too young)
- if (!$d || ($pcno == 11 && ($d < $main::systime - $pc11_max_age || $d > $main::systime + 900))) {
- dbg("PCPROT: Spot ignored, invalid date or out of range ($field[3] $field[4])\n") if isdbg('chanerr');
- return;
- }
-
- # is it 'baddx'
- if ($baddx->in($field[2])) {
- dbg("PCPROT: Bad DX spot, ignored") if isdbg('chanerr');
- return;
- }
-
- # do some de-duping
- $field[5] =~ s/^\s+//; # take any leading blanks off
- $field[2] = unpad($field[2]); # take off leading and trailing blanks from spotted callsign
- if ($field[2] =~ /BUST\w*$/) {
- dbg("PCPROT: useless 'BUSTED' spot") if isdbg('chanerr');
- return;
- }
- if (Spot::dup($field[1], $field[2], $d, $field[5])) {
- dbg("PCPROT: Duplicate Spot ignored\n") if isdbg('chanerr');
- return;
- }
- if ($censorpc) {
- my @bad;
- if (@bad = BadWords::check($field[5])) {
- dbg("PCPROT: Bad words: @bad, dropped") if isdbg('chanerr');
- return;
- }
- }
-
- my @spot = Spot::prepare($field[1], $field[2], $d, $field[5], $field[6], $field[7]);
- # global spot filtering on INPUT
- if ($self->{inspotsfilter}) {
- my ($filter, $hops) = $self->{inspotsfilter}->it(@spot);
- unless ($filter) {
- dbg("PCPROT: Rejected by input spot filter") if isdbg('chanerr');
- return;
- }
- }
-
- # add it
- Spot::add(@spot);
-
- #
- # @spot at this point contains:-
- # freq, spotted call, time, text, spotter, spotted cc, spotters cc, orig node
- # then spotted itu, spotted cq, spotters itu, spotters cq
- # you should be able to route on any of these
- #
-
- # fix up qra locators of known users
- my $user = DXUser->get_current($spot[4]);
- if ($user) {
- my $qra = $user->qra;
- unless ($qra && DXBearing::is_qra($qra)) {
- my $lat = $user->lat;
- my $long = $user->long;
- if (defined $lat && defined $long) {
- $user->qra(DXBearing::lltoqra($lat, $long));
- $user->put;
- }
- }
-
- # send a remote command to a distant cluster if it is visible and there is no
- # qra locator and we havn't done it for a month.
-
- unless ($user->qra) {
- my $node;
- my $to = $user->homenode;
- my $last = $user->lastoper || 0;
- if ($to ne $main::mycall && $send_opernam && $main::systime > $last + $DXUser::lastoperinterval && $to && ($node = Route::Node::get($to)) ) {
- my $cmd = "forward/opernam $spot[4]";
- # send the rcmd but we aren't interested in the replies...
- my $dxchan = $node->dxchan;
- if ($dxchan && $dxchan->is_clx) {
- route(undef, $to, pc84($main::mycall, $to, $main::mycall, $cmd));
- } else {
- route(undef, $to, pc34($main::mycall, $to, $cmd));
- }
- if ($to ne $field[7]) {
- $to = $field[7];
- $node = Route::Node::get($to);
- if ($node) {
- $dxchan = $node->dxchan;
- if ($dxchan && $dxchan->is_clx) {
- route(undef, $to, pc84($main::mycall, $to, $main::mycall, $cmd));
- } else {
- route(undef, $to, pc34($main::mycall, $to, $cmd));
- }
- }
- }
- $user->lastoper($main::systime);
- $user->put;
- }
- }
- }
-
- # local processing
- my $r;
- eval {
- $r = Local::spot($self, @spot);
- };
-# dbg("Local::spot1 error $@") if isdbg('local') if $@;
- return if $r;
-
- # DON'T be silly and send on PC26s!
- return if $pcno == 26;
-
- # send out the filtered spots
- send_dx_spot($self, $line, @spot) if @spot;
- return;
- }
-
- if ($pcno == 12) { # announces
- # announce duplicate checking
- $field[3] =~ s/^\s+//; # remove leading blanks
- if (AnnTalk::dup($field[1], $field[2], $field[3])) {
- dbg("PCPROT: Duplicate Announce ignored") if isdbg('chanerr');
- return;
- }
-
- if ($censorpc) {
- my @bad;
- if (@bad = BadWords::check($field[3])) {
- dbg("PCPROT: Bad words: @bad, dropped") if isdbg('chanerr');
- return;
- }
- }
-
- if ($field[2] eq '*' || $field[2] eq $main::mycall) {
-
- # global ann filtering on INPUT
- if ($self->{inannfilter}) {
- my ($ann_dxcc, $ann_itu, $ann_cq, $org_dxcc, $org_itu, $org_cq) = (0..0);
- my @dxcc = Prefix::extract($field[1]);
- if (@dxcc > 0) {
- $ann_dxcc = $dxcc[1]->dxcc;
- $ann_itu = $dxcc[1]->itu;
- $ann_cq = $dxcc[1]->cq();
- }
- @dxcc = Prefix::extract($field[5]);
- if (@dxcc > 0) {
- $org_dxcc = $dxcc[1]->dxcc;
- $org_itu = $dxcc[1]->itu;
- $org_cq = $dxcc[1]->cq();
- }
- my ($filter, $hops) = $self->{inannfilter}->it(@field[1..6], $self->{call},
- $ann_dxcc, $ann_itu, $ann_cq, $org_dxcc, $org_itu, $org_cq);
- unless ($filter) {
- dbg("PCPROT: Rejected by input announce filter") if isdbg('chanerr');
- return;
- }
- }
-
- # send it
- $self->send_announce($line, @field[1..6]);
- } else {
- $self->route($field[2], $line);
- }
-
- return;
- }
-
- if ($pcno == 13) {
- last SWITCH;
- }
- if ($pcno == 14) {
- last SWITCH;
- }
- if ($pcno == 15) {
- last SWITCH;