package Net::OSCAR; $VERSION = '0.62'; =head1 NAME Net::OSCAR - Implementation of AOL's OSCAR protocol for instant messaging =head1 SYNOPSIS use Net::OSCAR qw(:standard); use Net::OSCAR::OldPerl; # You may need to use this perls older than 5.6. $oscar = Net::OSCAR->new(); $oscar->set_callback_foo(\&foo); $oscar->signon($screenname, $password); while(1) { $oscar->do_one_loop(); # Do stuff } =head1 INSTALLATION perl Makefile.PL make make test make install See C for details. =head1 DEPENDENCIES This modules requires C and C. =head1 ABSTRACT C implements the OSCAR protocol which is used by AOL's AOL Instant Messenger service. To use the module, you create a C object, register some functions as handlers for various events by using the module's callback mechanism, and then continually make calls to the module's event processing methods. You probably want to use the :standard parameter when importing this module in order to have a few important constants added to your namespace. See L<"CONSTANTS"> below for a list of the constants exported by the C<:standard> tag. No official documentation exists for the OSCAR protocol, so it had to be figured out by analyzing traffic generated by AOL's official AOL Instant Messenger client. That doesn't really help this module's stability much. This module strives to be as compatible with C as possible, but some protocol-level differences prevent total compatibility. The TOC protocol implemented by C is simpler and more well-documented but less-powerful protocol then C. See the section on L for more information. =head1 EVENT PROCESSING There are three main ways for the module to handle event processing. The first is to call the L method, which performs a C call has a default timeout of 0.01 seconds which can be adjust using the L method. A second way of doing event processing is designed to make it easy to integrate C into an existing C. Connections that don't belong to the object will be ignored, and connections that do belong to the object will be removed from the C statement for event processing instead of using C's L method. The parameters are references to the readers, writers, and errors parameters used by the select statement. The method will ignore all connections which are not C objects or which are C objects from a different C object. It modifies its arguments so that its connections are removed from the connection lists. This makes it very convenient for use with multiple C objects or use with a C. =cut sub process_connections($\$\$\$) { my($self, $readers, $writers, $errors) = @_; # Filter out our connections and remove them from the to-do list foreach my $connection(@{$self->{connections}}) { my($read, $write) = (0, 0); next unless $connection->fileno; if($connection->{connected}) { next unless vec($$readers | $$errors, $connection->fileno, 1); vec($$readers, $connection->fileno, 1) = 0; $read = 1; } if(!$connection->{connected} or $connection->{outbuff}) { next unless vec($$writers | $$errors, $connection->fileno, 1); vec($$writers, $connection->fileno, 1) = 0; $write = 1; } if(vec($$errors, $connection->fileno, 1)) { vec($$errors, $connection->fileno, 1) = 0; $connection->{sockerr} = 1; print "socket error is: " . $$errors; $connection->disconnect(); } else { $connection->process_one($read, $write); } } } =pod =item do_one_loop Processes incoming data from our connections to the various OSCAR services. This method reads one command from any connections which have data to be read. See the L method to set the timeout interval used by this method. =cut sub do_one_loop($) { my $self = shift; my $timeout = $self->{timeout}; undef $timeout if $timeout == -1; my($rin, $win, $ein) = ('', '', ''); foreach my $connection(@{$self->{connections}}) { next unless exists($connection->{socket}); if($connection->{connected}) { vec($rin, fileno $connection->{socket}, 1) = 1; } else { vec($win, fileno $connection->{socket}, 1) = 1; } } $ein = $rin | $win; my $nfound = select($rin, $win, $ein, $timeout); $self->process_connections(\$rin, \$win, \$ein) if $nfound; } sub findgroup($$) { my($self, $groupid) = @_; my($group, $currgroup, $currid); my $thegroup = undef; $self->log_printf(OSCAR_DBG_DEBUG, "findgroup 0x%04X", $groupid); while(($group, $currgroup) = each(%{$self->{buddies}})) { $self->log_printf(OSCAR_DBG_DEBUG, "\t$group == 0x%04X", $currgroup->{groupid}); next unless exists($currgroup->{groupid}) and $groupid == $currgroup->{groupid}; $thegroup = $group; last; } my $a = keys %{$self->{buddies}}; # reset the iterator return $thegroup; } =pod =item findbuddy (BUDDY) Returns the name of the group that BUDDY is in, or undef if BUDDY could not be found in any group. If BUDDY is in multiple groups, will return the first one we find. =cut sub findbuddy($$) { my($self, $buddy) = @_; #$self->log_print(OSCAR_DBG_DEBUG, "findbuddy $buddy"); foreach my $group(keys %{$self->{buddies}}) { #$self->debug_print("\t$buddy? ", join(",", keys %{$self->{buddies}->{$group}->{members}})); return $group if $self->{buddies}->{$group}->{members}->{$buddy}; } return undef; } sub findbuddy_byid($$$) { my($self, $buddies, $bid) = @_; while(my($buddy, $value) = each(%$buddies)) { return $buddy if $value->{buddyid} == $bid; } return undef; } sub newid($;$) { my($self, $group) = @_; my $id = 0; my %ids = (); if($group) { %ids = map { $_->{buddyid} => 1 } values %$group; do { ++$id; } while($ids{$id}); } else { do { $id = ++$self->{nextid}->{__GROUPID__}; } while($self->findgroup($id)); } return $id; } =pod =item commit_buddylist Sends your modified buddylist to the OSCAR server. Changes to the buddylist won't actually take effect until this method is called. Methods that change the buddylist have a warning about needing to call this method in their documentation. =item rollback_buddylist Revert changes you've made to the buddylist, assuming you haven't called L<"commit_buddylist"> since making them. =item reorder_groups (GROUPS) Changes the ordering of the groups in your buddylist. Call L<"commit_buddylist"> to save the new order on the OSCAR server. =item reorder_buddies (GROUP, BUDDIES) Changes the ordering of the buddies in a group on your buddylist. Call L<"commit_buddylist"> to save the new order on the OSCAR server. =item add_permit (BUDDIES) Add buddies to your permit list. Call L<"commit_buddylist"> for the change to take effect. =item add_deny (BUDDIES) See L. =item remove_permit (BUDDIES) See L. =item remove_deny (BUDDIES) See L. =item get_permitlist Returns a list of all members of the permit list. =item get_denylist Returns a list of all members of the deny list. =cut sub commit_buddylist($) { my($self) = shift; return must_be_on($self) unless $self->{is_on}; Net::OSCAR::_BLInternal::NO_to_BLI($self); } sub reorder_groups($@) { my $self = shift; return must_be_on($self) unless $self->{is_on}; my @groups = @_; tied(%{$self->{buddies}})->setorder(@groups); } sub reorder_buddies($$@) { my $self = shift; return must_be_on($self) unless $self->{is_on}; my $group = shift; my @buddies = @_; tied(%{$self->{buddies}->{$group}->{members}})->setorder(@buddies); } sub add_permit($@) { shift->mod_permit(MODBL_ACTION_ADD, "permit", @_); } sub add_deny($@) { shift->mod_permit(MODBL_ACTION_ADD, "deny", @_); } sub remove_permit($@) { shift->mod_permit(MODBL_ACTION_DEL, "permit", @_); } sub remove_deny($@) { shift->mod_permit(MODBL_ACTION_DEL, "deny", @_); } sub get_permitlist($) { return keys %{shift->{permit}}; } sub get_denylist(@) { return keys %{shift->{deny}}; } =pod =item rename_group (OLDNAME, NEWNAME) Renames a group. Call L<"commit_buddylist"> for the change to take effect. =item add_buddy (GROUP, BUDDIES) Adds buddies to the given group on your buddylist. Call L<"commit_buddylist"> for the change to take effect. =item remove_buddy (GROUP, BUDDIES) See L. =cut sub rename_group($$$) { my($self, $oldgroup, $newgroup) = @_; return must_be_on($self) unless $self->{is_on}; return send_error($self, $self->{bos}, 0, "That group does not exist", 0) unless exists $self->{buddies}->{$oldgroup}; $self->{buddies}->{$newgroup} = $self->{buddies}->{$oldgroup}; delete $self->{buddies}->{$oldgroup}; } sub add_buddy($$@) { my($self, $group, @buddies) = @_; $self->mod_buddylist(MODBL_ACTION_ADD, MODBL_WHAT_BUDDY, $group, @buddies); } sub remove_buddy($$@) { my($self, $group, @buddies) = @_; $self->mod_buddylist(MODBL_ACTION_DEL, MODBL_WHAT_BUDDY, $group, @buddies); } =pod =item set_visibility (MODE) Sets the visibility mode, which determines how the permit and deny lists are interpreted. The visibility mode may be: =over 4 =item * VISMODE_PERMITALL: Permit everybody. =item * VISMODE_DENYALL: Deny everybody. =item * VISMODE_PERMITSOME: Permit only those on your permit list. =item * VISMODE_DENYSOME: Deny only those on your deny list. =item * VISMODE_PERMITBUDS: Same as VISMODE_PERMITSOME, but your permit list is made to be the same as the buddies from all the various groups in your buddylist (except the deny group!) Adding and removing buddies maintains this relationship. You shouldn't manually alter the permit or deny groups when using this visibility mode. =back These constants are contained in the C package, and will be imported into your namespace if you import C with the C<:standard> parameter. When someone is permitted, they can see when you are online and send you messages. When someone is denied, they can't see when you are online or send you messages. You cannot see them or send them messages. You can talk to them if you are in the same chatroom, although neither of you can invite the other one into a chatroom. Call L<"commit_buddylist"> for the change to take effect. =cut sub set_visibility($$) { my($self, $vismode) = @_; return must_be_on($self) unless $self->{is_on}; $self->{visibility} = $vismode; } =pod =item set_group_permissions (NEWPERMS) Set group permissions. This lets you block any OSCAR users or any AOL users. C should be a list of zero or more of the following constants: =over 4 =item GROUPPERM_OSCAR Permit AOL Instant Messenger users to contact you. =item GROUPPERM_AOL Permit AOL subscribers to contact you. =back Call L<"commit_buddylist"> for the change to take effect. =cut sub set_group_permissions($@) { my($self, @perms) = @_; my $perms = 0xFFFFFF00; return must_be_on($self) unless $self->{is_on}; foreach my $perm (@perms) { $perms |= $perm; } $self->{groupperms} = $perms; } =pod =item group_permissions Returns current group permissions. The return value is a list like the one that L<"set_group_permissions"> wants. =cut sub group_permissions($) { my $self = shift; my @retval = (); foreach my $perm (GROUPPERM_OSCAR, GROUPPERM_AOL) { push @retval, $perm if $self->{groupperms} & $perm; } return @retval; } =pod =item profile Returns your current profile. =cut sub profile($) { return shift->{profile}; } =pod =item get_app_data ([GROUP[, BUDDY]]) Gets application-specific data. Returns a hashref whose keys are app-data IDs. IDs with high-order byte 0x0001 are reserved for non-application-specific usage and must be registered with the C list. If you wish to set application-specific data, you should reserve a high-order byte for your application by emailing C. This data is stored in your server-side buddylist and so will be persistent, even across machines. If C is present, a hashref for accessing data specific to that group is returned. If C is present, a hashref for accessing data specific to that buddy is returned. Call L<"commit_buddylist"> to have the new data saved on the OSCAR server. =cut sub get_app_data($;$$) { my($self, $group, $buddy) = @_; return $self->{appdata} unless $group; return $self->{buddies}->{$group}->{data} unless $buddy; return $self->{buddies}->{$group}->{members}->{$buddy}->{data}; } sub mod_permit($$$@) { my($self, $action, $group, @buddies) = @_; return must_be_on($self) unless $self->{is_on}; if($action == MODBL_ACTION_ADD) { foreach my $buddy(@buddies) { next if exists($self->{$group}->{$buddy}); $self->{$group}->{$buddy}->{buddyid} = $self->newid($self->{group}); } } else { foreach my $buddy(@buddies) { delete $self->{$group}->{$buddy}; } } } sub mod_buddylist($$$$;@) { my($self, $action, $what, $group, @buddies) = @_; return must_be_on($self) unless $self->{is_on}; @buddies = ($group) if $what == MODBL_WHAT_GROUP; if($what == MODBL_WHAT_GROUP and $action == MODBL_ACTION_ADD) { return if exists $self->{buddies}->{$group}; $self->{buddies}->{$group} = { groupid => $self->newid(), members => bltie, data => tlvtie }; } elsif($what == MODBL_WHAT_GROUP and $action == MODBL_ACTION_DEL) { return unless exists $self->{buddies}->{$group}; delete $self->{buddies}->{$group}; } elsif($what == MODBL_WHAT_BUDDY and $action == MODBL_ACTION_ADD) { $self->mod_buddylist(MODBL_ACTION_ADD, MODBL_WHAT_GROUP, $group) unless exists $self->{buddies}->{$group}; @buddies = grep {not exists $self->{buddies}->{$group}->{members}->{$_}} @buddies; return unless @buddies; foreach my $buddy(@buddies) { $self->{buddies}->{$group}->{members}->{$buddy} = { buddyid => $self->newid($self->{buddies}->{$group}->{members}), data => tlvtie, online => 0, comment => undef }; } } elsif($what == MODBL_WHAT_BUDDY and $action == MODBL_ACTION_DEL) { return unless exists $self->{buddies}->{$group}; @buddies = grep {exists $self->{buddies}->{$group}->{members}->{$_}} @buddies; return unless @buddies; foreach my $buddy(@buddies) { delete $self->{buddies}->{$group}->{members}->{$buddy}; } $self->mod_buddylist(MODBL_ACTION_DEL, MODBL_WHAT_GROUP, $group) unless scalar keys %{$self->{buddies}->{$group}->{members}}; } } sub extract_userinfo($$) { my($self, $data) = @_; my $retval = {}; my $tlvcnt; ($retval->{screenname}, $retval->{evil}, $tlvcnt) = unpack("C/a* n n", $data); $retval->{screenname} = new Net::OSCAR::Screenname $retval->{screenname}; $retval->{evil} /= 10; substr($data, 0, 5+length($retval->{screenname})) = ""; $self->log_print(OSCAR_DBG_DEBUG, "Decoding userinfo TLV with tlvcnt $tlvcnt."); my($tlv, $chainlen) = tlv_decode($data, $tlvcnt); #$chainlen--; $self->log_print(OSCAR_DBG_DEBUG, "Done decoding userinfo TLV - chainlen $chainlen."); my($flags) = unpack("n", $tlv->{1}); $retval->{trial} = $flags & 0x1; $retval->{admin} = $flags & 0x2; $retval->{aol} = $flags & 0x4; $retval->{pay} = $flags & 0x8; $retval->{free} = $flags & 0x10; $retval->{away} = $flags & 0x20; $retval->{mobile} = $flags & 0x80; ($retval->{membersince}) = unpack("N", $tlv->{2}) if exists($tlv->{2}); ($retval->{onsince}) = unpack("N", $tlv->{3}) if exists($tlv->{3}); ($retval->{idle}) = unpack("n", $tlv->{4}) if exists($tlv->{4}); ($retval->{capabilities}) = $tlv->{0xD} if exists($tlv->{0xD}); substr($data, 0, $chainlen) = ""; if($data) { $tlv = tlv_decode($data); $retval->{profile} = $tlv->{0x2} if $tlv->{0x2}; $retval->{awaymsg} = $tlv->{0x4} if $tlv->{0x4}; $retval->{chatdata} = $tlv->{0x5} if $tlv->{0x5}; } $chainlen += 5+length($retval->{screenname}); return wantarray ? ($retval, $chainlen) : $retval; } =pod =item get_info (WHO) Requests a user's information, which includes their profile and idle time. See the L callback for more information. =item get_away (WHO) Similar to L, except requests the user's away message instead of their profile. =cut sub get_info($$) { my($self, $screenname) = @_; return must_be_on($self) unless $self->{is_on}; $self->{bos}->snac_put(reqdata => $screenname, family => 0x2, subtype => 0x5, data => pack("nCa*", 1, length($screenname), $screenname)); } sub get_away($$) { my($self, $screenname) = @_; return must_be_on($self) unless $self->{is_on}; $self->{bos}->snac_put(reqdata => $screenname, family => 0x2, subtype => 0x5, data => pack("nCa*", 3, length($screenname), $screenname)); } =pod =item send_im (WHO, MESSAGE[, AWAY]) Sends someone an instant message. If the message is an automated reply generated, perhaps, because you have an away message set, give the AWAY parameter a non-zero value. Note that C will not handle sending away messages to people who contact you when you are away - you must perform this yourself if you want it done. Returns a "request ID" that you can use in the C callback to identify the message. If the message was too long to send, returns zero. =cut sub send_im($$$;$) { my($self, $to, $msg, $away) = @_; return must_be_on($self) unless $self->{is_on}; my $packet = ""; my $reqid = (8<<16) | (unpack("n", randchars(2)))[0]; if(!$self->{svcdata}->{hashlogin}) { return 0 if length($msg) >= 7987; } else { return 0 if length($msg) > 2000; } $packet = randchars(8); $packet .= pack("n", 1); # channel $packet .= pack("Ca*", length($to), $to); $packet .= tlv(2 => pack("n3 C n C n3 a*", 0x501, 4, 0x101, 1, 0x201, 1, length($msg)+4, 0, 0, $msg)); if($away) { $packet .= tlv(4 => ""); } else { $packet .= tlv(3 => ""); #request server confirmation } $self->{bos}->snac_put(reqid => $reqid, reqdata => $to, family => 0x4, subtype => 0x6, data => $packet, flags2 => 0xb); return $reqid; } =pod =item set_typing (WHO, STATUS) Tells user WHO that your typing status has changed. STATUS can be any of the following values: MTN_TYPING_DONE, MTN_TEXT_TYPED MTN_TYPING_BEGUN. =cut sub set_typing($$$) { my($self, $who, $status) = @_; return must_be_on($self) unless $self->{is_on}; my $packet = ""; $packet .= pack("n*", 0, 0, 0, 0); $packet .= pack("n", 0x01); $packet .= pack("Ca*", length($who), $who); $packet .= pack("n", $status); $self->{bos}->snac_put(family => 0x4, subtype => 0x14, data => $packet); } =pod =item buddyhash Returns a reference to a tied hash which automatically normalizes its keys upon a fetch. Use this for hashes whose keys are AIM screennames since AIM screennames with different capitalization and spacing are considered equivalent. The keys of the hash as returned by the C and C functions will be C objects, so you they will automagically be compared without regards to case and whitespace. =cut sub buddyhash($) { bltie; } sub im_parse($$) { my($self, $data) = @_; my($from, $msg, $away) = ("", "", 0); my $chat = undef; my $chaturl = undef; my $cookie = substr($data, 0, 8, ""); #OSCAR is so nice, it feeds us BLTs and cookies as SNACs. my($channel) = unpack("n", substr($data, 0, 2, "")); if($channel != 1 and $channel != 2) { carp "Got ICBM on unsupported channel $channel - ignoring."; return; } else { $self->log_print(OSCAR_DBG_DEBUG, "Incoming ICBM on channel $channel."); } $self->log_print(OSCAR_DBG_DEBUG, "Extracting user info."); my ($senderinfo, $tlvlen) = $self->extract_userinfo($data); $from = $senderinfo->{screenname}; # Copying gAIM/libfaim is *so* much easier than understanding stuff. if($channel == 1) { substr($data, 0, $tlvlen) = ""; $self->log_print(OSCAR_DBG_DEBUG, "Decoding ICBM secondary TLV."); my $tlv = tlv_decode($data); $msg = $tlv->{2}; substr($msg, 0, 2) = ""; my($y) = unpack("n", substr($msg, 0, 2, "")); substr($msg, 0, $y+2) = ""; my($msglen, $flags1, $flags2) = unpack("nnn", substr($msg, 0, 6, "")); $msglen -= 4; $away = 1 if exists $tlv->{4}; if($tlv->{3}) { # server ack requested #$self->log_print(OSCAR_DBG_DEBUG, "Sending message ack."); #$self->{bos}->snac_put(family => 0x4, subtype => 0xC, data => # $cookie . pack("nCa*", 1, length($from), $from) #); } } else { $data = $senderinfo->{chatdata}; $away = 0; substr($data, 0, 26) = ""; my $tlv = tlv_decode($data); $msg = $tlv->{0xC}; if($tlv->{0x2711}) { ($chaturl) = unpack("xx C/a*", $tlv->{0x2711}); $chaturl =~ /-.*?-(.*)/; $chat = $1; $chat =~ s/%([0-9A-Z]{1,2})/chr(hex($1))/eig; } $self->{chatinvites}->{$chaturl} = { cookie => $cookie, sender => $from }; } return ($from, $msg, $away, $chat, $chaturl); } =pod =item evil (WHO[, ANONYMOUSLY]) C, or C, a user. Evilling a user increases their evil level, which makes them look bad and decreases the rate at which they can send messages. Evil level gradually decreases over time. If the second parameter is non-zero, the evil will be done anonymously, which does not increase the user's evil level by as much as a standard evil. You can't always evil someone. You can only do it when they do something like send you an instant message. =cut sub evil($$;$) { my($self, $who, $anon) = @_; return must_be_on($self) unless $self->{is_on}; $self->{bos}->snac_put(reqdata => $who, family => 0x04, subtype => 0x08, data => pack("n C a*", ($anon ? 1 : 0), length($who), $who) ); } sub capabilities($) { my $self = shift; my $caps; #AIM_CAPS_CHAT $caps .= pack("C*", map{hex($_)} split(/[ \t\n]+/, "74 8F 24 20 62 87 11 D1 82 22 44 45 53 54 00 00")); return $caps; } =pod =item set_away (MESSAGE) Set's the users away message, also marking them as being away. If the message is undef or the empty string, the user will be marked as no longer being away. =cut sub set_away($$) { my($self, $awaymsg) = @_; return must_be_on($self) unless $self->{is_on}; shift->set_info(undef, $awaymsg); } =pod =item set_info (PROFILE) Sets the user's profile. Call L<"commit_buddylist"> to have the new profile stored on the OSCAR server. =cut sub set_info($$;$) { my($self, $profile, $awaymsg) = @_; my %tlv; tie %tlv, "Net::OSCAR::TLV"; if(defined($profile)) { $tlv{0x1} = ENCODING; $tlv{0x2} = $profile; $self->{profile} = $profile; } if(defined($awaymsg)) { $tlv{0x3} = ENCODING; $tlv{0x4} = $awaymsg; } $tlv{0x5} = $self->capabilities(); $self->log_print(OSCAR_DBG_NOTICE, "Setting user information."); $self->{bos}->snac_put(family => 0x02, subtype => 0x04, data => tlv_encode(\%tlv)); } sub svcdo($$%) { my($self, $service, %snac) = @_; my $svcname = ""; if($service == CONNTYPE_ADMIN) { $svcname = "admin"; } elsif($service == CONNTYPE_CHATNAV) { $svcname = "chatnav"; } if($self->{$svcname} and ref($self->{$svcname})) { $self->{$svcname}->snac_put(%snac); } else { push @{$self->{"${svcname}_queue"}}, \%snac; $self->svcreq($service) unless $self->{$svcname}; } } sub svcreq($$) { my($self, $svctype) = @_; $self->log_print(OSCAR_DBG_INFO, "Sending service request for servicetype $svctype."); $self->{bos}->snac_put(family => 0x1, subtype => 0x4, data => pack("n", $svctype)); } =pod =item change_password (CURRENT PASSWORD, NEW PASSWORD) Changes the user's password. =cut sub change_password($$$) { my($self, $currpass, $newpass) = @_; return must_be_on($self) unless $self->{is_on}; if($self->{adminreq}->{ADMIN_TYPE_PASSWORD_CHANGE}++) { $self->callback_admin_error(ADMIN_TYPE_PASSWORD_CHANGE, ADMIN_ERROR_REQPENDING); return; } my %tlv; tie %tlv, "Net::OSCAR::TLV"; %tlv = ( 0x02 => $newpass, 0x12 => $currpass ); $self->svcdo(CONNTYPE_ADMIN, family => 0x07, subtype => 0x04, data => tlv_encode(\%tlv)); } =pod =item confirm_account Confirms the user's account. This can be used when the user's account is in the trial state, as determined by the presence of the C key in the information given when the user's information is requested. =cut sub confirm_account($) { my($self) = shift; return must_be_on($self) unless $self->{is_on}; if($self->{adminreq}->{ADMIN_TYPE_ACCOUNT_CONFIRM}++) { $self->callback_admin_error(ADMIN_TYPE_ACCOUNT_CONFIRM, ADMIN_ERROR_REQPENDING); return; } $self->svcdo(CONNTYPE_ADMIN, family => 0x07, subtype => 0x06); } =pod =item change_email (NEW EMAIL) Requests that the email address registered to the user's account be changed. This causes the OSCAR server to send an email to both the new address and the old address. To complete the change, the user must follow instructions contained in the email sent to the new address. The email sent to the old address contains instructions which allow the user to cancel the change within three days of the change request. It is important that the user's current email address be known to the OSCAR server so that it may email the account password if the user forgets it. =cut sub change_email($$) { my($self, $newmail) = @_; return must_be_on($self) unless $self->{is_on}; if($self->{adminreq}->{ADMIN_TYPE_EMAIL_CHANGE}++) { $self->callback_admin_error(ADMIN_TYPE_EMAIL_CHANGE, ADMIN_ERROR_REQPENDING); return; } $self->svcdo(CONNTYPE_ADMIN, family => 0x07, subtype => 0x04, data => tlv(0x11 => $newmail)); } =pod =item format_screenname (NEW FORMAT) Allows the capitalization and spacing of the user's screenname to be changed. The new format must be the same as the user's current screenname, except that case may be changed and spaces may be inserted or deleted. =cut sub format_screenname($$) { my($self, $newname) = @_; return must_be_on($self) unless $self->{is_on}; if($self->{adminreq}->{ADMIN_TYPE_SCREENNAME_FORMAT}++) { $self->callback_admin_error(ADMIN_TYPE_SCREENNAME_FORMAT, ADMIN_ERROR_REQPENDING); return; } $self->svcdo(CONNTYPE_ADMIN, family => 0x07, subtype => 0x04, data => tlv(1 => $newname)); } =pod =item chat_join (NAME[, EXCHANGE]) Creates (or joins?) a chatroom. The exchange parameter should probably not be specified unless you know what you're doing. Do not use this method to accept invitations to join a chatroom - use the L<"chat_accept"> method for that. =cut sub chat_join($$; $) { my($self, $name, $exchange) = @_; return must_be_on($self) unless $self->{is_on}; $exchange ||= 4; $self->log_print(OSCAR_DBG_INFO, "Creating chatroom $name ($exchange)."); my $reqid = (8<<16) | (unpack("n", randchars(2)))[0]; $self->{chats}->{pack("N", $reqid)} = $name; $self->svcdo(CONNTYPE_CHATNAV, family => 0x0D, subtype => 0x08, reqid => $reqid, data => #pack("n Ca*n3C na*", $exchange, # length("create"), "create", 0xFFFF, 0x100, 0x100, 0xD3, # length($name), $name #) pack("n Ca*n2C a*", $exchange, length("create"), "create", 0xFFFF, 0x100, 0x03, tlv( 0xD7 => "en", 0xD6 => "us-ascii", 0xD3 => $name ) ) ); } =pod =item chat_accept (CHAT) Use this to accept an invitation to join a chatroom. =item chat_decline (CHAT) Use this to decline an invitation to join a chatroom. =cut sub chat_accept($$) { my($self, $chat) = @_; return must_be_on($self) unless $self->{is_on}; delete $self->{chatinvites}->{$chat}; $self->log_print(OSCAR_DBG_NOTICE, "Accepting chat invite for $chat."); $self->svcdo(CONNTYPE_CHATNAV, family => 0x0D, subtype => 0x04, data => pack("nca* Cn", 4, length($chat), $chat, 0, 2) ); } sub chat_decline($$) { my($self, $chat) = @_; return must_be_on($self) unless $self->{is_on}; my($invite) = delete $self->{chatinvites}->{$chat} or do { $self->log_print(OSCAR_DBG_WARN, "Chat invite for $chat doesn't exist, so we can't decline it."); return; }; $self->log_print(OSCAR_DBG_NOTICE, "Declining chat invite for $chat."); $self->{bos}->snac_put(family => 0x04, subtype => 0x0B, data => $invite->{cookie} . pack("n", 2) . pack("Ca*", length($invite->{sender}), $invite->{sender}) . pack("nnn", 3, 2, 1) ); } sub crapout($$$;$) { my($self, $connection, $reason, $errno) = @_; send_error($self, $connection, $errno || 0, $reason, 1); $self->signoff(); } sub must_be_on($) { my $self = shift; send_error($self, $self->{bos}, 0, "You have not finished signing on.", 0); } =pod =item set_idle (TIME) Sets the user's idle time in seconds. Set to zero to mark the user as not being idle. Set to non-zero once the user becomes idle. The OSCAR server will automatically increment the user's idle time once you mark the user as being idle. =cut sub set_idle($$) { my($self, $time) = @_; return must_be_on($self) unless $self->{is_on}; $self->{bos}->snac_put(family => 0x1, subtype => 0x11, data => pack("N", $time)); } =pod =item clone Clones the object. This creates a new C object whose callbacks, loglevel, screenname debugging, and timeout are the same as those of the current object. This is provided as a convenience when using multiple C objects in order to allow you to set those parameters once and then call the L method on the object returned by clone. =cut sub clone($) { my $self = shift; my $clone = $self->new(); # Born in a science lab late one night # Without a mother or a father # Just a test tube and womb with a view... # Okay, now we don't want to just copy the reference. # If we did that, changing ourself would change the clone. $clone->{callbacks} = { %{$self->{callbacks}} }; $clone->{LOGLEVEL} = $self->{LOGLEVEL}; $clone->{SNDEBUG} = $self->{SNDEBUG}; $clone->{timeout} = $self->{timeout}; return $clone; } =pod =item selector_filenos Returns a list whose first element is a vec of all filehandles that we care about reading from and whose second element is a vec of all filehandles that we care about writing to. See the L<"process_connections"> method for details. =cut sub selector_filenos($) { my($rin, $win) = ('', ''); foreach my $connection(@{shift->{connections}}) { next unless $connection->{socket}; if($connection->{connected}) { vec($rin, fileno $connection->{socket}, 1) = 1; } if(!$connection->{connected} or $connection->{outbuff}) { vec($win, fileno $connection->{socket}, 1) = 1; } } return ($rin, $win); } =pod =item visibility Returns the user's current visibility setting. See L. =item groups Returns a list of groups in the user's buddylist. =item buddies (GROUP) Returns the names of the buddies in the specified group in the user's buddylist. The names may not be formatted - that is, they may have spaces and capitalization removed. The names are C objects, so you don't have to worry that they're case and whitespace insensitive when using them for comparison. =item buddy (BUDDY[, GROUP]) Returns information about a buddy on the user's buddylist. This information is a hashref which may have the following keys: =over 4 =item online The user is signed on. If this key is not present, all of the other keys may not be present. =item screenname The formatted version of the user's screenname. This includes all spacing and capitalization. This is a C object, so you don't have to worry about the fact that it's case and whitespace insensitive when comparing it. =item comment A user-defined comment associated with the buddy. See L<"set_buddy_comment">. Note that this key will be present but undefined if there is no comment. =item trial The user's account has trial status. =item aol The user is accessing the AOL Instant Messenger service from America OnLine. =item free Opposite of aol. =item away The user is away. =item admin The user is an administrator. =item membersince Time that the user's account was created, in the same format as the C