1 package Thrasher
::Protocol
::Purple
;
7 Thrasher::Protocol::Purple - test protocol for Thrasher Bird
11 This package is the base class for protocols using libpurple.
12 Each protocol may have slightly different details w.r.t. how they work
13 with Thrasher, which is handled by specializing them further,
14 such as with Thrasher::Protocol::Purple::AIM. As you find such
15 differences, please be sure to add hooks in here, not hack on
20 use base
'Thrasher::Protocol';
22 use Thrasher
::Component
qw(strip_resource);
23 use Thrasher
::Log
qw(log logger debug);
25 use Thrasher
::Session
;
26 use Thrasher
::XMPPStreamIn
;
27 use Glib
qw(G_PRIORITY_DEFAULT);
33 use Thrasher
::Protocol
::Purple
::Vars
qw(:all);
37 use Thrasher
::Plugin
::Vcard
;
38 use Thrasher
::Plugin
::EntityCapabilities
;
43 THPPW
::thrasher_purple_debug
($DEBUG);
45 # Initialize the wrapper
46 THPPW
::thrasher_wrapper_init
47 (Thrasher
::error_wrap
("timeout_add", \
&_timeout_add
),
48 Thrasher
::error_wrap
("input_add", \
&_input_add
),
49 Thrasher
::error_wrap
("source_remove", \
&_source_remove
),
50 Thrasher
::error_wrap
("incoming_msg", \
&_incoming_msg
),
51 Thrasher
::error_wrap
("presence_in", \
&_presence_in
),
52 Thrasher
::error_wrap
("subscription_add", \
&_subscription_add
),
53 Thrasher
::error_wrap
("legacy_user_add_user", \
&_legacy_user_add_user
),
54 Thrasher
::error_wrap
("connection_error", \
&_connection_error
),
55 Thrasher
::error_wrap
("connection", \
&_connection
),
56 Thrasher
::error_wrap
("incoming_chatstate", \
&_incoming_chatstate
),
59 # Initialize the remainder
60 THPPW
::thrasher_init
();
62 # Globalize component object so we can receive messages
63 our $global_component;
66 return 'Thrasher::EventLoop::Glib';
69 # This internal routine allows us to group the initial
70 # subscribe additions and throw them back.
74 if (! $session || !$session->{'jid'}) {
75 debug
("_initial_roster: No session. Must be post-logoff?");
79 debug
("_initial_roster($session->{jid}) called\n");
83 foreach my $username (keys %{$session->{initial_roster
}}) {
84 $username = $session->{protocol
}->process_remote_username($username);
85 $roster->{$username} = Thrasher
::Roster
::subscribed
();
86 # We should also handle presence info here (?)
89 $session->{protocol
}->set_current_legacy_roster($session, $roster);
91 delete $session->{initial_roster_timeout_active
};
92 $session->{initial_roster
} = [];
94 # Thrasher can now be sure the protocol is completely online and
95 # ready to e.g. send IMs queued while login was in progress.
96 if ($session->{'protocol_state'} eq 'logging in') {
97 $session->{'protocol'}->set_session_state($session, 'online');
100 # We don't want the timeout to loop, so destroy with 0
104 # This appears to only be called for things on our initial roster.
105 # FIXME: If that's true, change this.
106 sub _subscription_add
{
107 my $orig_jid = shift;
108 my $orig_sender = shift;
111 my $jid = Encode
::decode
("UTF-8", $orig_jid);
112 my $sender = Encode
::decode
("UTF-8", $orig_sender);
114 debug
("_subscription_add($orig_jid, $orig_sender) called\n");
116 my $session = $global_component->session_for($jid);
118 # Set a timeout if we have no previous jid information
119 if (not defined $session->{initial_roster
}) {
121 # Set a flag so we don't push subscription additions
123 $session->{initial_roster_timeout_active
} = 1;
125 # Heuristically manage our initial roster as we cannot
126 # tell when the libpurple protocols are done giving us the
127 # subscribe user list.
128 Glib
::Timeout
->add(5000,
134 # Verify we're actually within a timeout
135 if ($session->{initial_roster_timeout_active
}) {
136 # This is a bit ugly, but it allows us to bind sender/status
137 # info to JIDs for timeouts
138 $session->{initial_roster
}{$sender} = $status;
140 # We aren't in a timeout, we need to send new subscribe info up
144 # Subscription-in information has the presence information
145 # loaded onto it too, at least for AIM
147 _presence_in
($orig_jid, $orig_sender, undef, undef, $status);
150 debug
("_subscription_add done\n");
155 sub _legacy_user_add_user
{
156 my $jid_target = shift;
157 my $legacy_username_adding = shift;
159 $jid_target = Encode
::decode
("UTF-8", $jid_target);
160 $legacy_username_adding = Encode
::decode
("UTF-8", $legacy_username_adding);
162 log("$legacy_username_adding requesting add for $jid_target");
164 my $session = $global_component->session_for($jid_target);
167 $session->{protocol
}->adding_contact($legacy_username_adding,
170 log("Got request to add user $jid_target, but $jid_target is "
175 sub process_message
{ return $_[1]; }
176 sub process_remote_username
{ return $_[1]; }
178 # Callback of presence in
187 $jid = Encode
::decode
("UTF-8", $jid);
188 $sender = Encode
::decode
("UTF-8", $sender);
189 $alias = Encode
::decode
("UTF-8", $alias);
190 $group = Encode
::decode
("UTF-8", $group);
191 $status = Encode
::decode
("UTF-8", $status);
192 $message = Encode
::decode
("UTF-8", $message);
194 debug
("_presence_in($jid, $sender, $status) called\n");
196 my $session = $global_component->session_for($jid);
197 my $self = $session->{protocol
};
198 if (!defined($self)) {
199 debug
("No session defined for $jid, must be post-logoff?");
203 my $clean_sender = $self->process_remote_username($sender);
206 $message =~ s/([\x80-\xff])/'&#' . ord($1) . ';'/ge;
209 # Nothing is done with protocol?
210 my $xmpp_presence = $purple_presence_to_xmpp{$status};
211 if ($xmpp_presence) {
212 my ($type, $show) = @
{$xmpp_presence};
213 $self->legacy_presence_update($session,
220 log("Unknown presence status of $status was sent by "
221 ."$clean_sender to $jid.");
225 debug
("_presence_in done\n");
229 sub _incoming_chatstate
{
230 my ($orig_jid, $orig_sender, $state) = @_;
231 debug
("_incoming_chatstate($orig_jid, $orig_sender, $state) called\n");
234 # loosely <http://xmpp.org/extensions/xep-0085.html>
235 if ($state == $THPPW::PURPLE_TYPING
) {
236 $state_tag = 'composing';
238 elsif ($state == $THPPW::PURPLE_TYPED
) {
239 $state_tag = 'paused';
241 elsif ($state == $THPPW::PURPLE_NOT_TYPING
) {
242 $state_tag = 'inactive';
248 my $jid = Encode
::decode
('UTF-8', $orig_jid);
249 my $sender = Encode
::decode
('UTF-8', $orig_sender);
251 my $session = $global_component->session_for($jid);
253 debug
("No session?!!\n");
256 if (! $session->{'protocol'}) {
257 debug
("No session protocol?!!\n");
260 $session->{'protocol'}->incoming_chatstate($session, $sender, $state_tag);
266 my $jid = Encode
::decode
("UTF-8", $orig_jid);
267 debug
("_connection($jid) called\n");
269 my $session = $global_component->session_for($jid);
271 # Component::logout and thrasher.c:thrasher_logout() will
272 # happily destroy the session and thrasher_connection while
273 # libpurple is waiting asynchronously for connection events.
274 # Once the connection completes and libpurple starts firing
275 # callbacks, weird errors arise because the session is gone
276 # and Thrasher has lost track of what is connected.
278 # Maybe we should reject the logout and defer it to
279 # _connection{,_error}, relying on one of them always being
281 log("_connection($jid): No session? Assuming already logged out.");
282 # Ensure the thrasher_connection gets gone.
283 Glib
::Timeout
->add(1,
285 # Log off just after logon finishes, not during.
287 # Turns out purple_connection_set_state()
288 # (which called the connected ui_op) crashes
289 # if prpl_info is yanked out from under it.
290 THPPW
::thrasher_action_logout
($orig_jid);
297 my $protocol = $session->{'protocol'};
299 log("_connection($jid): No protocol?!!");
303 $session->{'purple_connection_created'} = 1;
304 delete($protocol->{'connection_started_at'}->{$jid});
305 my $continuation = delete($session->{'connection_cb'});
307 $continuation->($session);
310 log("_connection($jid): No connection_cb?!!");
314 # But libpurple prpl might not be ready to send IMs queued during
315 # login. Wait until _initial_roster() for online protocol_state.
317 # If after no _subscription_add()/_initial_roster() happens
318 # (perhaps the account has no current legacy roster at all?)
319 # ensure session is eventually set online anyway.
322 if (! $session->{'initial_roster_timeout_active'}
323 && ! $session->{'initial_roster'}
324 && $session->{'protocol_state'} eq 'logging in') {
325 debug
("Never called _initial_roster($session->{jid})?\n");
326 $protocol->set_session_state($session, 'online');
328 return 0; # No repeat
337 sub _connection_error
{
339 my $error_code = shift;
342 $jid = Encode
::decode
("UTF-8", $jid);
343 $message = Encode
::decode
("UTF-8", $message);
345 debug
("_connection_error($jid)\n");
347 my $session = $global_component->session_for($jid);
349 log("No session?!! Error was $error_code/'$message'.");
352 my $protocol = $session->{protocol
};
354 # Clear connection state.
355 delete($protocol->{'connection_started_at'}->{$jid});
358 log("_connection($jid): No protocol?!!");
362 my $attempt_reconnect = 0;
365 if ($session->{status
} =~ /disconnecting/) {
366 log("Got error code $error_code, but ignoring it since "
367 ."we're in the middle of disconnecting.");
371 # Some of these cases are poorly tested since it's either
372 # hard or borderline impossible for them to occur.
373 # We also have to think about whether to attempt reconnection
375 switch
($error_code) {
376 case
($ERROR_NETWORK_ERROR) {
377 $protocol->network_error($jid);
378 $error = "Network error, attempting reconnection";
379 $attempt_reconnect = 1;
381 case
($ERROR_INVALID_USERNAME) {
382 $protocol->invalid_username($jid);
383 $error = "Remote server reports invalid username; please reregister";
385 case
($ERROR_AUTHENTICATION_FAILED) {
386 $protocol->wrong_authentication($jid);
387 $error = "Username or password invalid; please register with correct information";
389 case
($ERROR_AUTHENTICATION_IMPOSSIBLE) {
390 $protocol->_handle_error
391 ($jid, 'Thrasher Bird can not negotiate an '
392 .'authentication technique with the remote '
393 .'service', 'service_unavailable');
394 # This is a bad one, we don't know what to do.
395 $error = "Authentication impossible";
397 case
($ERROR_NO_SSL_SUPPORT) {
398 $protocol->_handle_error
399 ($jid, 'libpurple was compiled without SSL '
400 .'support, but SSL is required by the '
401 .'remote service.', 'service_unavailable');
402 $error = "Thrasher Bird is unable to connect";
404 case
($ERROR_ENCRYPTION_ERROR) {
405 $protocol->_handle_error
406 ($jid, 'There was an error negotiating SSL with '
407 .'the remote service, or the remote service '
408 .'does not support encryption but an account '
409 .'option was set to require it.',
410 'service_unavailable');
411 $error = "Thrasher Bird is unable to connect";
413 case
($ERROR_NAME_IN_USE) {
414 $protocol->name_in_use($jid);
415 $error = "The remote service reports your username is in use";
417 case
($ERROR_INVALID_SETTINGS) {
418 $protocol->invalid_username($jid);
419 $error = "Remote server reports invalid username; please reregister";
421 case
($ERROR_OTHER_ERROR) {
422 my $error_message = "Unknown connection error.";
424 $error_message .= ' The legacy service reported: '
427 $protocol->_handle_error
428 ($jid, $error_message, 'internal_server_error');
431 log("Got connection error: $error_code for $jid");
435 # This needs to be kept in sync with libpurple's
436 # connection.c -> purple_connection_is_fatal, which
437 # tracks whether libpurple is going to automatically
438 # log out our connection in purple_connection_error_reason.
439 my $purple_will_kill = !($error_code == $ERROR_NETWORK_ERROR ||
440 $error_code == $ERROR_ENCRYPTION_ERROR);
441 $session->{purple_will_kill
} = $purple_will_kill;
442 $session->{purple_will_kill
} ||= $protocol->purple_forces_kill;
444 $protocol->{component
}->logout($session, undef,
447 # Probe the user's presence to trigger a re-connect attempt
448 # if they are still online. They may have gone offline in the
449 # meantime, in which case we don't want to reconnect.
450 if ($attempt_reconnect) {
451 my $full_jid = $session->{full_jid
};
453 $protocol->{component
}->send_presence_xml($full_jid,
458 log("Going to attempt reconnect in 15 seconds for $session->{full_jid}");
460 _timeout_add
(15000, $callback, undef, "Reconnect $session->{full_jid}");
463 my $continuation = delete($session->{'connection_cb'});
465 $continuation->(undef);
468 log("_connection_error($jid): No connection_cb?!!");
471 # If you want C-end handling, we need to throw some returns above
475 # Callback for incoming messages
477 my ($jid, $sender, $alias, $message, $flags) = @_;
479 $jid = Encode
::decode
("UTF-8", $jid);
480 $sender = Encode
::decode
("UTF-8", $sender);
481 $message = Encode
::decode
("UTF-8", $message);
483 debug
("_incoming_msg from $sender for $jid\n");
485 my $session = $global_component->session_for($jid);
486 my $protocol = $session->{protocol
};
488 my $clean_sender = $session->{protocol
}->process_remote_username($sender);
490 $message =~ s/([\x80-\xff])/'&#' . ord($1) . ';'/ge;
492 if ($flags & $THPPW::PURPLE_MESSAGE_AUTO_RESP
) {
493 $message = '(auto-reply) ' . $message;
496 # Type is currently hard coded...
497 $protocol->sending_message($clean_sender, $session->{legacy_login
},
500 debug
("_incoming_msg done\n");
502 # Thrasher::Protocol::sending_message currently has no returned value
509 my $registration_info = shift;
511 debug
("###registration($jid) called");
513 # As a special case, if the registration info's username is
514 # "fail", we return an error given by $registration_info->{password}.
515 if ($registration_info->{username
} eq 'fail') {
516 return 0, $registration_info->{password
};
518 return $self->SUPER::registration
($jid, $registration_info);
522 # This really should be overridden
523 sub name
{ 'Purple' }
525 sub identifier
{ 'aim' }
527 # This method identifies which protocol we're using in Pidgin.
531 die "prpl not set up for " . ref($self);
534 sub create_login_session
{
536 my $continuation = shift;
537 my $registration_info = shift;
538 my $full_jid = shift;
539 my $component = shift;
540 my $jid = strip_resource
($full_jid);
542 debug
("###create_login_session($full_jid)");
544 # FIXME: Check for existing logins.
545 my $session = new Thrasher
::Session
($full_jid,
548 $registration_info->{username
});
549 $global_component = $component;
550 $self->set_session_state($session, 'logging in');
551 $component->set_session_for($jid, $session);
553 for my $key (keys %$registration_info) {
554 $registration_info->{$key} =
555 Encode
::encode
("UTF-8", $registration_info->{$key});
558 if (!$self->valid_id($registration_info->{username
}) ||
559 !$self->valid_password($registration_info->{password
})) {
560 $self->wrong_authentication($full_jid);
561 $continuation->('not_acceptable');
562 $component->logout($session);
566 my $jid_enc = Encode
::encode
('UTF-8', $jid);
569 proto
=> $self->prpl,
571 %{ $self->{'configuration'}->{'extra_login_args'} || {} },
573 my $login_error = THPPW
::thrasher_action_login
(\
%login_args);
574 my $last_connection_started_at = $self->{'connection_started_at'}->{$jid};
576 # PurpleAccount already exists. But if component called here,
577 # the session must already be gone. Thus, must have logged out
578 # during the async libpurple connection attempt and now trying
580 if ($login_error == 2
581 && $last_connection_started_at
582 && time() - $last_connection_started_at > 600) {
583 # Async libpurple login started more than 10 minutes ago but
584 # _connection{,_error} has still not come back. Destroy the
585 # old login attempt and start a new one.
587 # E.g. the PURPLE_CONNECTED state was never reached due to a
588 # MSN ADL/FQY counting bug?
589 debug
('Discarding aged PurpleAccount attempt from '
590 . $last_connection_started_at);
591 THPPW
::thrasher_action_logout
($jid_enc);
592 $login_error = THPPW
::thrasher_action_login
(\
%login_args);
593 # In theory, logout removed the PurpleAccount so the new
594 # $login_error can't be 2. But--don't risk it!
596 if ($login_error == 2) {
597 # Reject for now. Eventually _connection or _connection_error
598 # will come back and login attempts will be possible again.
600 # Must not be confused with the bad credentials case lest
601 # Component put the failure in authentication_errors and lock
602 # logins until the registration changes.
604 # Could have this session "take over" the PurpleAccount, but
605 # what if credentials differ? Or if libpurple never finishes?
606 $continuation->('conflict', 1);
607 $component->logout($session);
608 if (not $self->{'connection_started_at'}->{$jid}) {
609 $self->{'connection_started_at'}->{$jid} = time();
614 elsif ($login_error != 0) {
615 # Rejected before we're even trying to connect pretty
616 # much means syntactically invalid credentials
617 $continuation->('not_acceptable');
618 $component->logout($session);
622 $self->{'connection_started_at'}->{$jid} = time();
623 $session->{'connection_cb'} = $continuation;
630 debug
("###initial_login called");
632 $session->{logged_in
} = 1;
639 # FIXME: Can occur if the first action after aim.transport comes
640 # online is to unregister.
641 if ($global_component) {
642 my $session = $global_component->session_for($jid);
644 $self->{component
}->logout($session);
648 log("What? No \$global_component in remove?!?");
651 # A user who attempted to unregister while the transport was
652 # offline won't log in when it comes back up (and thus doesn't
653 # need to log out) but might still be registered with the backend.
654 $self->{backend
}->remove($jid);
660 my $target_name = shift;
661 my $continuation = shift;
663 debug
("###subscribe($session->{jid}, $target_name) called");
665 $session->{subscribed
}->{$target_name} = 1;
667 THPPW
::thrasher_action_buddy_add
(Encode
::encode
("UTF-8",
669 Encode
::encode
("UTF-8", $target_name));
677 my $target_name = shift;
678 my $continuation = shift;
680 debug
("###unsubscribe($session->{jid}, $target_name) called");
682 if (!(delete $session->{subscribed
}->{$target_name})) {
683 print STDERR
"Warning, removing nonexistant contact\n";
686 THPPW
::thrasher_action_buddy_remove
(Encode
::encode
("UTF-8",
688 Encode
::encode
("UTF-8", $target_name));
695 my ($session, $continuation) = @_;
697 debug
("###logout($session->{jid}) called");
699 if ($session->{purple_connection_created
}
700 && !$session->{purple_will_kill
}) {
701 THPPW
::thrasher_action_logout
(Encode
::encode
("UTF-8", $session->{jid
}));
703 elsif (! $session->{purple_connection_created
}) {
704 debug
('No purple connection created to log out.');
706 # Update component and ConnectionManager with the status of
707 # this connection attempt before the connection_cb
708 # continuation is thrown out. If the attempt does succeed,
709 # _connection() will immediately log it out anyway.
710 my $connection_cb = delete($session->{'connection_cb'});
711 if ($connection_cb) {
712 $connection_cb->(undef, 1);
716 $continuation->($session);
717 return $self->SUPER::logout
(@_);
720 sub debug_logged_in
{
721 my $component = $global_component;
723 debug
("No component?!!\n");
727 my $protocol = $component->{'protocol'};
729 debug
("No protocol?!!\n");
733 print STDERR
'prpl = ' . $protocol->prpl() . "\n";
735 if ($protocol->{'username_to_session'}) {
736 print STDERR
"protocol->username_to_session:\n";
737 while (my ($legacy_name, $session)
738 = each(%{$protocol->{'username_to_session'}})) {
739 print STDERR
"\t$legacy_name => $session\n";
743 debug
("No username_to_session?!!\n");
746 if ($component->{'sessions'}) {
747 print STDERR
"component->sessions:\n";
748 while (my ($jid, $session) = each(%{$component->{'sessions'}})) {
749 print STDERR
"\t$jid => $session\n";
753 debug
("No component sessions?!!\n");
756 THPPW
::thrasher_action_debug_logged_in
();
762 my ($session, $to, $body_text, $type, $error_sub) = @_;
764 debug
("###send_message called");
765 if ($session->{'protocol_state'} eq 'logging in') {
766 debug
("###send_message deferred; $session->{jid} still logging in.\n");
767 $session->on_connection_complete(sub {
768 $self->send_message(@orig_args);
773 $body_text = $self->process_message($body_text);
775 debug
("###Message From: ".$session->{jid
}.", To: $to, body: $body_text\n");
777 my $result = THPPW
::thrasher_action_outgoing_msg
778 (Encode
::encode
("UTF-8", $session->{jid
}),
779 Encode
::encode
("UTF-8", $to),
780 Encode
::encode
("UTF-8", $body_text));
781 debug
("Message send result: $result\n");
784 sub outgoing_chatstate
{
785 my ($self, $session, $to, $chatstate) = @_;
786 debug
("###outgoing_chatstate($session->{jid}, $to, $chatstate)\n");
788 our $chatstate_to_purple ||= {
789 'composing' => $THPPW::PURPLE_TYPING
,
790 'paused' => $THPPW::PURPLE_TYPED
,
791 'inactive' => $THPPW::PURPLE_NOT_TYPING
,
792 'active' => $THPPW::PURPLE_NOT_TYPING
,
794 my $purple_typing_state = $chatstate_to_purple->{$chatstate};
795 if (! defined($purple_typing_state)) {
796 debug
("Untranslated chatstate: '$chatstate'\n");
800 THPPW
::thrasher_action_outgoing_chatstate
($session->{'jid'},
802 $purple_typing_state);
809 my $component = shift;
810 my $legacy_username = shift;
812 debug
("###subscribed called: $legacy_username permitted for $session->{jid}");
814 THPPW
::thrasher_action_buddy_authorize
815 (Encode
::encode
("UTF-8", $session->{jid
}),
816 Encode
::encode
("UTF-8", $legacy_username));
818 $self->SUPER::subscribed
($session, $component, $legacy_username);
824 my $component = shift;
825 my $legacy_username = shift;
827 debug
("###unsubscribed($session->{jid}, $legacy_username) called");
829 if (!defined($legacy_username)) {
830 confess
"Unsubscribing an undef user; shouldn't be called.";
833 THPPW
::thrasher_action_buddy_deauthorize
834 (Encode
::encode
("UTF-8", $session->{jid
}),
835 Encode
::encode
("UTF-8", $legacy_username));
839 my ($self, $id) = @_;
841 THPPW
::thrasher_action_ft_ui_ready
($id);
842 return 1; # repeat this notification.
849 return "Gateway prompt";
855 return $self->{gateway_desc
};
858 sub user_presence_update
{
861 my $type = shift || '';
862 my $show = shift || '';
863 my $status = shift || '';
865 debug
("user_presence_update called\n");
869 # State table for type/show to purple_status
870 if ($show eq 'away') {
874 $purple_status = $purple_presence{'xaway'};
878 $purple_status = $purple_presence{'away'};
882 logger
("Unknown type/show of [$type/$show]");
885 elsif ($show eq 'chat' || $show eq '') {
888 # This seems like it might have more states
889 $purple_status = $purple_presence{'available'};
891 elsif ($type eq 'unavailable') {
893 $purple_status = $purple_presence{'offline'};
896 logger
("Unknown type/show of [$type/$show]");
899 elsif ($show eq 'xa' || $show eq 'xaway') {
900 $purple_status = $purple_presence{'xaway'};
902 elsif ($show eq 'dnd') {
903 $purple_status = $purple_presence{'unavailable'};
906 logger
("Unknown type/show of [$type/$show] (show is completely unrecognized)");
909 if (defined($purple_status)) {
910 THPPW
::thrasher_action_presence
(
911 Encode
::encode
("UTF-8", $session->{jid
}),
912 $purple_status, # integer does not need encoding
913 Encode
::encode
("UTF-8", $status),
917 #debug("User presence update: type: $type, show: $show, purple: $purple_status, status: $status");
920 # Don't do anything with this right now.
921 sub user_targeted_presence_update
{
925 my $type = shift || '';
926 my $show = shift || '';
927 my $status = shift || '';
928 my $target_user = shift || '';
930 #log("User presence update to $target_user: type: $type, show: $show, status: $status");
933 # Subrefs for which to satiate the libpurple monster
935 my $interval = shift;
939 debug
("perl::timeout_add called\n", 3);
941 debug
("\tinterval = $interval\n", 3) if $interval;
942 debug
("\tcode = $code\n", 3) if $code;
943 debug
("\ttrigger = $trigger\n", 3) if $trigger;
945 my $ret = Glib
::Timeout
->add($interval,
949 debug
("Glib::Timeout->add returned [$ret]\n", 3);
955 debug
("perl::timeout_remove called with $_[0]\n", 3);
957 return Glib
::Source
->remove($_[0]);
967 debug
("_input_add\n", 3);
971 debug
("\t$i = $_\n");
976 debug
("\tfd = $fd\n", 3) if $fd;
977 debug
("\tcond = $cond\n", 3) if $cond;
978 debug
("\tcode = $code\n", 3) if $code;
979 debug
("\ttrigger = $trigger\n", 3) if $trigger;
981 $cond = ($cond == 1) ?
'G_IO_IN' : 'G_IO_OUT';
983 $cond = [$cond, 'G_IO_ERR', 'G_IO_HUP', 'G_IO_NVAL'];
985 my $ret = Glib
::IO
->add_watch($fd,
991 debug
("Glib::IO->add_watch returned [$ret]\n", 3);
993 debug
("_input_add done\n", 3);
998 # Returns if the given ID is a valid id for the service. This avoids
999 # some problems that services have when you jam illegal logins in.
1000 # For instance, log in to Yahoo with a Japanese username, and it
1001 # just hangs on the connection, rather than doing anything.
1002 # Note that this is more about not sending in logins that confuse
1003 # the remote services so badly we get no errors, NOT about precisely
1004 # labelling which fields are possible. If the remote service correctly
1005 # determines the password is invalid, then everything's fine.
1007 my ($self, $username) = @_;
1009 if ($username =~ m{/}) {
1016 sub valid_password
{
1017 my ($self, $password) = @_;
1019 # If the prpl requires a password, _purple_connection_new() will
1020 # fail when password is NULL or zero-length without returning an
1021 # error thrasher_login() can detect. Worse, the check in
1022 # purple_account_connect() is slightly different so it wouldn't
1023 # even be detectable through purple_account_request_password() and
1026 # Registering with an empty password therefore begins an
1027 # apparently successful async login that never completes or
1028 # errors. The user also can't re-register or log out because
1029 # they're already "logging in". :(
1031 # If the prpl has OPT_PROTO_PASSWORD_OPTIONAL or OPT_PROTO_NO_PASSWORD
1032 # the corresponding subclass should override this.
1036 sub purple_forces_kill
{ return 0; }