add more code gradually
[spider.git] / perl / QXProt.pm
index b277e3bd910687f8f6b4602a47122287a3c72dfa..689432d017bc4f54e2e019e97e8e6923f0235eea 100644 (file)
@@ -30,19 +30,20 @@ use Route::Node;
 use Script;
 use DXProt;
 use Verify;
+use Thingy;
 
 use strict;
 
 use vars qw($VERSION $BRANCH);
 $VERSION = sprintf( "%d.%03d", q$Revision$ =~ /(\d+)\.(\d+)/ );
-$BRANCH = sprintf( "%d.%03d", q$Revision$ =~ /\d+\.\d+\.(\d+)\.(\d+)/ ) || 0;
+$BRANCH = sprintf( "%d.%03d", q$Revision$ =~ /\d+\.\d+\.(\d+)\.(\d+)/,(0,0));
 $main::build += $VERSION;
 $main::branch += $BRANCH;
 
 sub init
 {
        my $user = DXUser->get($main::mycall);
-       $DXProt::myprot_version += $main::version*100;
+       $DXProt::myprot_version += ($main::version - 1 + 0.52)*100;
        $main::me = QXProt->new($main::mycall, 0, $user); 
        $main::me->{here} = 1;
        $main::me->{state} = "indifferent";
@@ -67,7 +68,7 @@ sub sendinit
 {
        my $self = shift;
        
-       $self->send($self->gen1);
+       $self->node_update;
 }
 
 sub normal
@@ -76,54 +77,49 @@ sub normal
                DXProt::normal(@_);
                return;
        }
-       my ($id, $fromnode, $msgid, $incs);
-       return unless ($id, $fromnode, $msgid, $incs) = $_[1] =~ /^QX(\d\d)\^([-A-Z0-9]+)\^([0-9A-F]{1,4})\^.*\^([0-9A-F]{2})$/;
 
-       my $noderef = Route::Node::get($fromnode);
-       $noderef = Route::Node::new($fromnode) unless $noderef;
-       my $user = DXChannel->get_current($fromnode);
-
-       my $il = length $incs; 
-       my $cs = sprintf("%02X", unpack("%32C*", substr($_[1], 0, length($_[1]) - ($il+1))));
-       if ($incs ne $cs) {
-               dbg("QXPROT: Checksum fail in: $incs ne calc: $cs" ) if isdbg('chanerr');
-               return;
-       }
-
-       return unless $noderef->newid($msgid);
-
-       $_[0]->handle($id, $fromnode, $msgid, $_[1]);
-       return;
-}
-
-sub handle
-{
-       no strict 'subs';
-       my $self = shift;
-       my $id = 0 + shift;
-       my $sub = "handle$id";
-       $self->$sub($self, @_) if $self->can($sub);
-       return;
-}
+       # Although this is called the 'QX' Protocol, this is historical
+       # I am simply using this module to save a bit of time.
+       # 
+       
+       return unless my ($tonode, $fromnode, $class, $msgid, $hoptime, $rest) = 
+               $_[1] =~ /^([^;]+;){5,5}\|(.*)$/;
 
-sub gen
-{
-       no strict 'subs';
        my $self = shift;
-       my $id = 0 + shift;
-       my $sub = "gen$id";
-       $self->$sub($self, @_) if $self->can($sub);
+       
+       # add this interface's hop time to the one passed
+       my $newhoptime = $self->{pingave} >= 999 ? 
+               $hoptime+10 : ($hoptime + int($self->{pingave}*10));
+       # split up the 'rest' which are 'a=b' pairs separated by commas
+    # and create a new thingy based on the class passed (if known)
+       # ignore pairs with a leading '_'.
+
+       my @par;
+
+       for (split /;/, $rest) {
+               next if /^_/;
+               next unless /^\w+=/;
+               s/\%([0-9A-F][0-9A-F])/chr(hex($1))/eg;
+               push @par, split(/=/,$_,2);
+       }
+       no strict 'refs';
+       my $pkg = 'Thingy::' . lcfirst $class;
+       my $t = $pkg->new(_tonode=>$tonode, _fromnode=>$fromnode,
+                                         _msgid=>$msgid, _hoptime=>$newhoptime,
+                                         _newdata=>$rest, _inon=>$self->{call},
+                                         @par) if defined *$pkg && $pkg->can('new');
+       $t->queue if $t;
        return;
 }
 
 my $last_node_update = 0;
-my $node_update_interval = 60*15;
+my $node_update_interval = 60*60;
 
 sub process
 {
        if ($main::systime >= $last_node_update+$node_update_interval) {
-#              sendallnodes();
-#              sendallusers();
                $last_node_update = $main::systime;
        }
 }
