package Net::PSYC::Client; use strict; our $VERSION = '1.0'; # # implements some basic client functionality... # # your perl-script (main::) needs to have following subs # - getPassword() # use Net::PSYC qw(same_host parse_uniform make_uniform psyctext W register_route get_connection); use Net::PSYC::Event qw(register_uniform unregister_uniform); use Net::PSYC::Tie::AbbrevHash; use base 'Exporter'; our @EXPORT = qw(register_context unregister_context register_new msg psycLink psycUnlink sendmsg UNI NICK enter talk get_context); my ($new, %ContextReg, $UNI, $NICK, %react, $SERVER_UNI, $SERVER_HOST, %tags, $action); our %friends; sub UNI () { $UNI } # we like the idea how this "sub" gets inlined, that's why it looks so weird sub NICK () { $NICK || $UNI? ($NICK = substr((parse_uniform($UNI))[4], 1)) : '' } # register_context ( uni, obj ) sub register_context { unless (ref $_[1]) { delete $ContextReg{$_[0]}; W1('register_context: %s', $_[0]); return; } W1('register_context(%s, %s)', $_[0], $_[1]); $ContextReg{$_[0]} = $_[1]; return; } # unregister_context ( uni ) sub unregister_context { W1('unregister_context(%s)', $_[0]); if (ref $_[0]) { foreach (keys %ContextReg) { if ($ContextReg{$_} eq $_[0]) { delete $ContextReg{$_}; return; } } } delete $ContextReg{$_[0]}; } sub register_new { $new = shift; } sub get_context { my $vars = shift; my ($addr ,$obj, $u); unless (ref $vars eq 'HASH') { $addr = $vars; $vars = {}; } else { $addr = $vars->{'_context'} || $vars->{'_source_relay'} || $vars->{'_source'}; } $obj = $ContextReg{$addr}; return $obj if ($obj); if ($addr && ($u = parse_uniform($addr))) { # if ($u->{'object'} !~ /^\@/ && $new) { if ($new) { my $name; if (same_host($u->{'host'}, $SERVER_HOST)) { if ($u->{'scheme'} eq 'psyc') { $name = substr($u->{'object'}, 1); } elsif ($u->{'user'}) { $name = $u->{'user'}; } else { $name = $addr; } } else { $name = $addr; # this is maybe evil. should be done in the core lib since # the psyc-routing rules should make that clear #register_route($addr, $vars->{'_INTERNAL_origin'}) # if (exists $vars->{'_INTERNAL_origin'}); } $obj = &{$new}( $addr, $name, 0 ); register_context($addr, $obj); } } else { W0('%s does not seem to be a valid UNL. Dropping message.', $addr); return 0; } return $obj; } # internal sub verify_enter { my $vars = shift; my ($addr ,$obj, $u); $addr = $vars->{'_context'} || $vars->{'_source'}; return 1 if (exists $ContextReg{$addr}); unless (defined delete $tags{$vars->{'_tag'}} || Net::PSYC::get_connection(UNI()) eq $vars->{'_INTERNAL_origin'}) { W0('%s is trying to join us into a room without a proper _tag', $addr); return 0; } $u = parse_uniform($addr); if (ref $u && $u->{'object'} =~ /^\@/ && $new) { my $silent = (exists $vars->{'_control'} && $vars->{'_control'} eq '_silent') ? 1 : 0; my $name; if (same_host($u->{'host'}, $SERVER_HOST)) { $name = substr($u->{'object'}, 1); } else { $name = $addr; } $obj = &{$new}( $addr, $name, $silent ); register_context($addr, $obj); return 1; } return 0; } sub getUni { my $obj = shift; return unless(ref $obj); foreach(keys %ContextReg) { if($ContextReg{$_} == $obj) { return $_; } } return 0; } # incoming messages # msg ( source, mc, data, vars) sub msg { my ($source, $mc, $data, $vars) = @_; W1('Client->msg(%s, %s, %.10s ..., %p)', $source, $mc, $data, $vars); my $func = $react{$mc}; if ($func) { &{$func}($source, $mc, $data, $vars); } else { my $c = get_context($vars); # we expect msg to do the right thing unless (ref $c) { # this is a message from a room we never entered # we may think about sending _request_place_leave W0('Client->msg(%s, %s, %s, %s) is baklawa.', $source, $mc, $data, $vars); return 1; } my $time = int($vars->{'_time_place'}||$vars->{'_time_log'}); if ($time + 5 < time()) { my @t = localtime(int($time)); $vars->{'_time'} = sprintf '%02d:%02d:%02d', $t[2], $t[1], $t[0]; } $c->msg($source, $mc, $data, $vars) if $c; } return 1; } sub zwirbel { my $word = shift; return $word if (length($word) <= 3); my @t = split(//, $word); foreach (1..$#t-1) { my $n = int(rand($#t - 1)) + 1; if ($n != $_) { my $a = $t[$_]; $t[$_] = $t[$n]; $t[$n] = $a; } } return join('', @t); } sub sendmsg { my ($target, $mc, $data, $vars, $MMPvars) = @_; if ($target =~ /^xmpp/) { register_route($target, get_connection(UNI())); } W1('Client->sendmsg(%s, %s, %.10s ..., %s)', $target, $mc, $data, $vars); $MMPvars ||= {}; $MMPvars->{'_source_identification'} = UNI(); if ($mc =~ /^_message/) { # this is cooler but muve may not be entirely ready for it yet # $vars->{'_nick'} ||= NICK() unless (lc(NICK()) eq NICK()); $vars->{'_nick'} ||= NICK(); # hmm.. a bit excentric to force zwirbel on all users of perlpsyc $vars->{'_action'} ||= zwirbel($action) if ($action && $mc eq '_message_public'); } !Net::PSYC::sendmsg($target, $mc, $data, $vars, $MMPvars); } # link to a given uni sub psycLink { $UNI = shift; my $u = parse_uniform($UNI); unless ($u) { return W1('I cannot link to that uni: %s (It is not valid)', $UNI); } # register_uniform($UNI); register_uniform(); # this is no good $SERVER_HOST = $u->{'host'}; $SERVER_UNI = make_uniform($u->{'user'}, $u->{'host'}, $u->{'port'}, $u->{'transport'}, ''); Net::PSYC::sendmsg($UNI, '_request_link', '', { _password => main::getPassword(), _tag => "perlpsyc$$", }); # we need to do that raw.. since we want no _identification == UNI } sub psycUnlink () { unregister_uniform(UNI()); sendmsg(UNI(), '_request_exit'); } sub enter { my $target = shift; my $vars = {}; unless (parse_uniform($target)) { $target =~ s/^[@|#]//o; $target = $SERVER_UNI."/\@$target"; } $vars->{'_tag'} = unpack( "H*",pack("F", rand(10))); $tags{$vars->{'_tag'}} = $target; $vars->{'_nick'} = NICK(); sendmsg($target, '_request_enter', '', $vars); } sub talk { my $target = shift; unless ($target =~ /psyc:\/\//) { $target = $SERVER_UNI."/\~$target"; } my $vars = { '_nick' => NICK(), }; $vars->{'_action'} = $action if $action; sendmsg($target, '_message_private', join(' ', @_), {_nick=>NICK()}); } tie %react, 'Net::PSYC::Tie::AbbrevHash'; %react = ( '_echo_place_leave' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = get_context($vars->{'_context'}); return unless ($obj); unless ($obj->{'type'} eq 'place') { return W0('Got an _echo_place_leave from someone (%s) who does not appear to be a room. Dropping message.', $source); } $obj->msg(@_); $obj->leanforwardandchokeyourself(); }, '_echo_set' => sub { my ($source, $mc, $data, $vars) = @_; return unless($source eq UNI()); if ($vars->{'_key'} eq 'name' && exists $vars->{'_value'}) { $NICK = $vars->{'_value'}; } elsif ($vars->{'_key'} eq 'speakaction' && exists $vars->{'_value'}) { $action = $vars->{'_value'}; } my $c = get_context($vars); return unless ($c); # we expect msg to do the right thing $c->msg($source, $mc, $data, $vars) if $c; }, '_message_echo_private' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = get_context($vars->{'_source_relay'} || $source); return unless ($obj); $obj->msg(@_); }, '_notice_place_leave' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = get_context($vars); return unless ($obj); $obj->msg(@_); $obj->leave( $vars->{'_nick'} || $source ) if $obj->can('leave'); }, '_echo_place_enter' => sub { my ($source, $mc, $data, $vars) = @_; unless (verify_enter($vars)) { return; } my $obj = get_context($vars); return unless ($obj); $obj->enter( $vars->{'_nick'} || $source, UNI()) if $obj->can('enter'); $obj->msg(@_); }, '_notice_place_enter' => sub { my ($source, $mc, $data, $vars) = @_; unless (verify_enter($vars)) { return; } my $obj = get_context($vars); return unless ($obj); $obj->enter( $vars->{'_nick'} || $source, $source ) if $obj->can('enter'); $obj->msg(@_); }, '_query_password' => sub { my ($source, $mc, $data, $vars) = @_; sendmsg(UNI(), '_set_password', '', {_password=> main::getPassword() }); }, '_status_place_members' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = get_context($source); return unless ($obj); if ($obj->can('members') && exists $vars->{'_list_members_nicks'}) { my $members = {}; # name -> uni for (0 .. @{$vars->{'_list_members_nicks'}} - 1) { $members->{$vars->{'_list_members_nicks'}->[$_]} = $vars->{'_list_members'}->[$_]; } $obj->members($members); } $obj->msg(@_); }, '_notice_link' => sub { my ($source, $mc, $data, $vars) = @_; # suboptimal, we parse even if the tag is given.. my @user = parse_uniform($source); my @uni = parse_uniform(UNI()); # allow psyc server to have a better name than we thought, or to # redirect us to a better identity. if ($$vars{'_tag_reply'} eq "perlpsyc$$" or same_host($user[1], $uni[1]) && lc($user[4]) eq lc($uni[4])) { $UNI = $source; if (exists $vars->{'_nick'}) { $NICK = $vars->{'_nick'}; } Net::PSYC::get_connection($UNI)->TRUST(6); # this hack was specific to psyc developers who also use irc clients ;) # but no longer necessary since psyced will always set echo on for # psyc clients. #sendmsg(UNI(), '_request_execute', 'set echo on'); } else { W0('Got a _notice_link from a UNI (%s) we did not try to link to.', $source); } }, '_notice_unlink' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = get_context($source); $obj->msg(@_) if $obj; # forcing my way out of psycion... $obj should probably handle this.. FIXME print "\r", '-' x 44, "\n\r\n"; exit(); }, '_status_place' => sub { my ($source, $mc, $data, $vars) = @_; my $c = get_context($source); $c->msg(@_) if $c; }, '_info_description' => sub { my ($source, $mc, $data, $vars) = @_; if ($source eq UNI()) { $action = $vars->{'_action'}; } my $obj = get_context($source); $obj->msg(@_) if $obj; }, '_list_friends_present' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = get_context($source); return unless ($obj); my $friends = {}; # name -> uni for (0 .. @{$vars->{'_list_friends_nicknames'}} - 1) { $friends->{$vars->{'_list_friends_nicknames'}->[$_]} = $vars->{'_list_friends'}->[$_]; } if ($obj->can('friends')) { $obj->friends($friends); } else { %friends = %$friends; } $obj->msg(@_); }, '_notice' => sub { my ($source, $mc, $data, $vars) = @_; my $obj = $ContextReg{$vars->{'_context'}||$vars->{'_source'}}; $obj = get_context(UNI()) unless ($obj); $obj->msg(@_) if $obj; }, ); $react{'_error'} = $react{'_notice'}; $react{'_status'} = $react{'_notice'}; $react{'_info_nickname'} = $react{'_info_description'}; 1; __END__ =pod =head1 NAME Net::PSYC::Client =head1 DESCRIPTION Net::PSYC::Client offers an easy-to-implement interface to build chat clients using the PSYC protocol. =head1 SYNOPSIS use Net::PSYC::Client; =head1 PERL API =over 4 =item psycLink( B<$identity> ) Tries to link to (login to) the given B<$identity>. A password, if necessary, has to be returned by main::getPassword(). =item psycUnlink() Performs a log-out of the PSYC server. =item register_new( B<$sub> ) B<$sub> is called each time a new object has to be created. Normally this occurs when the user joins a room or receives a private message. &{B<$sub>}( uni, name[, silent]) has to return an object of a class implementing the interface specified below. The silent flag is 1 if the room does not allow talking. You may refrain from allowing input in that case and, when in a windowing environment, refrain from opening an extra window and rather direct messages to the console. =item enter( B<$room> ) Sends a request for entrance to the given room. B<$room> may be the full UNI or just the name of the room in which case it is assumed to be on the server the user is linked to. =item talk( B<$person>, B<@text> ) Sends a private message to the given person. B<$person> may be a UNI or just a local nickname. =item sendmsg( B<$target>, B<$mc>, [B<$data>[, B<$vars>[, B<$MMPvars>]]] ) For a description of sendmsg() have a look at L. Use this sendmsg instead of the one from L in case you are sending messages _as_ the user you are linked to. Basically it just assigns the _identification variable for you. =item register_context( B<$uni>, B<$obj> ) Register an object to receive all conference messages from the given UNI. Depending on whether UNI represents a person or a chatroom the object has to implement the appropriate interface: =back =head1 CLIENT INTERFACE In order to make it easy to implement chat-clients based on the PSYC protocol there are only a few requirements to be met by the actual user interface. =head2 Default functions (both Person and Chatoom conference) =over 4 =item B<$obj>-Emsg( B<$source>, B<$mc>, B<$data>, B<$vars> ) Every message that's worth printing is delivered to the msg() method. You could implement different templates for every method (B<$mc>) though this basic implementation should be enough for simple applications. sub msg { my ($source, $mc, $data, $vars) = @_; print "*** ", Net::PSYC::psyctext($data, $vars), "\n"; return 1; } =item B<$obj>-Eleanforwardandchokeyourself() When the user himself has left a room B<$obj>-Eleanforwardandchokeyourself() is called to tell the object that it is no longer needed. Keep the user from sending any more messages to that room and close the window. The name of this callback method may change one day. Suggest something. =item B<$obj>-E{'type'} would be 'person' or 'place'. =back =head2 Chatroom In addition to I every object representing a chatroom has to have the following methods =over 4 =item B<$room>-Emembers( B<\%members> ) B<\%members> is a reference to a hash containing nicknames and psyc addresses of all users currently in the room represented by B<$room>. The structure of the hash is { nickname =E psyc address }. =item B<$room>-Eenter( B<$nick>, B<$uni>) enter() is called whenever someone enters B<$room>. =item B<$leave>-Eleave( B<$nick> ) leave() is called whenever someone leaves B<$room>. =item B<$room>-Etopic( B<$topic>, B<$nick> ) Whenever the topic in this chatroom is changed this method is called with the new B<$topic> and the B<$nick> who changed it, if available. =back =head1 SEE ALSO L, L, L, L for a bunch of scripts and a client (psycion) using L =head1 AUTHORS Arne GEdeke =head1 COPYRIGHT Copyright (c) 2004-2016 Arne GEdeke. All rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. =cut