ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/Net-Knuddels/Net/Knuddels.pm
Revision: 1.5
Committed: Wed Jan 12 19:26:51 2005 UTC (19 years, 4 months ago) by root
Branch: MAIN
Changes since 1.4: +14 -1 lines
Log Message:
*** empty log message ***

File Contents

# Content
1 package Net::Knuddels;
2
3 use Net::Knuddels::Dictionary;
4
5 use strict;
6 use utf8;
7
8 use Carp;
9 use Math::BigInt;
10
11 sub hash_pw($$) {
12 my ($challenge, $pw) = @_;
13
14 my $l1 = length $pw;
15 my $l2 = length $challenge;
16
17 my $k = chr ($l1 ^ ($l2 << 4));
18
19 my $l = $l1 < $l2 ? $l2 : $l1;
20
21 my $xor = substr +($pw x 100) ^ ($challenge x 100) ^ ($k x 100), 0, $l;
22
23 my ($i, $j);
24
25 --$l;
26
27 if ($l <= 17) {
28 for (0 .. $l) {
29 $i = $i * 3 + ord substr $xor, $l - $_;
30 $j = $j * 5 + ord substr $xor, $_;
31
32 $i = unpack "l", pack "L", (new Math::BigInt $i) & 0xffffffff;
33 $j = unpack "l", pack "L", (new Math::BigInt $j) & 0xffffffff;
34 }
35 } else {
36 for ($_ = $l; $_ >= 0; $_ -= int $_/19) {
37 $i = $i * 5 + ord substr $xor, $_;
38 $j = $j * 3 + ord substr $xor, $l - $_;
39
40 $i = unpack "l", pack "L", (new Math::BigInt $i) & 0xffffffff;
41 $j = unpack "l", pack "L", (new Math::BigInt $j) & 0xffffffff;
42 }
43 }
44
45 $i ^= $j;
46
47 ($i & 0xffffff) ^ ($i >> 24);
48 }
49
50 package Net::Knuddels::Protocol;
51
52 =head2 CLASS Net::Knuddels::Protocol
53
54 You B<must> call the C<destroy> method of this class when you no longer
55 use it, as circular references will keep the object alive otherwise.
56
57 =cut
58
59 sub new {
60 my $class = shift;
61
62 my %data;
63
64 my $self = bless {
65 @_
66 }, $class;
67
68 $self->register ("(" => sub {
69 $self->{challenge} = $_[1];
70 $self->{room} = $_[2];
71 $self->feed_event ("connected");
72 });
73
74 $self;
75 }
76
77 sub feed_data($$) {
78 my ($self, $data) = @_;
79
80 # split data stream into packets
81
82 $data = "$self->{rbuf}$data";
83
84 while () {
85 1 <= length $data or last;
86 my $len = ord substr $data, 0, 1;
87
88 my $skip;
89 if ($len & 0x80) {
90 my $tail = (($len >> 5) & 3) - 1;
91 $len = ($len & 0x1f) + 1;
92
93 $tail < length $data or last;
94 $len += (ord substr $data, $_ + 1, 1) << ($_ * 8 + 5)
95 for 0 .. $tail;
96
97 $skip = 2 + $tail;
98 } else {
99 $skip = 1;
100 $len++;
101 }
102
103 $len + $skip <= length $data or last;
104 substr $data, 0, $skip, "";
105 my $msg = substr $data, 0, $len, "";
106
107 $self->feed_msg ($msg);
108 }
109
110 $self->{rbuf} = $data;
111 }
112
113 my $RE_dec = join "|", keys %$Net::Knuddels::Dictionary;
114
115 sub feed_msg($$) {
116 my ($self, $msg) = @_;
117
118 my $bin = unpack "b*", $msg;
119 my $res = "";
120
121 while ($bin =~ /\G($RE_dec)/cmog) {
122 my $frag = $Net::Knuddels::Dictionary->{$1};
123 $frag = pack "b*", substr $bin, 0, 16, "" if $frag eq "\\\\\\";
124 $res .= $frag;
125 }
126 $bin =~ /\G(.*[^0].*)$/ and die "Net::Knuddels::Receiver: undecodable message tail '$1'";
127
128 $self->feed_event (split /\0/, $res);
129 }
130
131 sub feed_event($@) {
132 my ($self, $type, @arg) = @_;
133
134 for ($type, "ALL") {
135 my $ev = $self->{cb}{$_};
136 $_->($type, @arg) for values %$ev;
137 }
138 }
139
140 sub register {
141 my ($self, $type, $cb) = @_;
142
143 $self->{cb}{$type}{$cb} = $cb;
144 }
145
146 sub destroy {
147 my ($self) = @_;
148
149 delete $self->{cb};
150 }
151
152 1;
153