@@ -131,100 +127,64 @@ sub process
 sub disconnect
 {
        my $self = shift;
+       my $t = Thingy::Route->new_node_disconnect($main::mycall, $main::mycall, $self->{call});
+       $t->queue;
        $self->DXProt::disconnect(@_);
 }
 
-sub sendallnodes
-{
-}
-
-sub sendallusers
-{
-
-}
-
 my $msgid = 1;
 
-sub frame
+sub nextmsgid
 {
-       my $pcno = shift;
-       my $ht;
-       
-       $ht = sprintf "%X", $msgid;
-       my $line = join '^', sprintf("QX%02d", $pcno), $main::mycall, $ht, @_;
-       my $cs = sprintf "%02X", unpack("%32C*", $line) & 255;
-       $msgid = 1 if ++$msgid > 0xffff;
-       return "$line^$cs";
+       my $r = $msgid;
+       $msgid = 1 if ++$msgid > 99999;
+       return $r;
 }
 
-sub handle1
+sub node_update
 {
-       my $self = shift;
-       
-       my @f = split /\^/, $_[2];
-       my $inv = Verify->new($f[5]);
-       unless ($inv->verify($main::me->user->passphrase, $f[6], $main::mycall, $self->call)) {
-               $self->sendnow('D','Sorry...');
-               $self->disconnect;
-       }
-       if ($self->{outbound}) {
-               $self->send($self->gen1);
-       } 
-       if ($self->{sort} ne 'S' && $f[2] eq 'DXSpider') {
-               $self->{user}->{sort} = $self->{sort} = 'S';
-               $self->{user}->{priv} = $self->{priv} = 1 unless $self->{priv};
-       }
-       $self->{version} = $f[3];
-       $self->{build} = $f[4];
-       $self->state('normal');
-       $self->{lastping} = 0;
+       my $t = Thingy::Route->new_node_update();
+       $t->queue if $t;
 }
 
-sub gen1
-{
-       my $self = shift;
-       my $inp = Verify->new;
-       return frame(1, 1, "DXSpider", $main::version + 53, $main::build, $inp->challenge, $inp->response($self->user->passphrase, $self->call, $main::mycall));
-}
-
-sub handle2
-{
-
-}
-
-sub gen2
+sub t_send
 {
        my $self = shift;
+       my $t = shift;
+       confess('$t is not a Thingy') unless $t->isa('Thingy');
        
-       my $node = shift;
-       my $sort = shift;
-       my @out;
-       my $dxchan;
-       
-       while (@_) {
-               my $str = '';
-               for (; @_ && length $str <= 230;) {
-                       my $ref = shift;
-                       my $call = $ref->call;
-                       my $flag = 0;
-                       
-                       $flag += 1 if $ref->here;
-                       $flag += 2 if $ref->conf;
-                       if ($ref->is_node) {
-                               my $ping = int($ref->pingave * 10);
-                               $str .= "^N$flag$call,$ping";
-                               my $v = $ref->build || $ref->version;
-                               $str .= ",$v" if defined $v;
-                       } else {
-                               $str .= "^U$flag$call";
+       # manufacture the protocol line if required
+       unless (exists $t->{_newprot}) {
+               my ($class) = ref $self =~ /::(\w+)$/;
+               unless (exists $t->{_rest}) {
+                       $t->{_rest} = "";
+                       while (my ($k,$v) = each %$t) {
+                               next if $k =~ /^_/;
+                               if (ref $v && @$v) {
+                                       my $val = "";
+                                       for(@$v) {
+                                               my $vv = $_;
+                                               $vv =~ s/([\%;=,\x00-\x1f\x7f-\xff])/sprintf("%%%02X", ord($1))/eg;
+                                               $val .= $vv . ',';
+                                       }
+                                       if (length $val) {
+                                               chop $val;
+                                               $t->{_rest} .= "$k=$val;";
+                                       }
+                               } elsif (length $v) {
+                                       $v =~ s/([\%;=\x00-\x1f\x7f-\xff])/sprintf("%%%02X", ord($1))/eg;
+                                       $t->{_rest} .= "$k=$v;";
+                               }
                        }
+                       chop $t->{_rest} if length $t->{_rest};
                }
-               push @out, $str if $str;
+               
+               $t->{_hoptime} ||= 1;
+               $t->{_msgid} = nextmsgid() unless $t->{_msgid};
+               $t->{_newprot} = join(';', $t->{_tonode}, $t->{_fromnode}, uc $class,
+                                                         $t->{_msgid}, $t->{_hoptime}) . '|' . $t->{_rest};
        }
-       my $n = @out;
-       my $h = get_hops(90);
-       @out = map { sprintf "PC90^%s^%X^%s%d%s^%s^", $node->call, $main::systime, $sort, --$n, $_, $h } @out;
-       return @out;
+       $self->SUPER::send($t->{_newprot});
 }
 
 1;