commit ceccb689a9fad2654819e44ca6b75fde6c20c655 Author: milkbone57 Date: Fri Jul 11 20:17:20 2003 +0000 Initial revision diff --git a/Milkbone.pm b/Milkbone.pm new file mode 100644 index 0000000..8eda6b9 --- /dev/null +++ b/Milkbone.pm @@ -0,0 +1,422 @@ +# ----------------------------------------------------------------------------- +# Author(s) : Bill Atkins and Eric Thul +# Title : MOS core tools +# Date : 12.21.02 +# Desc : the hamster that keep MOS going +# QoTP : got milkbone tools? +# Notes : for more information see the plugin documentation +# License : under the same terms as mos.pl +# ----------------------------------------------------------------------------- + +package Milkbone; + +use strict; +use warnings; + +use Carp qw(longmess); +use PAR; +use Milkbone::HookEntry; +use Benchmark; + +our $VERSION = "0.355"; + +require Exporter; + +our @ISA = qw( Exporter ); +our @EXPORT = qw( hook is_running abort queued_hook register_hook option path data %ARGS slurp + deregister_hook strip_html user_file set_option MOSLoop set_interval unload_plugin load_plugin init_plugin reload_core); +our @EXPORT_OK = qw( ); + +my ($running, $dirty); +my $interval = 0.006; +our (%hooks, %options, %data, %plugins); +our (%ARGS) = (); + +$running = 1; + +$| = 1; + +sub is_running +{ + $running; +} + +sub set_interval +{ + $interval = shift; +} + +sub abort +{ + $running = 0; + die @_; +} + +sub init +{ + # initialization + register_hook("load_options", \&load_options); + register_hook("load_plugins", \&load_plugins); + register_hook("post_mainloop", \&post_mainloop); + register_hook("pre_mainloop", \&pre_mainloop); + register_hook("save_options", \&save_options); + register_hook("loaded_plugins", \&plugin_list); + register_hook("error", \&error); +} + +# ------------------------------------------ +# Plugin and Hook System +# ------------------------------------------ + +sub load_options +{ + my ($directive, $temp, @args); + + open(GLOBAL, ") + { + next if /^\#/; + next if /^\s*\n/; + + chomp; + + ($directive, @args) = split /\s|\,/; + + if(@args > 1) + { + my @val = grep { !/^$/ } @args; + $options{$directive} = \@val; + } + else + { + my $val = $args[0]; + $options{$directive} = \$val; + } + } + + close(GLOBAL); +} + +sub save_options +{ + return unless $dirty; + my ($val, @temp, $str); + open(GLOBAL, ">mb.conf") or die "Can't open config file: $!"; + print GLOBAL "# mb.conf - milkbone global configuration file\n\n"; + + for(keys(%options)) + { + $val = $options{$_}; + + if(ref($val) eq "SCALAR") + { + print GLOBAL "$_ $$val\n"; + } + else + { + @temp = @$val; + s/\s*// for @temp; + @temp = grep { $_ ne "" } @temp; + $str = join(', ', @$val); + $str =~ s/, (, )*/, /g; + print GLOBAL "$_ " . $str . "\n"; + } + } + + close(GLOBAL); +} + +sub option +{ + my $val = $options{$_[0]}; + + return undef unless $val; + + if(ref($val) eq "SCALAR") + { + $$val; + } + else + { + @$val; + } +} + +sub set_option +{ + my ($name, $val) = @_; + + $options{$name} = \$val; + $dirty = 1; +} + +sub load_plugin +{ + my ($plugin) = @_; + + return unless $plugin; + + if(-e path("plugins/$plugin.pl")) + { + } + # PAR support - disabled in non-Mercury releases + #elsif(-e path("plugins/$plugin.zip")) + #{ + # PAR->import(path("plugins/$plugin.zip")); + #} + elsif(-e path("plugins/$plugin") and -d path("plugins/$plugin")) + { + unshift @INC, path("plugins/$plugin"); + } + else + { + die "Couldn't load plugin $plugin"; + return; + } + + $plugins{$plugin} = 1; +} + +sub unload_plugin +{ + my ($plugin) = @_; + my $package = $plugin; + $package =~ s/-//g; + + for my $element (keys %hooks) + { + my @new; + for(@{$hooks{$element}}) + { + unless($_->{package} eq $package) + { + push @new, $_; + } + } + $hooks{$element} = \@new; + } + + delete $plugins{$plugin}; + + for(keys %INC) + { + delete $INC{$_} if /^plugins\/$plugin/; + } + + print %plugins; +} + +sub plugin_list +{ + [keys(%plugins)]; +} + +sub init_plugin +{ + my ($plugin) = @_; + eval "require \"$plugin.pl\";" or hook("error", -short => "Couldn't require $plugin.pl for $plugin: $! $@"); +} + +sub load_plugins +{ + + my @plugins = grep({ $_ } (option("Modules"), option("Plugins"))); + push @INC, "./plugins"; + load_plugin($_) for @plugins; + init_plugin($_) for(@plugins); +} + +sub hook +{ + my ($hook, %args) = @_; + my %old; + my ($res); + + warn "Unregistered hook called: $hook" . ' ' . join(' ', caller) . "\n" if !exists($hooks{$hook}); + + %old = %ARGS; + + for $hook (@{$hooks{$hook}}) + { + %ARGS = %args; + + $ARGS{$_} = $hook->{args}->{$_} for %{$hook->{args}}; + eval { $res = $hook->call; }; + print longmess($! . $@) if $@; + } + + %ARGS = %old; + + return $res; +} + +sub reload_core +{ + delete $INC{'Milkbone.pm'}; + eval "require 'Milkbone.pm';"; +} + +sub register_hook +{ + my ($hook, $coderef, $args) = @_; + my ($package, $file, $line) = caller; + my $entry = new Milkbone::HookEntry($hook, $coderef, $args, $package); + + push @{$hooks{$hook}}, $entry; +} + +sub deregister_hook +{ + my ($hook) = @_; + my ($file, $package, $line) = caller; + my @temp = @{$hooks{$hook}}; + + die "Required hook missing" if !exists($hooks{$hook}) and $hook eq "tick"; + warn "Unregistered hook deleted: $hook" if !exists($hooks{$hook}); + + @temp = grep { ($_->{package} ne $package) && ($_->{hook_name} ne $hook) } @temp; + + @{$hooks{$hook}} = @temp; +} + +sub MOSLoop +{ + my $i = 0; + while(is_running) + { + hook("tick"); + select(undef, undef, undef, $interval) if $i++ % 5 == 0 && $interval; + } +} + +# ------------------------------------------ +# Default Hooks +# ------------------------------------------ + +sub post_mainloop +{ + hook("save_options"); +} + +sub pre_mainloop +{ +} + +sub error +{ + unlink "errlog.txt" and warn "errlog.txt has exceeded 500K." if -s "errlog.txt" > 500 * 1024; + open(LOG, ">>errlog.txt") or die "Can't open error file: $!"; + print LOG $ARGS{-short} . "\n" if $ARGS{-short}; + print LOG $ARGS{-long} . "\n" if $ARGS{-long}; + close(LOG); +} + +# ------------------------------------------ +# Utilities +# ------------------------------------------ + +sub path +{ + my ($in) = @_; + + if($^O =~ /Win32/) + { + $in =~ s~/~\\~g; + } + else + { + $in =~ s~\\~/~g; + } + return $in; +} + +sub data : lvalue +{ + $data{$_[0]}; +} + +sub slurp +{ + my ($file, $no_chomp) = @_; + open(FILE, $file) or return "FAILED"; + my @all = ; + close(FILE) or return "FAILED"; + + chomp @all unless $no_chomp; + + if(wantarray) + { + return @all; + } + else + { + return join('', @all); + } +} + +sub strip_html +{ + $_ = shift; + s/
/\n/gi; + s/<.*?>//gi; + s/&/&/gi; + s/>/>/gi; + s/</ join(' ', @_), -long => longmess(join(' ', @_)), -fatal => 1); + print longmess(@_); + abort; +}; + +*CORE::GLOBAL::warn = sub { + hook("error", -short => join(' ', @_), -long => longmess(join(' ', @_))); + print longmess(@_); +}; + +1; diff --git a/build-temp.bat b/build-temp.bat new file mode 100644 index 0000000..966f293 --- /dev/null +++ b/build-temp.bat @@ -0,0 +1,4 @@ +cd plugins +perl compress_dist.pl +cd .. +perlapp --nocompress --lib lib --verbose --icon images\mbone.ico --force --xclude mos.pl --freestanding --exe milkbone.exe --trim POSIX --add Tk;Tk::Photo;Tk::Text;Tk::TextUndo;Tk::ROText;Tk::HList;Tk::Tree;Tk::ItemStyle;Tk::LabEntry;Socket;Digest::MD5;Win32::API;Win32::GuiTest;Tk::Font;UNIVERSAL;Win32::Sound;Winamp::Control;Tk::BrowseEntry \ No newline at end of file diff --git a/build.bat b/build.bat new file mode 100644 index 0000000..44bd32e --- /dev/null +++ b/build.bat @@ -0,0 +1,5 @@ +cd plugins +perl compress_dist.pl +cd .. +perlapp --nocompress --gui --lib lib --verbose --icon images\mbone.ico --force --xclude mos.pl --freestanding --exe milkbone.exe --trim POSIX --add Tk;Tk::Photo;Tk::Text;Tk::TextUndo;Tk::ROText;Tk::HList;Tk::Tree;Tk::ItemStyle;Tk::LabEntry;Socket;Digest::MD5;Win32::API;Win32::GuiTest;Tk::Font;Winamp::Control;Tk::BrowseEntry;Win32::Sound +upx milkbone.exe \ No newline at end of file diff --git a/docs/BUGS.txt b/docs/BUGS.txt new file mode 100644 index 0000000..9f13666 --- /dev/null +++ b/docs/BUGS.txt @@ -0,0 +1,9 @@ +periods in screen names +fatal error garbage +deregister might need some work +if you have the buddy group minused so it doesnt show the members and when someone logs on after it is minused, they show up listed under the list still even though it is minimized +might not log on easily after getting booted +weird double convos +typing goes away when you send +mobile icons = buggy +graceful failing on linux (and vice-versa) \ No newline at end of file diff --git a/docs/CREDIT.txt b/docs/CREDIT.txt new file mode 100644 index 0000000..d5ba89f --- /dev/null +++ b/docs/CREDIT.txt @@ -0,0 +1,14 @@ +dan chokola Original inspiration to rewrite aim +marci caraballo "Goodbye and Sign Off" feature +aj kappe Broadcast Messages +eric thul File-Sharing Searches +greg blair Special characters +dave mcpherson Profile Change Alert +matthew sachs Wrote the Net::OSCAR module +paul christian Automatic sarcasm +aj kappe history of sent messages +aj kappe wishbone instead of wishlist +dan chokola remote command line +marc dashevsky the spiffy BrowseEntry control (not yet used) +eric thul and bill atkins plugin/MOS separation +rob mccool for documenting httpd.conf and not noticing that we stole his words :) \ No newline at end of file diff --git a/docs/FILEMAP.txt b/docs/FILEMAP.txt new file mode 100644 index 0000000..d62e00a --- /dev/null +++ b/docs/FILEMAP.txt @@ -0,0 +1,21 @@ +mb.pl +---- + +The core of milkbone. mb.pl coordinates all other modules and contains the code for the MBMain function, which +governs milkbone's basic flow of execution. All Net::OSCAR events are handled in mb.pl and dispatched thence to the +appropriate module. + +Milkbone.pm +---- + +Milkbone.pm contains common Milkbone routines, including: + + 1. slurp and pour - for whole-file input and output + 2. hook - for calling user-installed hooks (as part of the plugin system) + 3. begin_log and end_log - for logging errors to errlog.txt + 4. path - for using Windows paths on UNIX and vice-versa + +Milkbone\OS.pm +---- + +MIlkbone::OS encapsulates OS-specific calls to allow easier porting. \ No newline at end of file diff --git a/docs/HISTORY.txt b/docs/HISTORY.txt new file mode 100644 index 0000000..33c7582 --- /dev/null +++ b/docs/HISTORY.txt @@ -0,0 +1,25 @@ +the history of milkbone +----------------------- + +When people ask how milkbone came to be, I like to tell a little story. It goes like this: + +------------------ +now it came to pass that in those days Satan came unto the peoples of the +internet and said, 'behold - i give you AOL.' and the soccer moms and the +other unfaithful ones took up the gift and said, "yea, this is good and we +shall use it and spread it over the earth." and they didst take AOL in the +form of innocent-looking cd's sent through the mail and they didst use it and +pay exorbitant rates for it and they didst tell their friends until all the +world was caught up in the unholy trap. + +and then it happened that the origins of AOL were discovered and some of the +people did say: 'alas! for we have been deceived' and they cast off the poision +of AOL and created then a Holy Hand Messenger and, counting from one to three, +didst say unto the world, 'hark! for freedom lies in aimLess' and they didst +pull the pin and didst count from one to three, not including four and not +counting zero. and aimLess then was released and all was well. +------------------ + +Except for the part about Satan creating AOL, this story is purely fictional. It was originally our slogan until we realized it was WAY too long. (aimLess and Holy Hand Messenger were former names for milkbone) + +The real history is a little different, but I have no interest in typing it now. \ No newline at end of file diff --git a/docs/HOOKS.txt b/docs/HOOKS.txt new file mode 100644 index 0000000..1f2629a --- /dev/null +++ b/docs/HOOKS.txt @@ -0,0 +1,227 @@ +Milkbone.pm:25:our @EXPORT = qw( hook is_running abort queued_hook register_hook option path data %ARGS slurp +Milkbone.pm:26: deregister_hook strip_html user_file set_option MOSLoop set_interval); +Milkbone.pm:31:our (%hooks, %options, %data); +Milkbone.pm:56: register_hook("load_options", \&load_options); +Milkbone.pm:57: register_hook("load_plugins", \&load_plugins); +Milkbone.pm:58: register_hook("post_mainloop", \&post_mainloop); +Milkbone.pm:59: register_hook("pre_mainloop", \&pre_mainloop); +Milkbone.pm:60: register_hook("save_options", \&save_options); +Milkbone.pm:61: register_hook("error", \&error); +Milkbone.pm:187:sub hook +Milkbone.pm:189: my ($hook, %args) = @_; +Milkbone.pm:193: warn "Unregistered hook called: $hook" . ' ' . join(' ', caller) . "\n" if !exists($hooks{$hook}); +Milkbone.pm:197: for $hook (@{$hooks{$hook}}) +Milkbone.pm:201: $ARGS{$_} = $hook->{args}->{$_} for %{$hook->{args}}; +Milkbone.pm:202: $res = $hook->call; +Milkbone.pm:210:sub register_hook +Milkbone.pm:212: my ($hook, $coderef, $args) = @_; +Milkbone.pm:214: my $entry = new Milkbone::HookEntry($hook, $coderef, $args, $package); +Milkbone.pm:216: push @{$hooks{$hook}}, $entry; +Milkbone.pm:219:sub deregister_hook +Milkbone.pm:221: my ($hook) = @_; +Milkbone.pm:223: my @temp = @{$hooks{$hook}}; +Milkbone.pm:225: die "Required hook missing" if !exists($hooks{$hook}) and $hook eq "tick"; +Milkbone.pm:226: warn "Unregistered hook deleted: $hook" if !exists($hooks{$hook}); +Milkbone.pm:228: @temp = grep { ($_->{package} ne $package) && ($_->{hook_name} ne $hook) } @temp; +Milkbone.pm:230: @{$hooks{$hook}} = @temp; +Milkbone.pm:238: hook("tick"); +Milkbone.pm:249: hook("save_options"); +Milkbone.pm:336: hook("error", -short => join(' ', @_), -long => longmess(join(' ', @_)), -fatal => 1); +Milkbone.pm:342: hook("error", -short => join(' ', @_), -long => longmess(join(' ', @_))); +mos.pl:35: hook("load_options"); +mos.pl:36: hook("load_plugins"); +mos.pl:42: hook("load_options"); +mos.pl:43: hook("load_plugins"); +mos.pl:44: hook("pre_mainloop"); +mos.pl:46: hook("create_logon_prompt"); +mos.pl:51: hook("post_mainloop"); +plugins/Milkbot.pl:21:register_hook("signed_in", \&init); +plugins/Milkbot.pl:24: hook("protocol_get_realname", -user => $me = $ARGS{-me}); +plugins/Milkbot.pl:26: register_hook("milkbot_set_command", sub { +plugins/Milkbot.pl:29: hook("milkbot_get_commands"); +plugins/Milkbot.pl:30: register_hook("msg_in", \&on_im); +plugins/Milkbot.pl:41: hook("milkbot_command", -cmd => $cmd); +plugins/Milkbot.pl:42: hook("milkbot_command_$cmd", -user => $from, -msg => $msg); +plugins/Milkbot.pl:51: hook("protocol_send_im", -dest => $to, -msg => $msg, -away => $away); +plugins/Forum-Check/Forum-Check.pl:12:register_hook("pre_mainloop", sub { +plugins/Forum-Check/Forum-Check.pl:17: my $mw = hook("tk_getmain"); +plugins/Forum-Check/Forum-Check.pl:24: hook("error", -short => "New posts in the BATKINS forum."); +plugins/Forum-Check/Speak.pl:20:register_hook("msg_in", sub { say("You have a message from $ARGS{-user}"); }); +plugins/Win32X.pl:13:register_hook("flash_window", sub { +plugins/XAMP.pl:17:register_hook "signed_in", sub { +plugins/XAMP.pl:19: hook("protocol_commit_info"); +plugins/XAMP.pl:21: hook("tk_getmain")->repeat(5000, sub { +plugins/XAMP.pl:24: hook("protocol_commit_info") if $commit; +plugins/XAMP.pl:28: register_hook("request_mod", sub { +plugins/XAMP.pl:44: hook("protocol_mod_prof", -name => "%a", -value => $artist); +plugins/XAMP.pl:45: hook("protocol_mod_away", -name => "%a", -value => $artist); +plugins/XAMP.pl:56: hook("protocol_mod_prof", -name => "%s", -value => $song); +plugins/XAMP.pl:57: hook("protocol_mod_away", -name => "%s", -value => $song); +plugins/Net-OSCAR/Net/OSCAR.pm:2154:Added hooks to allow client do MD5 digestion for authentication (auth_challenge +plugins/Net-OSCAR/Net/OSCAR.pm:2612:Rocco Caputo for helping to work out the hooks that let use be used with +plugins/Net-OSCAR/Net-OSCAR.pl:28:register_hook("protocol_add_buddy", sub { $oscar->add_buddy($ARGS{-group}, $ARGS{-buddy}); } ); +plugins/Net-OSCAR/Net-OSCAR.pl:29:register_hook("protocol_away_status", sub { $is_away; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:30:register_hook("protocol_commit_blist", sub { $oscar->commit_buddylist; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:31:register_hook("protocol_commit_info", sub { $oscar->set_away($away) if $is_away; $oscar->set_info($prof); } ); +plugins/Net-OSCAR/Net-OSCAR.pl:32:register_hook("protocol_get_away", sub { $oscar->{away}; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:33:register_hook("protocol_get_groups", sub { [ $oscar->groups ]; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:34:register_hook("protocol_get_prof", sub { $oscar->{profile} } ); +plugins/Net-OSCAR/Net-OSCAR.pl:35:register_hook("protocol_get_realname", sub { $oscar->buddy($ARGS{-user})->{screenname} or $ARGS{-user}; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:36:register_hook("protocol_go_away", sub { $is_away = 1; hook("protocol_commit_info"); } ); +plugins/Net-OSCAR/Net-OSCAR.pl:37:register_hook("protocol_idle_since", sub { $oscar->buddy($ARGS{-who})->{idle}; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:38:register_hook("protocol_is_away", sub { return $oscar->buddy($ARGS{-user})->{away}; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:39:register_hook("protocol_is_on", sub { $oscar->buddy($ARGS{-who})->{online} } ); +plugins/Net-OSCAR/Net-OSCAR.pl:40:register_hook("protocol_mod_away", \&mod_away ); +plugins/Net-OSCAR/Net-OSCAR.pl:41:register_hook("protocol_mod_prof", \&mod_prof ); +plugins/Net-OSCAR/Net-OSCAR.pl:42:register_hook("protocol_on_since", sub { $oscar->buddy($ARGS{-who})->{onsince}; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:43:register_hook("protocol_remove_buddy", sub { $oscar->remove_buddy($ARGS{-group}, $oscar->buddy($ARGS{-buddy})->{screenname}); } ); +plugins/Net-OSCAR/Net-OSCAR.pl:44:register_hook("protocol_request_info", sub { request_info($ARGS{-user}); }); +plugins/Net-OSCAR/Net-OSCAR.pl:45:register_hook("protocol_return", sub { $oscar->set_away(''); %away_sent = (); $is_away = 0; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:46:register_hook("protocol_send_im", \&send_im ); +plugins/Net-OSCAR/Net-OSCAR.pl:47:register_hook("protocol_set_away", \&set_away ); +plugins/Net-OSCAR/Net-OSCAR.pl:48:register_hook("protocol_set_prof", \&set_prof ); +plugins/Net-OSCAR/Net-OSCAR.pl:49:register_hook("protocol_signon", \&signon); +plugins/Net-OSCAR/Net-OSCAR.pl:50:register_hook("protocol_signed_in", sub { $signed_in; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:51:register_hook("protocol_signoff", sub { $oscar->signoff; } ); +plugins/Net-OSCAR/Net-OSCAR.pl:64: hook("error", -short => $desc, -long => longmess($desc), -fatal => $fatal); +plugins/Net-OSCAR/Net-OSCAR.pl:69: hook("buddy_in", -buddy => $_[1], -group => $_[2]); +plugins/Net-OSCAR/Net-OSCAR.pl:74: hook("buddy_out", -buddy => $_[1], -group => $_[2]); +plugins/Net-OSCAR/Net-OSCAR.pl:79: hook("msg_in", -user => $_[1], -msg => $_[2], -away => $_[3]) if $_[1]; +plugins/Net-OSCAR/Net-OSCAR.pl:80: hook("msg_in_$_[1]", -user => $_[1], -msg => $_[2], -away => $_[3]); +plugins/Net-OSCAR/Net-OSCAR.pl:86: hook("signed_in", -me => $user); +plugins/Net-OSCAR/Net-OSCAR.pl:92: hook("error", +plugins/Net-OSCAR/Net-OSCAR.pl:95: hook("rate_alert") if $_[1] == RATE_LIMIT; +plugins/Net-OSCAR/Net-OSCAR.pl:104: hook("protocol_info_received_$_[1]", -profile => $prof, -away => $_[2]->{awaymsg}); +plugins/Net-OSCAR/Net-OSCAR.pl:115: hook("tk_getmain")->after(30, \&tick); +plugins/Net-OSCAR/Net-OSCAR.pl:130: hook("tk_getmain")->after(30, \&tick); +plugins/Net-OSCAR/Net-OSCAR.pl:141: hook("after", -time => 50, -code => [\&request_info, $other_user]); +plugins/Net-OSCAR/Net-OSCAR.pl:168: register_hook("request_mod", sub{} ); # for when those hooks just arent yet registered, use register_hook(tm) +plugins/Net-OSCAR/Net-OSCAR.pl:169: hook("request_mod"); +plugins/Net-OSCAR/Net-OSCAR.pl:178: register_hook("request_mod", sub{} ); # ditto, can't wait till hook priorities! +plugins/Net-OSCAR/Net-OSCAR.pl:179: hook("request_mod"); +plugins/Sound/Sound.pl:7:register_hook("signed_in", sub { play_sound("signed_in"); }); +plugins/Sound/Sound.pl:8:register_hook("protocol_go_away", sub { play_sound("go_away"); }); +plugins/Sound/Sound.pl:9:register_hook("protocol_return", sub { play_sound("return"); }); +plugins/Sound/Sound.pl:10:register_hook("msg_in", sub { play_sound("msg_in"); }); +plugins/Sound/Sound.pl:11:register_hook("protocol_send_im", sub { play_sound("send_im"); }); +plugins/Sound/Sound.pl:12:register_hook("buddy_in", sub { play_sound("buddy_in"); }); +plugins/Sound/Sound.pl:13:register_hook("buddy_out", sub { play_sound("buddy_out"); }); +plugins/Sound/Sound.pl:14:register_hook("error", sub { play_sound("error"); }); +plugins/Sound/Sound.pl:15:register_hook("protocol_signoff", sub { play_sound("signoff"); }); +plugins/Sound/Sound.pl:19: return if hook("protocol_away_status") and !option("SoundsWhileAway") and $_[0] ne "go_away"; +plugins/Tk-About/Tk-About.pl:6:my $mw = hook("tk_getmain"); +plugins/Tk-About/Tk-About.pl:8:register_hook "show_about", sub { $mw->MBAbout->focus; }; +plugins/Tk-AddBuddy/Milkbone/AddBuddy.pm:32: hook("protocol_add_buddy", -group => $group, -buddy => $name); +plugins/Tk-AddBuddy/Milkbone/AddBuddy.pm:33: hook("protocol_commit_blist"); +plugins/Tk-AddBuddy/Milkbone/AddBuddy.pm:41: my $groups = hook("protocol_get_groups"); +plugins/Tk-AddBuddy/Milkbone/AddBuddy.pm:63: hook("tk_seticon", -wnd => $self); +plugins/Tk-AddBuddy/Tk-AddBuddy.pl:9:register_hook("dlg_add_buddy", sub { +plugins/Tk-BList/Tk-BList.pl:12:my $mw = hook("tk_getmain"); +plugins/Tk-BList/Tk-BList.pl:14:register_hook("signed_in", sub { +plugins/Tk-BList/Tk-BList.pl:20: register_hook("buddy_in", sub { Milkbone::BList::on_buddy_in } , {-self => $blist}); +plugins/Tk-BList/Tk-BList.pl:21: register_hook("buddy_out", \&Milkbone::BList::on_buddy_out, {-self => $blist}); +plugins/Tk-BList/Tk-BList.pl:22: register_hook("add_blist_menu_item", \&Milkbone::BList::add_blist_menu_item, {-self => $blist}); +plugins/Tk-BList/Tk-BList.pl:24: register_hook("protocol_go_away", sub { +plugins/Tk-BList/Tk-BList.pl:30: register_hook("protocol_return", sub { +plugins/Tk-BList/Tk-BList.pl:37: register_hook("error_fatal", sub { +plugins/Tk-BList/Tk-BList.pl:41: register_hook("msg_in", sub { +plugins/Tk-BList/Tk-BList.pl:50:sub reg_change_hook +plugins/Tk-BList/Tk-BList.pl:54: register_hook("get_profile", sub { +plugins/Tk-BList/Milkbone/BList.pm:51: hook("create_convo", -user => hook("protocol_get_realname", -user => $name), -fabricated => 1); +plugins/Tk-BList/Milkbone/BList.pm:62: my $groups = hook("protocol_get_groups"); +plugins/Tk-BList/Milkbone/BList.pm:77: if(hook("protocol_is_away", -user => $realname)) +plugins/Tk-BList/Milkbone/BList.pm:89: if(hook("protocol_is_away", -user => $realname)) +plugins/Tk-BList/Milkbone/BList.pm:123: hook("protocol_go_away"); +plugins/Tk-BList/Milkbone/BList.pm:130: hook("protocol_return"); +plugins/Tk-BList/Milkbone/BList.pm:142: hook("protocol_set_info", -info => $info); +plugins/Tk-BList/Milkbone/BList.pm:199: hook("dlg_add_buddy", -parent => shift); +plugins/Tk-BList/Milkbone/BList.pm:209: hook("get_profile", -user => $buddy, -group => $group); +plugins/Tk-BList/Milkbone/BList.pm:221: hook("protocol_remove_buddy", -group => $group, -buddy => $name); +plugins/Tk-BList/Milkbone/BList.pm:222: hook("protocol_commit_blist"); +plugins/Tk-BList/Milkbone/BList.pm:223: hook("buddy_out", -buddy => $name, -group => $group); +plugins/Tk-BList/Milkbone/BList.pm:246: $self->{menu_file}->command(-label => "Set Away...", -command => sub { hook("on_set_away");} ); +plugins/Tk-BList/Milkbone/BList.pm:247: $self->{menu_file}->command(-label => "Set Profile...", -command => sub { hook("on_set_profile");} ); +plugins/Tk-BList/Milkbone/BList.pm:250: $self->{menu_file}->command(-label => "Goodbye and Exit", -command => sub { hook("goodbye") }); +plugins/Tk-BList/Milkbone/BList.pm:258: $self->{menu_help}->command(-label => "About...", -command => sub { hook("show_about") }); +plugins/Tk-BList/Milkbone/BList.pm:288: hook("tk_seticon", -wnd => $self); +plugins/Tk-BList/Milkbone/BList.pm:316: return unless hook("protocol_away_status"); +plugins/Tk-Convo/Milkbone/Convo.pm:53: hook("protocol_send_im", -dest => $self->{buddy}, -msg => $msg, -away => 0); +plugins/Tk-Convo/Milkbone/Convo.pm:69: hook("remove_convo", -user => shift->{buddy}); +plugins/Tk-Convo/Milkbone/Convo.pm:74: hook("get_profile", -user => shift->{buddy}); +plugins/Tk-Convo/Milkbone/Convo.pm:154: hook("tk_seticon", -wnd => $self); +plugins/Tk-Convo/Tk-Convo.pl:10:my $mw = hook("tk_getmain"); +plugins/Tk-Convo/Tk-Convo.pl:13:register_hook("create_convo", sub { +plugins/Tk-Convo/Tk-Convo.pl:24: register_hook("msg_in_$buddy", sub { +plugins/Tk-Convo/Tk-Convo.pl:27: hook("create_convo", -user => hook("protocol_get_realname", -user => $ARGS{-user})); +plugins/Tk-Convo/Tk-Convo.pl:30: hook("flash_window", -wnd => $convo); +plugins/Tk-Convo/Tk-Convo.pl:33: register_hook("buddy_in_$buddy", sub { +plugins/Tk-Convo/Tk-Convo.pl:37: register_hook("buddy_out_$buddy", sub { +plugins/Tk-Convo/Tk-Convo.pl:44: $convo->withdraw if hook("protocol_away_status") != 0; +plugins/Tk-Convo/Tk-Convo.pl:46: hook("flash_window", -wnd => $convo) unless $ARGS{-fabricated}; +plugins/Tk-Convo/Tk-Convo.pl:49:register_hook("remove_convo", sub { +plugins/Tk-Convo/Tk-Convo.pl:50: deregister_hook("msg_in_$ARGS{-user}"); +plugins/Tk-Convo/Tk-Convo.pl:51: deregister_hook("buddy_in_$ARGS{-user}"); +plugins/Tk-Convo/Tk-Convo.pl:52: deregister_hook("buddy_out_$ARGS{-user}"); +plugins/Tk-Convo/Tk-Convo.pl:58:register_hook("msg_in", sub { +plugins/Tk-Convo/Tk-Convo.pl:59: hook("create_convo", -user => hook("protocol_get_realname", -user => $ARGS{-user})); +plugins/Tk-Convo/Tk-Convo.pl:62:register_hook("get_convo", sub { +plugins/Tk-Convo/Tk-Convo.pl:66:register_hook("protocol_go_away", sub { +plugins/Tk-Convo/Tk-Convo.pl:70:register_hook("protocol_return", sub { +plugins/Tk-Convo/Tk-Convo.pl:74:register_hook("goodbye", sub { +plugins/Tk-Convo/Tk-Convo.pl:77: hook("protocol_send_im", -dest => $_, -msg => $goodbye, -away => 0) for(keys(%convos)); +plugins/Tk-Convo/Tk-Convo.pl:80:register_hook("buddy_in", sub { +plugins/Tk-Convo/Tk-Convo.pl:81: hook("buddy_in_$ARGS{-buddy}", -group => $ARGS{-group}) if $convos{$ARGS{-buddy}}; +plugins/Tk-Convo/Tk-Convo.pl:84:register_hook("buddy_out", sub { +plugins/Tk-Convo/Tk-Convo.pl:85: hook("buddy_out_$ARGS{-buddy}", -group => $ARGS{-group}) if $convos{$ARGS{-buddy}}; +plugins/Tk-File/Milkbone/File.pm:51: hook("tk_seticon", -wnd => $self); +plugins/Tk-File/Milkbone/File.pm:67: hook("protocol_set_prof", -data => $text); +plugins/Tk-File/Milkbone/File.pm:71: hook("protocol_set_away", -data => $text); +plugins/Tk-File/Tk-File.pl:10:my $mw = hook("tk_getmain"); +plugins/Tk-File/Tk-File.pl:12:register_hook "edit_file", sub { +plugins/Tk-File/Tk-File.pl:29:register_hook "on_set_profile", sub { +plugins/Tk-File/Tk-File.pl:30: hook("edit_file", -type => "profile"); +plugins/Tk-File/Tk-File.pl:33:register_hook "on_set_away", sub { +plugins/Tk-File/Tk-File.pl:34: hook("edit_file", -type => "away"); +plugins/Tk-File/Tk-File.pl:37:register_hook "signed_in", sub { +plugins/Tk-File/Tk-File.pl:57: hook("protocol_set_prof", -data => $prof); +plugins/Tk-File/Tk-File.pl:58: hook("protocol_set_away", -data => $away); +plugins/Tk-GUI/Tk-GUI.pl:38:register_hook("tk_seticon", sub { $ARGS{-wnd}->Icon(-image => $icon); }); +plugins/Tk-GUI/Tk-GUI.pl:40:register_hook("tick", \&tick); +plugins/Tk-GUI/Tk-GUI.pl:41:register_hook("tk_getmain", sub { +plugins/Tk-GUI/Tk-GUI.pl:45:register_hook("after", sub { +plugins/Tk-GUI/Tk-GUI.pl:61:register_hook("error", sub { +plugins/Tk-GUI/Tk-GUI.pl:64: hook("tk_seticon", -wnd => $error_box); +plugins/Tk-GUI/Tk-GUI.pl:72: hook("protocol_signoff") if $fatal; +plugins/Tk-GUI/Tk-GUI.pl:73: exit if $fatal and hook("protocol_signed_in"); +plugins/Tk-Logon/Milkbone/Logon.pm:14:my $mw = hook("tk_getmain"); +plugins/Tk-Logon/Milkbone/Logon.pm:45: hook("protocol_signoff") if $self->{signed_on}; +plugins/Tk-Logon/Milkbone/Logon.pm:59: hook("protocol_signon", -user => $self->{sname}, -pass => $self->{pass}); +plugins/Tk-Logon/Milkbone/Logon.pm:72: hook("protocol_signoff"); +plugins/Tk-Logon/Milkbone/Logon.pm:128: $self->{controls}->Button(-text => 'about', -command => sub { hook("show_about") }, +plugins/Tk-Logon/Milkbone/Logon.pm:140: hook("tk_seticon", -wnd => $self); +plugins/Tk-Logon/Tk-Logon.pl:9:my $mw = hook("tk_getmain"); +plugins/Tk-Logon/Tk-Logon.pl:12:register_hook("create_logon_prompt", sub { +plugins/Tk-Logon/Tk-Logon.pl:16: register_hook("signed_in", sub { +plugins/Tk-Logon/Tk-Logon.pl:21: register_hook("error", sub { +plugins/Tk-PluginsConf/Milkbone/PluginsConf.pm:72: hook("tk_seticon", -wnd => $self); +plugins/Tk-PluginsConf/Tk-PluginsConf.pl:9:register_hook("dlg_plugins", sub { +plugins/Tk-Profile/Milkbone/Profile.pm:57: hook("remove_profile", -who => $self->{who}); +plugins/Tk-Profile/Milkbone/Profile.pm:82: my $on_time = time - hook("protocol_on_since", -who => $who); +plugins/Tk-Profile/Milkbone/Profile.pm:88: if(hook("protocol_idle_since", -who => $who)) +plugins/Tk-Profile/Milkbone/Profile.pm:90: my $idle_time = hook("protocol_idle_since", -who => $who); +plugins/Tk-Profile/Milkbone/Profile.pm:110: hook("tk_seticon", -wnd => $self); +plugins/Tk-Profile/Tk-Profile.pl:9:my $mw = hook("tk_getmain"); +plugins/Tk-Profile/Tk-Profile.pl:11:register_hook("get_profile", sub { +plugins/Tk-Profile/Tk-Profile.pl:20: hook("protocol_request_info", -user => $who); +plugins/Tk-Profile/Tk-Profile.pl:24: register_hook("protocol_info_received_$who", sub { +plugins/Tk-Profile/Tk-Profile.pl:41:register_hook("remove_profile", sub { +plugins/Tk-Profile/Tk-Profile.pl:42: deregister_hook("protocol_info_received_" . $ARGS{-who}); +plugins/Win32-Tray/Win32-Tray.pl:12:register_hook("signed_in", sub { +plugins/Win32-Tray/Win32-Tray.pl:24: hook("tk_getmain")->repeat(500, sub { +plugins/Win32-Tray/Win32-Tray.pl:28: register_hook("post_mainloop", sub { +plugins/Milkbot-Xmms/Milkbot-Xmms.pl:27:register_hook("milkbot_get_commands", \&init); +plugins/Milkbot-Xmms/Milkbot-Xmms.pl:70: hook("milkbot_set_command", -name => $_, -desc => $commands{$_}[1]); +plugins/Milkbot-Xmms/Milkbot-Xmms.pl:72: register_hook("milkbot_command", sub { $cmd = $ARGS{-cmd}; +plugins/Milkbot-Xmms/Milkbot-Xmms.pl:73: register_hook("milkbot_command_$cmd", sub { +plugins/Milkbot-Xmms/Milkbot-Xmms.pl:75: deregister_hook("milkbot_command_$cmd") if $cmd; +plugins/Milkbot-Xmms/Milkbot-Xmms.pl:82: hook("protocol_send_im", -dest => $to, -msg => $msg, -away => $away); diff --git a/docs/KLUDGES.txt b/docs/KLUDGES.txt new file mode 100644 index 0000000..d8610d2 --- /dev/null +++ b/docs/KLUDGES.txt @@ -0,0 +1 @@ +the browser system diff --git a/docs/OSCARMODS.txt b/docs/OSCARMODS.txt new file mode 100644 index 0000000..d40c349 --- /dev/null +++ b/docs/OSCARMODS.txt @@ -0,0 +1,2 @@ +Accepts typing notifications +Retrieves mobile information \ No newline at end of file diff --git a/docs/README.txt b/docs/README.txt new file mode 100644 index 0000000..625542f --- /dev/null +++ b/docs/README.txt @@ -0,0 +1,57 @@ +milkbone 0.13 README +------------------------------------- + +milkbone is a rewrite of the AOL Instant Messenger client program for the Win32 operating system. milkbone is +written entirely in perl (well, there's a BIT of C) and is open-source freeware. milkbone may be freely +redistributed under the GPL as long as this file remains intact and any modifications to the core are noted. + +USE +------------------------------------ + +Win32 +----- + +To run milkbone, simply execute mb.exe or, if you have Perl installed on your system, run src\mb.pl (from the source +distro). + +Log on with your username and password. Then just go for it. :) + +UNIX/Linux +---------- + +milkbone is designed for use on Win32 systems. However, since the milkbone core is written in perl and uses the +Perl/Tk library, milkbone will also run on UNIX-based systems. See COMPATIBILITY.txt for known compatibility issues. + +Mac OS +------ + +Milkbone should work on Mac OS X, although this has never been tested. Anywone who feels like testing can go +ahead and do so. + +DEVELOPMENT +------------------------------------ + +milkbone was written and is maintained by the aimLess Consortium. The following individuals are members of the +Consortium's development team at this time: + +Bill Atkins (thebone@batkins.com) Lead Programmer +Sidharth Malhotra Programmer +Eric Thul Programmer and UNIX Porter +Dan Chokola Programmer + +The Consortium is always looking for Perl programmers to lend us a hand. Contact us if you're interested. + +The milkbone core is distributed from http://milkbone.batkins.com/ The milkbone mailing list is available at +aimless@batkins.com. + +DONATIONS +----------------------------------- + +The aimLess Consortium is always willing to accept donations. If you want to help milkbone by donating a few dollars, contact +thebone@batkins.com. Remember, we do all this for free; a little donation would make the entire Consortium feel good +about itself. + +CONTACT US +------------------------------------- + +Please send any comments/suggestions/concerns to thebone@batkins.com. \ No newline at end of file diff --git a/docs/SLOGANS.txt b/docs/SLOGANS.txt new file mode 100644 index 0000000..cb34740 --- /dev/null +++ b/docs/SLOGANS.txt @@ -0,0 +1,9 @@ +it's a dog-eat-dog world, and I'm wearing milkbone underwear. +if you have't tried milkbone yet, then perhaps you're a nazi. +milkbone is as stable as a frying pan +milkbone > milk + bone +an amalgam of sweetness +milkbone - now with mystical kung-fu power +it's kosher + it's all set and poised to serioualy mess up aim and its evil monopoly, so join the revolution. + bringing the spunk that is milkbone" \ No newline at end of file diff --git a/docs/SPECS.txt b/docs/SPECS.txt new file mode 100644 index 0000000..cd7d290 --- /dev/null +++ b/docs/SPECS.txt @@ -0,0 +1,34 @@ + # ----------------------------------------------------------------------------- +# Bill Atkins and Eric Thul +# Milkbone DesignDoc +# 12.21.02 +# ----------------------------------------------------------------------------- + + +1. Problem Statement: + a. Goal - To create an enhanced version of AIM and establish a new messaging protocol. + b. Constraints - Run on multi platforms ( UNIX, MAC OSX, WIN32 ), also use the Oscar protocol for communication with AIM and AOL IM. Have a pluggable backplane and be modular. Use minimal memory and hard drive space. Provide the major features of AIM and much more, ex: Buddylists, Chatting, IM, Icons, Profiles. Make the core as small and basic as possible. + c. Inputs - Username and password ( AIM info), also text and pictures, away messages, profiles. + d. Output - Messages from other users, profiles, icons, logs. + e. Simplification - make a totally modular program with each component developed separately and able to be plugged into the core. + +2. Relevent Objects ( modules ): + a. Core + b. Protocol ( backend ) + c. GUI ( frontend ) + ____ Event Handles ( hooks ) ____ + d. BuddyList + e. ProfileManager + f. ChatManager + g. IMManager + h. AwayMessage + i. Preferences + j. Logger + k. Login + +3. N/A + +4. Open Issues: + a. Establishing the new protocol may pose a bit of a challenge with running a server and all the other connections to Oscar and other char client protocols. + b. To break the char limit on the profiles. + c. todo list ( see relevant file ) \ No newline at end of file diff --git a/docs/VERSION.txt b/docs/VERSION.txt new file mode 100644 index 0000000..63887f3 --- /dev/null +++ b/docs/VERSION.txt @@ -0,0 +1,463 @@ +0.355 +-- + +* Removed PAR usage +* Plugin reloading works pretty darned well +* Uses ~/.milkbone on Linux, AppData on Win2K, XP, etc. and profiles on Win98, etc.; needs testing on NT +* Reloading of the core (Milkbone.pm) is now supported) +* mb.conf is only saved if changed +* Reloading of plugins works +* Plugin list now updates properly +* Sounds off by default +* Cleaned out some useless files +* Removed some useless sounds +* Now runs on Perl 5.6 on non-Win32 machines +* Tray icon now allows hiding and showing of windows (enabled by default) +* XAMP rewritten - it's bunches cleaner now +* Timeout renamed to AwayTimeout +* Fixed goodbye bug (again) +* Duplicate convo bug fixed +* Line break thing in profiles is gone +* Errors no longer kill milkbone on Linux +* Fixed font sizes on Linux + * Fancier toolbar +* < and > work in profiles and convos (you can send them as < and > for now) +* Splash screen +* Mouse wheel works in Linux +* No more boxes in focused widgets on Linux +* Sounds now play asynchronously on Linux (without starting too many play processes) +* Fixed weirdness with context menu on Linux +* Fancier buttons and menus on Linux (a LOT fancier) +* Preliminary aliasing support +* Alias-saving works (but is disabled) +* Sound works on Linux (without running artsd) +* Really fixed the Busted SNAC bug + +0.354 6-9-2003 +--- + +* May have fixed the Busted SNAC/ Empty Convo Bug (hopefully it is fixed, because that name is WAY too long) +* Fixed newline problem in profiles + +0.353 5-4-2003 +--- + +* Typing status doesn't clear after sending +* Mouse-wheeling in buddy list +* Smaller buddy list scrollbar +* New buddy list font +* Fixed goodbye +* Fixed protocol_send_im +* Extraneous timers now die instead of eating CPU time; this is a good thing +* Added support for chat to net-oscar (Tk-Chat plugin forthcoming) +* buddy_in and buddy_info_changed are now separate hooks +* Add buddy group dialog + +0.352a 4-30-2003 +--- + +* Fixed a deadly profile-viewing bug + +0.352 4-30-2003 +--- + +* Underline support +* Increased the time required to get a rate error for viewing profiles +* Scrolls properly after a buddy comes in or out +* Tray icon is disabled until I figure out why it slows down the boot process so much (this makes for faster loading) +* Toolbar buttons depress appropriately +* Sends more timely notifications +* Includes the 0.351a patch +* Minor problems with typing notifications +* Added support for sending underlines +* Can set style of current text by clicking toolbar buttons +* No more errors from the toolbar +* Empty convo title bug might be gone :) + +0.351a 4-26-2003 +--- + +* Fixed some bugs in typing notification that caused rate errors + +0.351 4-26-2003 +--- + +* send_im adds to the convo window (for milkbot) +* Fixed a reregistration bug in Net-OSCAR +* Putting %v in your profile or away will reveal your milkbone version +* Sends typing notifications +* Displays typing notifications +* Beta of a plugin loader (soon there'll be an unloader, too) +* Fixed Goodbye and Exit +* Fixed warnings from XAMP +* Much-improved combo boxes +* No flicker when loading AddBuddy +* Beta of FCL +* Better error-handling (less death) +* Now indicates whether a buddy is on a cell phone (with an image pirated from AIM :D ) +* deregister_hook is now decidedly functional +* Plugin Configurator +* Can now send out <'s and >'s to aim users +* Fixed bug in convo titles (related to getting the buddy's real name) +* Milkbot and Milkbot-Music now work on Win32 +* Fixed background color bugs in convos +* Speeded up Tk-GUI's load time + +0.35 4-20-2003 +--- + +* Milkbot split into generic module and music-specific module (dan chokola) +* Some work on FCL +* Patched Tk - cut down on a few memory leaks +* Now autoscrolls again in convos +* Fixed bug when receiving IM's from nonbuddies +* Works with Winamp 3.0 +* Scrolling titles in Winamp +* Timestamps (hit F2 in a convo) +* Tray icon with Exit (on Win32) + +0.341 4-13-2003 +--- + +* Sends < and > properly +* Fixed error after looking at size-adjusted fonts +* Profiles still look pretty when you maximize them +* Fixed extra newlines at the end of aways and profiles +* Fixed Cancel bug in AddBuddy + +0.34 4-11-2003 +--- + +* Fixed the newline-formatting bug in Tk-Convo +* The Add Buddy box now has a dropdown (a cheap-looking dropdown, but a dropdown nonetheless) +* CVS server set up (link at http://milkbone.batkins.com) +* Fixed font size problem +* Fixed color problems in profiles (pointed out by greg) +* Fixed add buddy dialog (pointed out by mario) + +0.333a 4-10-2003 +--- + +* Fixed the silly bug in the set profile boxes (thanks for breaking it, dan! ;) ) + +0.333 4-10-2003 +--- + +* FCL works in profiles and aways +* Added support for font faces and colors (sizes coming soon) +* Nested tags work (no more 's) +* Added support for underlined text + +0.332a 4-9-2003 +--- + +* Profile focusing works better +* Fixed Add Buddy dialog +* Fixed Remove Buddy +* About box now has escape enabled +* Fixed lag in edit boxes +* Prepare for CVS..... (next version) +* Reorganized the source tree +* Fixed away message counter + +0.332 4-8-2003 +--- + +* Buddy list now tracks number of waiting messages +* Waiting message count in buddy list +* XAMP is ready (the rewrite of AIMAMP) (dan chokola) +* Top window is no longer editable +* Automatic compression script in src/plugins +* Links in messages +* Better indenting in the buddy list (dan chokola) +* Away icons that are just slightly cooler +* Plugins are in ZIP's instead of PAR's for easier access (suggested by DJ) +* Rate errors with profiles are (almost) gone +* Error messages no longer crash milkbone +* Formatting screen names now works properly with buddy list +* Tray icon in win32 (experimental) +* No more gap in the convo windows (kudos to dan chokola for pointing out my wrongness) +* Plugin order in mb.conf no longer affects operation +* Profiles now work like browsers +* Current version now shown in about box, profile, and logon screen +* Expiration message gone +* New conversations flash in the taskbar +* Tk-AddBuddy now a separate plugin +* AOL profile message works a little better (dan chokola) +* Profile color and font revised (dan chokola) +* Sound on Linux (dan chokola) +* Tk-GUI now contains cross-platform Tk code (replaces Tk-Win32 and Tk-Linux) +* <, <, &, and " now appear correctly in conversations +* AIMAMP now works on Windows and Linux +* AIMAMP plugin separates artist and title (dan chokola) +* (Much) better support for profile and away templates (dan chokola) +* Carriage return issues in convos fixed (dan chokola) + +0.331 2-18-2003 +--- + +* Fixed a bug in the browser that caused it to retain text settings when it shouldn't have +* MUCH better processor usage (0% most of the time, with peaks of around 8%) + +0.33 2-17-2003 +--- + +* Preliminary support for FCL (fonts, colors, links) - bold text is now properly displayed +* Brand-new convos now flash in the title bar +* Fixed an insidious bug in AIMAMP (pointed out by eric) + +0.323 2-15-2003 +--- + +* Less CPU Usage (pointed out by dj and tyler) +* Proper icon for away/profile editor +* Errlog is now opened only when necessary so you can empty it while milkbone's still running +* AIMAMP errors gone (pointed out by eric) +* Error log now never exceeds 500K (pointed out by eric) +* Fixed those annoying 'signed in' messages + +0.322 2-9-2003 +--- + +* REALLY fixed the random crashes + +0.321 2-9-2003 +--- + +* Fixed AIMAMP errors in errlog.txt +* Fixed repetitive buddy in messages +* Fixed random crash nonsense + +0.32 2-9-2003 +--- + +* Preliminary support for remembering the size of convo windows +* Configuration changes are now saved +* " no longer appears in profiles +* Warning for rate limits +* Had to disable the combo in AddBuddy - it's all manual until next version +* You no longer have to change songs after changing your profile +* %s now works properly in away messages +* Errors no longer kill milkbone (unless they're fatal - then they do) +* You can talk to people who aren't on your buddy list without getting hash errors +* Can add buddies into a specific group +* Buddy deletion actually works now +* Idle time display +* Empty groups show up +* Buddy groups in order + +0.31 2-7-2003 +--- + +* Built-in AIMAMP (put %s in your away message or profile) +* Online time in profile window +* An even suaver buddy list +* Multiple buddy selection has become an oxymoron +* By switching from GIF's to BMP's, milkbone uses less RAM +* Remove buddies from the context menu (molhotra) +* Add buddies from the File menu (molhotra) +* Away messages will only be returned every Timeout seconds (thanks to greg for the suggestion) +* Away messages are ow sent to incoming messages (thanks to dan for pointing this out) +* Convos now hide when going away +* About box works +* Tab problem is now fully fixed (no message boxes) +* You can now save away messages again (thanks, mundane detail!!) +* Fixed focusing problems (mostly - profiles are still a little funky) +* Added preliminary (non-working) code for buddy coloring +* Buddy list now properly updates when a user logs out (this worked before the Rewrite, of course) +* Added a Timeout directive +* Get Buddy Info bug may be fixed +* PAR is now actually used +* Initialization file is now $plugin.pl instead of main.pl +* Error logging actually works (props to bwahl for pointing this out) +* Properly removes buddies from list on signout +* Focuses new convos and profiles +* Writes to errlog are now appended instead of overwritten + +0.3 2-5-2003 +--- + +* Goodbye directive +* Buddy list turns gray when you go away (and that even rhymes) +* Right-clicking for profiles (molhotra) +* User data goes into profiles folders +* Tk compiled into milkbone.exe because of PAR problems +* %n replacement +* The Port directive allows TCNJ and Stockton students (and many others) to finally get through their firewalls +* StdLib module contains standard Perl modules +* Assorted aesthetic improvements to the Profile module - icon, nice title +* Unlimited Redo's/Undo's with Ctrl-Z +* More robust profile retrieving (this took a while for some reason) +* Hook registration and deregistration +* More informative deaths +* MUCH smaller distro (thanks to PAR and UPX) +* Fixed the infamous Tab Problem +* Slicker away message icons +* Global configuration in mb.conf +* Eliminated the tray icon (it will return when it decides to do something) +* Plugins can be written as single files (idea from molhotra) +* Hitting enter in mid-text no longer breaks up stream +* Recompiled for use with Perl 5.8 +* Suave-looking buddy lists +* The --load command-line option will unpack all dependencies (except Tk, which is the one that matters :) ) +* PAR is now used for decompression instead of perlapp +* Code can now be plugged into milkbone, although no documentation for this has been written +* Completely redesigned the core + +0.21 12-9-2 +---- + +* New logo +* Prevented (hopefully) a bug that made the bone quit at random + +0.2 12-8-2 +---- + +* Extended time for update-checking +* Created the filemap +* Shaved around 600K from the final executable +* Removed useless images; replaced POSIX with straight constants +* Replaced the bloated LWP package with HTTP::Lite +* New logo +* Recompiled everything for compatibility with Perl 5.8 +* Bought the PDK (no more expirations :) ) +* Preliminary hooking +* Logging to errlog.txt (for el gato :) ) +* Prettier deaths (with message boxes) +* Installer will now overwrite shortcuts - this is a Good Thing +* Milkbone::OS now separates OS-dependent code from the core +* All modules now stored in src/lib +* Checks for empty messages before going away +* Re-released 0.161 + +0.161 11-26-2 +---- + +* When XP groups your convos in the taskbar, they show up as "milkbone" instead of "mb" +* Editing of multiple-line profiles works +* Hides messages and changes button when away +* Tweaked the compilation options to reduce milkbone's startup time to just under 1.5 seconds +* Goodbye and Exit now clears the tray icon (chokola) +* Now logins (as well as logouts) are noted in the conversation window (this was a LOT harder than you'd think :) ) +* Tweaked the update system to prevent annoying messages that might appear behind a proxy + +0.16 11-24-2 +---- + +* Update alerts +* Taskbar windows flash + +0.155 11-23-2 +---- + +* Cleaned up the installer script +* About box from menu +* Set aways and profiles from File menu +* Added an annoying drumming for new messages until I can get flashing to work (it only drums + if the new message isn't in the focus +* Logo bar above buddy list +* Fixed " and <-type strings in profiles, aways, and messages +* More compact buddy list +* Fixed bug that killed milkbone when user looks at a non-away user's profile (after viewing this profile, + viewing any other profiles would crash milkbone) +* Allowed messages that come in while away to be displayed +* Messed around with the hashes to fix the SNAC error +* Overloaded CORE::GLOBAL::die to prevent improper signoff errors (which led to inexplcable busted SNAC's) and + and to prevent the tray icon from living after death + + +0.154 11-21-2 +---- + +* Profiles scroll (with mousewheel) and wrap and respond to the escape button +* Fixed auto-response (chokola) +* Times instead of courier in profiles + +0.153 11-19-2 +---- + +* Can set profiles +* Can set away messages +* Removed the silly buttons +* Added away messages to profiles +* Made path names more portable to make UNIX ports easier +* Added code to lessen the chance that the icon would remain in the tray after a crash + +0.152 11-14-02 +---- + +* Away icons +* Added Get Profile function (uses the world's greatest kludge to accomplish viewing of profiles) +* Spiffed up the about box and gave it its own class +* Set up status bot on screen name billiamive +* Moved misplaced functions in mb.pl +* Plus and minus buttons +* Installer now sends documentation as well + + +0.151 11-14-02 +---- + +* Funky buttons +* milkbonestatus (edit: billiamive) screenname will now reveal the status of the project +* About box +* Signed off message is inserted into conversation +* Removed autofocus nonsense +* Added code to flash windows - doesn't seem to work +* Word-wrap + +0.15 11-13-02 +---- + +* Realtime updating of buddy list +* Now displays an error box when AIM has a problem +* Incoming and outgoing messages work with buddy list (I think) +* Buddy list +* New "Connecting to" text :) +* Removed a very sneaky bug that logged users off if they hit Enter at the logon screen + +0.14 11-12-02 +---- + +* New compiler +* Screen name formatting works (e.g. Gato Gregorio instead of gatogregorio) +* TIMING ISSUES FIXED!!!! - a tweak in MBLoop has eliminated the flicker - WOOT! +* Can now cancel and then sign in again without lockup +* First installer +* Tested new logo +* All windows use bone icon +* Fixed a bug that might have appeared when two similar screen names sent IMs +* Automated test and build scripts +* Fixed illegal op problem +* File | Close closes IM windows +* Escape key closes IMs + +0.131 11-0-02 +---- + +* Minor bugfixes + +0.13 11-08-02 +---- + +* rewritten for use with Matthew Sachs' Net::OSCAR +* allowed incoming IM's +* allowed conversations +* tweaked Net::OSCAR to remove dependence on Scalar::Util's bootstrap + +0.12 10-13-02 +---- + +* fixed timing issues with 0.10 +* fixed minor interface bugs + +0.11 +---- + +NONEXISTENT - version numbering error + + +0.10 0-12-02 +---- + +* connected to AIM with the TOC protocol (Aryeh Goldsmith's Net::AIM) \ No newline at end of file diff --git a/docs/WISHBONE.txt b/docs/WISHBONE.txt new file mode 100644 index 0000000..c96d28b --- /dev/null +++ b/docs/WISHBONE.txt @@ -0,0 +1,31 @@ +convos shouldnt start minimized +warning for zips +chat capabilities +buddy icons +direct connect +custom keys +sarcasm +a real line in the away messages +CLEANER CODE!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! +history (hitting up brings last message) +switch screen name +minimized convos +secure convos (with fortis?) +shoudl only flash if not active +ctrl-r restores last window +hooks - should flag for only one; should be able to add to top or bottom of queue +Tk-Profile and Tk-Convo classes as spiffy-looking as Tk-BList and Tk-Logon +multiple connections +sorted buddy lists +auto-splitting +buddy conf files +MUCH bettter configuration handling +Refocuses convos windows that have been hidden when double-clicked in the buddy list +emotes +show current track +buddy ordering +chat rooms +logon saver +add buddy groups +documentation +file transfer \ No newline at end of file diff --git a/docs/milkbone.pod b/docs/milkbone.pod new file mode 100644 index 0000000..e69de29 diff --git a/errlog.txt b/errlog.txt new file mode 100644 index 0000000..b305b7a --- /dev/null +++ b/errlog.txt @@ -0,0 +1,20 @@ +Your message to could not be sent for the following reason: PERLmonk86 is not logged in, so the attempted operation (sending an IM, getting user information) was unsuccessful +Your message to could not be sent for the following reason: PERLmonk86 is not logged in, so the attempted operation (sending an IM, getting user information) was unsuccessful at plugins/Net-OSCAR/Net/OSCAR.pm line 1852 + Net::OSCAR::do_callback('error','Net::OSCAR=HASH(0x889a074)','Net::OSCAR::Connection=HASH(0x88abd0c)',4,'Your message to could not be sent for the following reason: P...',0) called at plugins/Net-OSCAR/Net/OSCAR.pm line 1856 + Net::OSCAR::callback_error('Net::OSCAR=HASH(0x889a074)','Net::OSCAR::Connection=HASH(0x88abd0c)',4,'Your message to could not be sent for the following reason: P...',0) called at plugins/Net-OSCAR/Net/OSCAR/Common.pm line 323 + Net::OSCAR::Common::send_error('Net::OSCAR=HASH(0x889a074)','Net::OSCAR::Connection=HASH(0x88abd0c)',4,'Your message to could not be sent for the following reason: %...',0,'Net::OSCAR::Screenname=SCALAR(0x84c2cfc)') called at plugins/Net-OSCAR/Net/OSCAR/Callbacks.pm line 148 + Net::OSCAR::Callbacks::process_snac('Net::OSCAR::Connection=HASH(0x88abd0c)','HASH(0x8a38294)') called at plugins/Net-OSCAR/Net/OSCAR/Connection.pm line 339 + Net::OSCAR::Connection::process_one('Net::OSCAR::Connection=HASH(0x88abd0c)',1,0) called at plugins/Net-OSCAR/Net/OSCAR.pm line 459 + Net::OSCAR::process_connections('Net::OSCAR=HASH(0x889a074)','SCALAR(0x84c2ccc)','SCALAR(0x8a2c234)','SCALAR(0x84c2cd8)') called at plugins/Net-OSCAR/Net/OSCAR.pm line 495 + Net::OSCAR::do_one_loop('Net::OSCAR=HASH(0x889a074)') called at plugins/Net-OSCAR/Net-OSCAR.pl line 202 + NetOSCAR::tick() called at /usr/lib/perl5/site_perl/5.8.0/i586-linux/Tk/After.pm line 83 + eval {...} called at /usr/lib/perl5/site_perl/5.8.0/i586-linux/Tk/After.pm line 83 + Tk::After::once('Tk::After=ARRAY(0x8a373c4)') called at plugins/Tk-GUI/Tk-GUI.pl line 61 + eval {...} called at plugins/Tk-GUI/Tk-GUI.pl line 61 + TkGUI::tick() called at lib/Milkbone/HookEntry.pm line 41 + Milkbone::HookEntry::call('Milkbone::HookEntry=HASH(0x85dd320)') called at Milkbone.pm line 245 + eval {...} called at Milkbone.pm line 245 + Milkbone::hook('tick') called at Milkbone.pm line 288 + Milkbone::MOSLoop() called at /home/bill/milkbone/mos.pl line 53 + main::main() called at /home/bill/milkbone/mos.pl line 59 + diff --git a/images/away.bmp b/images/away.bmp new file mode 100644 index 0000000..b0da5c6 Binary files /dev/null and b/images/away.bmp differ diff --git a/images/blank.gif b/images/blank.gif new file mode 100644 index 0000000..ca2b3ae Binary files /dev/null and b/images/blank.gif differ diff --git a/images/cell.gif b/images/cell.gif new file mode 100644 index 0000000..3b87e87 Binary files /dev/null and b/images/cell.gif differ diff --git a/images/icon.bmp b/images/icon.bmp new file mode 100644 index 0000000..6c3f4ba Binary files /dev/null and b/images/icon.bmp differ diff --git a/images/icon.jpg b/images/icon.jpg new file mode 100644 index 0000000..1195b8a Binary files /dev/null and b/images/icon.jpg differ diff --git a/images/logo.bmp b/images/logo.bmp new file mode 100644 index 0000000..556b10e Binary files /dev/null and b/images/logo.bmp differ diff --git a/images/logon.bmp b/images/logon.bmp new file mode 100644 index 0000000..09467ed Binary files /dev/null and b/images/logon.bmp differ diff --git a/images/mbone.ico b/images/mbone.ico new file mode 100644 index 0000000..d2990d0 Binary files /dev/null and b/images/mbone.ico differ diff --git a/images/splash.bmp b/images/splash.bmp new file mode 100644 index 0000000..556b10e Binary files /dev/null and b/images/splash.bmp differ diff --git a/lib/File/Temp.pm b/lib/File/Temp.pm new file mode 100644 index 0000000..3c38963 --- /dev/null +++ b/lib/File/Temp.pm @@ -0,0 +1,1876 @@ +package File::Temp; + +=head1 NAME + +File::Temp - return name and handle of a temporary file safely + +=begin __INTERNALS + +=head1 PORTABILITY + +This module is designed to be portable across operating systems +and it currently supports Unix, VMS, DOS, OS/2, Windows and +Mac OS (Classic). When +porting to a new OS there are generally three main issues +that have to be solved: + +=over 4 + +=item * + +Can the OS unlink an open file? If it can not then the +C<_can_unlink_opened_file> method should be modified. + +=item * + +Are the return values from C reliable? By default all the +return values from C are compared when unlinking a temporary +file using the filename and the handle. Operating systems other than +unix do not always have valid entries in all fields. If C fails +then the C comparison should be modified accordingly. + +=item * + +Security. Systems that can not support a test for the sticky bit +on a directory can not use the MEDIUM and HIGH security tests. +The C<_can_do_level> method should be modified accordingly. + +=back + +=end __INTERNALS + +=head1 SYNOPSIS + + use File::Temp qw/ tempfile tempdir /; + + $dir = tempdir( CLEANUP => 1 ); + ($fh, $filename) = tempfile( DIR => $dir ); + + ($fh, $filename) = tempfile( $template, DIR => $dir); + ($fh, $filename) = tempfile( $template, SUFFIX => '.dat'); + + $fh = tempfile(); + +MkTemp family: + + use File::Temp qw/ :mktemp /; + + ($fh, $file) = mkstemp( "tmpfileXXXXX" ); + ($fh, $file) = mkstemps( "tmpfileXXXXXX", $suffix); + + $tmpdir = mkdtemp( $template ); + + $unopened_file = mktemp( $template ); + +POSIX functions: + + use File::Temp qw/ :POSIX /; + + $file = tmpnam(); + $fh = tmpfile(); + + ($fh, $file) = tmpnam(); + $fh = tmpfile(); + + +Compatibility functions: + + $unopened_file = File::Temp::tempnam( $dir, $pfx ); + +=begin later + +Objects (NOT YET IMPLEMENTED): + + require File::Temp; + + $fh = new File::Temp($template); + $fname = $fh->filename; + +=end later + +=head1 DESCRIPTION + +C can be used to create and open temporary files in a safe way. +The tempfile() function can be used to return the name and the open +filehandle of a temporary file. The tempdir() function can +be used to create a temporary directory. + +The security aspect of temporary file creation is emphasized such that +a filehandle and filename are returned together. This helps guarantee +that a race condition can not occur where the temporary file is +created by another process between checking for the existence of the +file and its opening. Additional security levels are provided to +check, for example, that the sticky bit is set on world writable +directories. See L<"safe_level"> for more information. + +For compatibility with popular C library functions, Perl implementations of +the mkstemp() family of functions are provided. These are, mkstemp(), +mkstemps(), mkdtemp() and mktemp(). + +Additionally, implementations of the standard L +tmpnam() and tmpfile() functions are provided if required. + +Implementations of mktemp(), tmpnam(), and tempnam() are provided, +but should be used with caution since they return only a filename +that was valid when function was called, so cannot guarantee +that the file will not exist by the time the caller opens the filename. + +=cut + +# 5.6.0 gives us S_IWOTH, S_IWGRP, our and auto-vivifying filehandls +# People would like a version on 5.005 so give them what they want :-) +use 5.005; +use strict; +use Carp; +use File::Spec 0.8; +use File::Path qw/ rmtree /; +use Fcntl 1.03; +use Errno; +require VMS::Stdio if $^O eq 'VMS'; + +# Need the Symbol package if we are running older perl +require Symbol if $] < 5.006; + + +# use 'our' on v5.6.0 +use vars qw($VERSION @EXPORT_OK %EXPORT_TAGS $DEBUG); + +$DEBUG = 0; + +# We are exporting functions + +use base qw/Exporter/; + +# Export list - to allow fine tuning of export table + +@EXPORT_OK = qw{ + tempfile + tempdir + tmpnam + tmpfile + mktemp + mkstemp + mkstemps + mkdtemp + unlink0 + }; + +# Groups of functions for export + +%EXPORT_TAGS = ( + 'POSIX' => [qw/ tmpnam tmpfile /], + 'mktemp' => [qw/ mktemp mkstemp mkstemps mkdtemp/], + ); + +# add contents of these tags to @EXPORT +Exporter::export_tags('POSIX','mktemp'); + +# Version number + +$VERSION = '0.13'; + +# This is a list of characters that can be used in random filenames + +my @CHARS = (qw/ A B C D E F G H I J K L M N O P Q R S T U V W X Y Z + a b c d e f g h i j k l m n o p q r s t u v w x y z + 0 1 2 3 4 5 6 7 8 9 _ + /); + +# Maximum number of tries to make a temp file before failing + +use constant MAX_TRIES => 10; + +# Minimum number of X characters that should be in a template +use constant MINX => 4; + +# Default template when no template supplied + +use constant TEMPXXX => 'X' x 10; + +# Constants for the security level + +use constant STANDARD => 0; +use constant MEDIUM => 1; +use constant HIGH => 2; + +# OPENFLAGS. If we defined the flag to use with Sysopen here this gives +# us an optimisation when many temporary files are requested + +my $OPENFLAGS = O_CREAT | O_EXCL | O_RDWR; + +unless ($^O eq 'MacOS') { + for my $oflag (qw/ BINARY NOINHERIT /) { + my ($bit, $func) = (0, "Fcntl::O_" . $oflag); + no strict 'refs'; + $OPENFLAGS |= $bit if eval { + # Make sure that redefined die handlers do not cause problems + # eg CGI::Carp + local $SIG{__DIE__} = sub {}; + local $SIG{__WARN__} = sub {}; + $bit = &$func(); + 1; + }; + } +} + +# On some systems the O_TEMPORARY flag can be used to tell the OS +# to automatically remove the file when it is closed. This is fine +# in most cases but not if tempfile is called with UNLINK=>0 and +# the filename is requested -- in the case where the filename is to +# be passed to another routine. This happens on windows. We overcome +# this by using a second open flags variable + +my $OPENTEMPFLAGS = $OPENFLAGS; +unless ($^O eq 'MacOS') { + for my $oflag (qw/ TEMPORARY /) { + my ($bit, $func) = (0, "Fcntl::O_" . $oflag); + no strict 'refs'; + $OPENTEMPFLAGS |= $bit if eval { + # Make sure that redefined die handlers do not cause problems + # eg CGI::Carp + local $SIG{__DIE__} = sub {}; + local $SIG{__WARN__} = sub {}; + $bit = &$func(); + 1; + }; + } +} + +# INTERNAL ROUTINES - not to be used outside of package + +# Generic routine for getting a temporary filename +# modelled on OpenBSD _gettemp() in mktemp.c + +# The template must contain X's that are to be replaced +# with the random values + +# Arguments: + +# TEMPLATE - string containing the XXXXX's that is converted +# to a random filename and opened if required + +# Optionally, a hash can also be supplied containing specific options +# "open" => if true open the temp file, else just return the name +# default is 0 +# "mkdir"=> if true, we are creating a temp directory rather than tempfile +# default is 0 +# "suffixlen" => number of characters at end of PATH to be ignored. +# default is 0. +# "unlink_on_close" => indicates that, if possible, the OS should remove +# the file as soon as it is closed. Usually indicates +# use of the O_TEMPORARY flag to sysopen. +# Usually irrelevant on unix + +# Optionally a reference to a scalar can be passed into the function +# On error this will be used to store the reason for the error +# "ErrStr" => \$errstr + +# "open" and "mkdir" can not both be true +# "unlink_on_close" is not used when "mkdir" is true. + +# The default options are equivalent to mktemp(). + +# Returns: +# filehandle - open file handle (if called with doopen=1, else undef) +# temp name - name of the temp file or directory + +# For example: +# ($fh, $name) = _gettemp($template, "open" => 1); + +# for the current version, failures are associated with +# stored in an error string and returned to give the reason whilst debugging +# This routine is not called by any external function +sub _gettemp { + + croak 'Usage: ($fh, $name) = _gettemp($template, OPTIONS);' + unless scalar(@_) >= 1; + + # the internal error string - expect it to be overridden + # Need this in case the caller decides not to supply us a value + # need an anonymous scalar + my $tempErrStr; + + # Default options + my %options = ( + "open" => 0, + "mkdir" => 0, + "suffixlen" => 0, + "unlink_on_close" => 0, + "ErrStr" => \$tempErrStr, + ); + + # Read the template + my $template = shift; + if (ref($template)) { + # Use a warning here since we have not yet merged ErrStr + carp "File::Temp::_gettemp: template must not be a reference"; + return (); + } + + # Check that the number of entries on stack are even + if (scalar(@_) % 2 != 0) { + # Use a warning here since we have not yet merged ErrStr + carp "File::Temp::_gettemp: Must have even number of options"; + return (); + } + + # Read the options and merge with defaults + %options = (%options, @_) if @_; + + # Make sure the error string is set to undef + ${$options{ErrStr}} = undef; + + # Can not open the file and make a directory in a single call + if ($options{"open"} && $options{"mkdir"}) { + ${$options{ErrStr}} = "doopen and domkdir can not both be true\n"; + return (); + } + + # Find the start of the end of the Xs (position of last X) + # Substr starts from 0 + my $start = length($template) - 1 - $options{"suffixlen"}; + + # Check that we have at least MINX x X (eg 'XXXX") at the end of the string + # (taking suffixlen into account). Any fewer is insecure. + + # Do it using substr - no reason to use a pattern match since + # we know where we are looking and what we are looking for + + if (substr($template, $start - MINX + 1, MINX) ne 'X' x MINX) { + ${$options{ErrStr}} = "The template must contain at least ". + MINX . " 'X' characters\n"; + return (); + } + + # Replace all the X at the end of the substring with a + # random character or just all the XX at the end of a full string. + # Do it as an if, since the suffix adjusts which section to replace + # and suffixlen=0 returns nothing if used in the substr directly + # and generate a full path from the template + + my $path = _replace_XX($template, $options{"suffixlen"}); + + + # Split the path into constituent parts - eventually we need to check + # whether the directory exists + # We need to know whether we are making a temp directory + # or a tempfile + + my ($volume, $directories, $file); + my $parent; # parent directory + if ($options{"mkdir"}) { + # There is no filename at the end + ($volume, $directories, $file) = File::Spec->splitpath( $path, 1); + + # The parent is then $directories without the last directory + # Split the directory and put it back together again + my @dirs = File::Spec->splitdir($directories); + + # If @dirs only has one entry (i.e. the directory template) that means + # we are in the current directory + if ($#dirs == 0) { + $parent = File::Spec->curdir; + } else { + + if ($^O eq 'VMS') { # need volume to avoid relative dir spec + $parent = File::Spec->catdir($volume, @dirs[0..$#dirs-1]); + $parent = 'sys$disk:[]' if $parent eq ''; + } else { + + # Put it back together without the last one + $parent = File::Spec->catdir(@dirs[0..$#dirs-1]); + + # ...and attach the volume (no filename) + $parent = File::Spec->catpath($volume, $parent, ''); + } + + } + + } else { + + # Get rid of the last filename (use File::Basename for this?) + ($volume, $directories, $file) = File::Spec->splitpath( $path ); + + # Join up without the file part + $parent = File::Spec->catpath($volume,$directories,''); + + # If $parent is empty replace with curdir + $parent = File::Spec->curdir + unless $directories ne ''; + + } + + # Check that the parent directories exist + # Do this even for the case where we are simply returning a name + # not a file -- no point returning a name that includes a directory + # that does not exist or is not writable + + unless (-d $parent) { + ${$options{ErrStr}} = "Parent directory ($parent) is not a directory"; + return (); + } + unless (-w _) { + ${$options{ErrStr}} = "Parent directory ($parent) is not writable\n"; + return (); + } + + + # Check the stickiness of the directory and chown giveaway if required + # If the directory is world writable the sticky bit + # must be set + + if (File::Temp->safe_level == MEDIUM) { + my $safeerr; + unless (_is_safe($parent,\$safeerr)) { + ${$options{ErrStr}} = "Parent directory ($parent) is not safe ($safeerr)"; + return (); + } + } elsif (File::Temp->safe_level == HIGH) { + my $safeerr; + unless (_is_verysafe($parent, \$safeerr)) { + ${$options{ErrStr}} = "Parent directory ($parent) is not safe ($safeerr)"; + return (); + } + } + + + # Now try MAX_TRIES time to open the file + for (my $i = 0; $i < MAX_TRIES; $i++) { + + # Try to open the file if requested + if ($options{"open"}) { + my $fh; + + # If we are running before perl5.6.0 we can not auto-vivify + if ($] < 5.006) { + $fh = &Symbol::gensym; + } + + # Try to make sure this will be marked close-on-exec + # XXX: Win32 doesn't respect this, nor the proper fcntl, + # but may have O_NOINHERIT. This may or may not be in Fcntl. + local $^F = 2; + + # Store callers umask + my $umask = umask(); + + # Set a known umask + umask(066); + + # Attempt to open the file + my $open_success = undef; + if ( $^O eq 'VMS' and $options{"unlink_on_close"} ) { + # make it auto delete on close by setting FAB$V_DLT bit + $fh = VMS::Stdio::vmssysopen($path, $OPENFLAGS, 0600, 'fop=dlt'); + $open_success = $fh; + } else { + my $flags = ( $options{"unlink_on_close"} ? + $OPENTEMPFLAGS : + $OPENFLAGS ); + $open_success = sysopen($fh, $path, $flags, 0600); + } + if ( $open_success ) { + + # Reset umask + umask($umask) if defined $umask; + + # Opened successfully - return file handle and name + return ($fh, $path); + + } else { + # Reset umask + umask($umask) if defined $umask; + + # Error opening file - abort with error + # if the reason was anything but EEXIST + unless ($!{EEXIST}) { + ${$options{ErrStr}} = "Could not create temp file $path: $!"; + return (); + } + + # Loop round for another try + + } + } elsif ($options{"mkdir"}) { + + # Store callers umask + my $umask = umask(); + + # Set a known umask + umask(066); + + # Open the temp directory + if (mkdir( $path, 0700)) { + # created okay + # Reset umask + umask($umask) if defined $umask; + + return undef, $path; + } else { + + # Reset umask + umask($umask) if defined $umask; + + # Abort with error if the reason for failure was anything + # except EEXIST + unless ($!{EEXIST}) { + ${$options{ErrStr}} = "Could not create directory $path: $!"; + return (); + } + + # Loop round for another try + + } + + } else { + + # Return true if the file can not be found + # Directory has been checked previously + + return (undef, $path) unless -e $path; + + # Try again until MAX_TRIES + + } + + # Did not successfully open the tempfile/dir + # so try again with a different set of random letters + # No point in trying to increment unless we have only + # 1 X say and the randomness could come up with the same + # file MAX_TRIES in a row. + + # Store current attempt - in principal this implies that the + # 3rd time around the open attempt that the first temp file + # name could be generated again. Probably should store each + # attempt and make sure that none are repeated + + my $original = $path; + my $counter = 0; # Stop infinite loop + my $MAX_GUESS = 50; + + do { + + # Generate new name from original template + $path = _replace_XX($template, $options{"suffixlen"}); + + $counter++; + + } until ($path ne $original || $counter > $MAX_GUESS); + + # Check for out of control looping + if ($counter > $MAX_GUESS) { + ${$options{ErrStr}} = "Tried to get a new temp name different to the previous value $MAX_GUESS times.\nSomething wrong with template?? ($template)"; + return (); + } + + } + + # If we get here, we have run out of tries + ${ $options{ErrStr} } = "Have exceeded the maximum number of attempts (" + . MAX_TRIES . ") to open temp file/dir"; + + return (); + +} + +# Internal routine to return a random character from the +# character list. Does not do an srand() since rand() +# will do one automatically + +# No arguments. Return value is the random character + +# No longer called since _replace_XX runs a few percent faster if +# I inline the code. This is important if we are creating thousands of +# temporary files. + +sub _randchar { + + $CHARS[ int( rand( $#CHARS ) ) ]; + +} + +# Internal routine to replace the XXXX... with random characters +# This has to be done by _gettemp() every time it fails to +# open a temp file/dir + +# Arguments: $template (the template with XXX), +# $ignore (number of characters at end to ignore) + +# Returns: modified template + +sub _replace_XX { + + croak 'Usage: _replace_XX($template, $ignore)' + unless scalar(@_) == 2; + + my ($path, $ignore) = @_; + + # Do it as an if, since the suffix adjusts which section to replace + # and suffixlen=0 returns nothing if used in the substr directly + # Alternatively, could simply set $ignore to length($path)-1 + # Don't want to always use substr when not required though. + + if ($ignore) { + substr($path, 0, - $ignore) =~ s/X(?=X*\z)/$CHARS[ int( rand( $#CHARS ) ) ]/ge; + } else { + $path =~ s/X(?=X*\z)/$CHARS[ int( rand( $#CHARS ) ) ]/ge; + } + + return $path; +} + +# internal routine to check to see if the directory is safe +# First checks to see if the directory is not owned by the +# current user or root. Then checks to see if anyone else +# can write to the directory and if so, checks to see if +# it has the sticky bit set + +# Will not work on systems that do not support sticky bit + +#Args: directory path to check +# Optionally: reference to scalar to contain error message +# Returns true if the path is safe and false otherwise. +# Returns undef if can not even run stat() on the path + +# This routine based on version written by Tom Christiansen + +# Presumably, by the time we actually attempt to create the +# file or directory in this directory, it may not be safe +# anymore... Have to run _is_safe directly after the open. + +sub _is_safe { + + my $path = shift; + my $err_ref = shift; + + # Stat path + my @info = stat($path); + unless (scalar(@info)) { + $$err_ref = "stat(path) returned no values"; + return 0; + }; + return 1 if $^O eq 'VMS'; # owner delete control at file level + + # Check to see whether owner is neither superuser (or a system uid) nor me + # Use the real uid from the $< variable + # UID is in [4] + if ($info[4] > File::Temp->top_system_uid() && $info[4] != $<) { + + Carp::cluck(sprintf "uid=$info[4] topuid=%s \$<=$< path='$path'", + File::Temp->top_system_uid()); + + $$err_ref = "Directory owned neither by root nor the current user" + if ref($err_ref); + return 0; + } + + # check whether group or other can write file + # use 066 to detect either reading or writing + # use 022 to check writability + # Do it with S_IWOTH and S_IWGRP for portability (maybe) + # mode is in info[2] + if (($info[2] & &Fcntl::S_IWGRP) || # Is group writable? + ($info[2] & &Fcntl::S_IWOTH) ) { # Is world writable? + # Must be a directory + unless (-d _) { + $$err_ref = "Path ($path) is not a directory" + if ref($err_ref); + return 0; + } + # Must have sticky bit set + unless (-k _) { + $$err_ref = "Sticky bit not set on $path when dir is group|world writable" + if ref($err_ref); + return 0; + } + } + + return 1; +} + +# Internal routine to check whether a directory is safe +# for temp files. Safer than _is_safe since it checks for +# the possibility of chown giveaway and if that is a possibility +# checks each directory in the path to see if it is safe (with _is_safe) + +# If _PC_CHOWN_RESTRICTED is not set, does the full test of each +# directory anyway. + +# Takes optional second arg as scalar ref to error reason + +sub _is_verysafe { + + # Need POSIX - but only want to bother if really necessary due to overhead + require POSIX; + + my $path = shift; + print "_is_verysafe testing $path\n" if $DEBUG; + return 1 if $^O eq 'VMS'; # owner delete control at file level + + my $err_ref = shift; + + # Should Get the value of _PC_CHOWN_RESTRICTED if it is defined + # and If it is not there do the extensive test + my $chown_restricted; + $chown_restricted = &POSIX::_PC_CHOWN_RESTRICTED() + if eval { &POSIX::_PC_CHOWN_RESTRICTED(); 1}; + + # If chown_resticted is set to some value we should test it + if (defined $chown_restricted) { + + # Return if the current directory is safe + return _is_safe($path,$err_ref) if POSIX::sysconf( $chown_restricted ); + + } + + # To reach this point either, the _PC_CHOWN_RESTRICTED symbol + # was not avialable or the symbol was there but chown giveaway + # is allowed. Either way, we now have to test the entire tree for + # safety. + + # Convert path to an absolute directory if required + unless (File::Spec->file_name_is_absolute($path)) { + $path = File::Spec->rel2abs($path); + } + + # Split directory into components - assume no file + my ($volume, $directories, undef) = File::Spec->splitpath( $path, 1); + + # Slightly less efficient than having a function in File::Spec + # to chop off the end of a directory or even a function that + # can handle ../ in a directory tree + # Sometimes splitdir() returns a blank at the end + # so we will probably check the bottom directory twice in some cases + my @dirs = File::Spec->splitdir($directories); + + # Concatenate one less directory each time around + foreach my $pos (0.. $#dirs) { + # Get a directory name + my $dir = File::Spec->catpath($volume, + File::Spec->catdir(@dirs[0.. $#dirs - $pos]), + '' + ); + + print "TESTING DIR $dir\n" if $DEBUG; + + # Check the directory + return 0 unless _is_safe($dir,$err_ref); + + } + + return 1; +} + + + +# internal routine to determine whether unlink works on this +# platform for files that are currently open. +# Returns true if we can, false otherwise. + +# Currently WinNT, OS/2 and VMS can not unlink an opened file +# On VMS this is because the O_EXCL flag is used to open the +# temporary file. Currently I do not know enough about the issues +# on VMS to decide whether O_EXCL is a requirement. + +sub _can_unlink_opened_file { + + if ($^O eq 'MSWin32' || $^O eq 'os2' || $^O eq 'VMS' || $^O eq 'dos' || $^O eq 'MacOS') { + return 0; + } else { + return 1; + } + +} + +# internal routine to decide which security levels are allowed +# see safe_level() for more information on this + +# Controls whether the supplied security level is allowed + +# $cando = _can_do_level( $level ) + +sub _can_do_level { + + # Get security level + my $level = shift; + + # Always have to be able to do STANDARD + return 1 if $level == STANDARD; + + # Currently, the systems that can do HIGH or MEDIUM are identical + if ( $^O eq 'MSWin32' || $^O eq 'os2' || $^O eq 'cygwin' || $^O eq 'dos' || $^O eq 'MacOS') { + return 0; + } else { + return 1; + } + +} + +# This routine sets up a deferred unlinking of a specified +# filename and filehandle. It is used in the following cases: +# - Called by unlink0 if an opened file can not be unlinked +# - Called by tempfile() if files are to be removed on shutdown +# - Called by tempdir() if directories are to be removed on shutdown + +# Arguments: +# _deferred_unlink( $fh, $fname, $isdir ); +# +# - filehandle (so that it can be expclicitly closed if open +# - filename (the thing we want to remove) +# - isdir (flag to indicate that we are being given a directory) +# [and hence no filehandle] + +# Status is not referred to since all the magic is done with an END block + +{ + # Will set up two lexical variables to contain all the files to be + # removed. One array for files, another for directories + # They will only exist in this block + # This means we only have to set up a single END block to remove all files + # @files_to_unlink contains an array ref with the filehandle and filename + my (@files_to_unlink, @dirs_to_unlink); + + # Set up an end block to use these arrays + END { + # Files + foreach my $file (@files_to_unlink) { + # close the filehandle without checking its state + # in order to make real sure that this is closed + # if its already closed then I dont care about the answer + # probably a better way to do this + close($file->[0]); # file handle is [0] + + if (-f $file->[1]) { # file name is [1] + unlink $file->[1] or warn "Error removing ".$file->[1]; + } + } + # Dirs + foreach my $dir (@dirs_to_unlink) { + if (-d $dir) { + rmtree($dir, $DEBUG, 1); + } + } + + } + + # This is the sub called to register a file for deferred unlinking + # This could simply store the input parameters and defer everything + # until the END block. For now we do a bit of checking at this + # point in order to make sure that (1) we have a file/dir to delete + # and (2) we have been called with the correct arguments. + sub _deferred_unlink { + + croak 'Usage: _deferred_unlink($fh, $fname, $isdir)' + unless scalar(@_) == 3; + + my ($fh, $fname, $isdir) = @_; + + warn "Setting up deferred removal of $fname\n" + if $DEBUG; + + # If we have a directory, check that it is a directory + if ($isdir) { + + if (-d $fname) { + + # Directory exists so store it + # first on VMS turn []foo into [.foo] for rmtree + $fname = VMS::Filespec::vmspath($fname) if $^O eq 'VMS'; + push (@dirs_to_unlink, $fname); + + } else { + carp "Request to remove directory $fname could not be completed since it does not exist!\n" if $^W; + } + + } else { + + if (-f $fname) { + + # file exists so store handle and name for later removal + push(@files_to_unlink, [$fh, $fname]); + + } else { + carp "Request to remove file $fname could not be completed since it is not there!\n" if $^W; + } + + } + + } + + +} + +=head1 FUNCTIONS + +This section describes the recommended interface for generating +temporary files and directories. + +=over 4 + +=item B + +This is the basic function to generate temporary files. +The behaviour of the file can be changed using various options: + + ($fh, $filename) = tempfile(); + +Create a temporary file in the directory specified for temporary +files, as specified by the tmpdir() function in L. + + ($fh, $filename) = tempfile($template); + +Create a temporary file in the current directory using the supplied +template. Trailing `X' characters are replaced with random letters to +generate the filename. At least four `X' characters must be present +in the template. + + ($fh, $filename) = tempfile($template, SUFFIX => $suffix) + +Same as previously, except that a suffix is added to the template +after the `X' translation. Useful for ensuring that a temporary +filename has a particular extension when needed by other applications. +But see the WARNING at the end. + + ($fh, $filename) = tempfile($template, DIR => $dir); + +Translates the template as before except that a directory name +is specified. + + ($fh, $filename) = tempfile($template, UNLINK => 1); + +Return the filename and filehandle as before except that the file is +automatically removed when the program exits. Default is for the file +to be removed if a file handle is requested and to be kept if the +filename is requested. In a scalar context (where no filename is +returned) the file is always deleted either on exit or when it is closed. + +If the template is not specified, a template is always +automatically generated. This temporary file is placed in tmpdir() +(L) unless a directory is specified explicitly with the +DIR option. + + $fh = tempfile( $template, DIR => $dir ); + +If called in scalar context, only the filehandle is returned +and the file will automatically be deleted when closed (see +the description of tmpfile() elsewhere in this document). +This is the preferred mode of operation, as if you only +have a filehandle, you can never create a race condition +by fumbling with the filename. On systems that can not unlink +an open file or can not mark a file as temporary when it is opened +(for example, Windows NT uses the C flag)) +the file is marked for deletion when the program ends (equivalent +to setting UNLINK to 1). The C flag is ignored if present. + + (undef, $filename) = tempfile($template, OPEN => 0); + +This will return the filename based on the template but +will not open this file. Cannot be used in conjunction with +UNLINK set to true. Default is to always open the file +to protect from possible race conditions. A warning is issued +if warnings are turned on. Consider using the tmpnam() +and mktemp() functions described elsewhere in this document +if opening the file is not required. + +Options can be combined as required. + +=cut + +sub tempfile { + + # Can not check for argument count since we can have any + # number of args + + # Default options + my %options = ( + "DIR" => undef, # Directory prefix + "SUFFIX" => '', # Template suffix + "UNLINK" => 0, # Do not unlink file on exit + "OPEN" => 1, # Open file + ); + + # Check to see whether we have an odd or even number of arguments + my $template = (scalar(@_) % 2 == 1 ? shift(@_) : undef); + + # Read the options and merge with defaults + %options = (%options, @_) if @_; + + # First decision is whether or not to open the file + if (! $options{"OPEN"}) { + + warn "tempfile(): temporary filename requested but not opened.\nPossibly unsafe, consider using tempfile() with OPEN set to true\n" + if $^W; + + } + + if ($options{"DIR"} and $^O eq 'VMS') { + + # on VMS turn []foo into [.foo] for concatenation + $options{"DIR"} = VMS::Filespec::vmspath($options{"DIR"}); + } + + # Construct the template + + # Have a choice of trying to work around the mkstemp/mktemp/tmpnam etc + # functions or simply constructing a template and using _gettemp() + # explicitly. Go for the latter + + # First generate a template if not defined and prefix the directory + # If no template must prefix the temp directory + if (defined $template) { + if ($options{"DIR"}) { + + $template = File::Spec->catfile($options{"DIR"}, $template); + + } + + } else { + + if ($options{"DIR"}) { + + $template = File::Spec->catfile($options{"DIR"}, TEMPXXX); + + } else { + + $template = File::Spec->catfile(File::Spec->tmpdir, TEMPXXX); + + } + + } + + # Now add a suffix + $template .= $options{"SUFFIX"}; + + # Determine whether we should tell _gettemp to unlink the file + # On unix this is irrelevant and can be worked out after the file is + # opened (simply by unlinking the open filehandle). On Windows or VMS + # we have to indicate temporary-ness when we open the file. In general + # we only want a true temporary file if we are returning just the + # filehandle - if the user wants the filename they probably do not + # want the file to disappear as soon as they close it. + # For this reason, tie unlink_on_close to the return context regardless + # of OS. + my $unlink_on_close = ( wantarray ? 0 : 1); + + # Create the file + my ($fh, $path, $errstr); + croak "Error in tempfile() using $template: $errstr" + unless (($fh, $path) = _gettemp($template, + "open" => $options{'OPEN'}, + "mkdir"=> 0 , + "unlink_on_close" => $unlink_on_close, + "suffixlen" => length($options{'SUFFIX'}), + "ErrStr" => \$errstr, + ) ); + + # Set up an exit handler that can do whatever is right for the + # system. This removes files at exit when requested explicitly or when + # system is asked to unlink_on_close but is unable to do so because + # of OS limitations. + # The latter should be achieved by using a tied filehandle. + # Do not check return status since this is all done with END blocks. + _deferred_unlink($fh, $path, 0) if $options{"UNLINK"}; + + # Return + if (wantarray()) { + + if ($options{'OPEN'}) { + return ($fh, $path); + } else { + return (undef, $path); + } + + } else { + + # Unlink the file. It is up to unlink0 to decide what to do with + # this (whether to unlink now or to defer until later) + unlink0($fh, $path) or croak "Error unlinking file $path using unlink0"; + + # Return just the filehandle. + return $fh; + } + + +} + +=item B + +This is the recommended interface for creation of temporary directories. +The behaviour of the function depends on the arguments: + + $tempdir = tempdir(); + +Create a directory in tmpdir() (see L). + + $tempdir = tempdir( $template ); + +Create a directory from the supplied template. This template is +similar to that described for tempfile(). `X' characters at the end +of the template are replaced with random letters to construct the +directory name. At least four `X' characters must be in the template. + + $tempdir = tempdir ( DIR => $dir ); + +Specifies the directory to use for the temporary directory. +The temporary directory name is derived from an internal template. + + $tempdir = tempdir ( $template, DIR => $dir ); + +Prepend the supplied directory name to the template. The template +should not include parent directory specifications itself. Any parent +directory specifications are removed from the template before +prepending the supplied directory. + + $tempdir = tempdir ( $template, TMPDIR => 1 ); + +Using the supplied template, create the temporary directory in +a standard location for temporary files. Equivalent to doing + + $tempdir = tempdir ( $template, DIR => File::Spec->tmpdir); + +but shorter. Parent directory specifications are stripped from the +template itself. The C option is ignored if C is set +explicitly. Additionally, C is implied if neither a template +nor a directory are supplied. + + $tempdir = tempdir( $template, CLEANUP => 1); + +Create a temporary directory using the supplied template, but +attempt to remove it (and all files inside it) when the program +exits. Note that an attempt will be made to remove all files from +the directory even if they were not created by this module (otherwise +why ask to clean it up?). The directory removal is made with +the rmtree() function from the L module. +Of course, if the template is not specified, the temporary directory +will be created in tmpdir() and will also be removed at program exit. + +=cut + +# ' + +sub tempdir { + + # Can not check for argument count since we can have any + # number of args + + # Default options + my %options = ( + "CLEANUP" => 0, # Remove directory on exit + "DIR" => '', # Root directory + "TMPDIR" => 0, # Use tempdir with template + ); + + # Check to see whether we have an odd or even number of arguments + my $template = (scalar(@_) % 2 == 1 ? shift(@_) : undef ); + + # Read the options and merge with defaults + %options = (%options, @_) if @_; + + # Modify or generate the template + + # Deal with the DIR and TMPDIR options + if (defined $template) { + + # Need to strip directory path if using DIR or TMPDIR + if ($options{'TMPDIR'} || $options{'DIR'}) { + + # Strip parent directory from the filename + # + # There is no filename at the end + $template = VMS::Filespec::vmspath($template) if $^O eq 'VMS'; + my ($volume, $directories, undef) = File::Spec->splitpath( $template, 1); + + # Last directory is then our template + $template = (File::Spec->splitdir($directories))[-1]; + + # Prepend the supplied directory or temp dir + if ($options{"DIR"}) { + + $template = File::Spec->catdir($options{"DIR"}, $template); + + } elsif ($options{TMPDIR}) { + + # Prepend tmpdir + $template = File::Spec->catdir(File::Spec->tmpdir, $template); + + } + + } + + } else { + + if ($options{"DIR"}) { + + $template = File::Spec->catdir($options{"DIR"}, TEMPXXX); + + } else { + + $template = File::Spec->catdir(File::Spec->tmpdir, TEMPXXX); + + } + + } + + # Create the directory + my $tempdir; + my $suffixlen = 0; + if ($^O eq 'VMS') { # dir names can end in delimiters + $template =~ m/([\.\]:>]+)$/; + $suffixlen = length($1); + } + if ( ($^O eq 'MacOS') && (substr($template, -1) eq ':') ) { + # dir name has a trailing ':' + ++$suffixlen; + } + + my $errstr; + croak "Error in tempdir() using $template: $errstr" + unless ((undef, $tempdir) = _gettemp($template, + "open" => 0, + "mkdir"=> 1 , + "suffixlen" => $suffixlen, + "ErrStr" => \$errstr, + ) ); + + # Install exit handler; must be dynamic to get lexical + if ( $options{'CLEANUP'} && -d $tempdir) { + _deferred_unlink(undef, $tempdir, 1); + } + + # Return the dir name + return $tempdir; + +} + +=back + +=head1 MKTEMP FUNCTIONS + +The following functions are Perl implementations of the +mktemp() family of temp file generation system calls. + +=over 4 + +=item B + +Given a template, returns a filehandle to the temporary file and the name +of the file. + + ($fh, $name) = mkstemp( $template ); + +In scalar context, just the filehandle is returned. + +The template may be any filename with some number of X's appended +to it, for example F. The trailing X's are replaced +with unique alphanumeric combinations. + +=cut + + + +sub mkstemp { + + croak "Usage: mkstemp(template)" + if scalar(@_) != 1; + + my $template = shift; + + my ($fh, $path, $errstr); + croak "Error in mkstemp using $template: $errstr" + unless (($fh, $path) = _gettemp($template, + "open" => 1, + "mkdir"=> 0 , + "suffixlen" => 0, + "ErrStr" => \$errstr, + ) ); + + if (wantarray()) { + return ($fh, $path); + } else { + return $fh; + } + +} + + +=item B + +Similar to mkstemp(), except that an extra argument can be supplied +with a suffix to be appended to the template. + + ($fh, $name) = mkstemps( $template, $suffix ); + +For example a template of C and suffix of C<.dat> +would generate a file similar to F. + +Returns just the filehandle alone when called in scalar context. + +=cut + +sub mkstemps { + + croak "Usage: mkstemps(template, suffix)" + if scalar(@_) != 2; + + + my $template = shift; + my $suffix = shift; + + $template .= $suffix; + + my ($fh, $path, $errstr); + croak "Error in mkstemps using $template: $errstr" + unless (($fh, $path) = _gettemp($template, + "open" => 1, + "mkdir"=> 0 , + "suffixlen" => length($suffix), + "ErrStr" => \$errstr, + ) ); + + if (wantarray()) { + return ($fh, $path); + } else { + return $fh; + } + +} + +=item B + +Create a directory from a template. The template must end in +X's that are replaced by the routine. + + $tmpdir_name = mkdtemp($template); + +Returns the name of the temporary directory created. +Returns undef on failure. + +Directory must be removed by the caller. + +=cut + +#' # for emacs + +sub mkdtemp { + + croak "Usage: mkdtemp(template)" + if scalar(@_) != 1; + + my $template = shift; + my $suffixlen = 0; + if ($^O eq 'VMS') { # dir names can end in delimiters + $template =~ m/([\.\]:>]+)$/; + $suffixlen = length($1); + } + if ( ($^O eq 'MacOS') && (substr($template, -1) eq ':') ) { + # dir name has a trailing ':' + ++$suffixlen; + } + my ($junk, $tmpdir, $errstr); + croak "Error creating temp directory from template $template\: $errstr" + unless (($junk, $tmpdir) = _gettemp($template, + "open" => 0, + "mkdir"=> 1 , + "suffixlen" => $suffixlen, + "ErrStr" => \$errstr, + ) ); + + return $tmpdir; + +} + +=item B + +Returns a valid temporary filename but does not guarantee +that the file will not be opened by someone else. + + $unopened_file = mktemp($template); + +Template is the same as that required by mkstemp(). + +=cut + +sub mktemp { + + croak "Usage: mktemp(template)" + if scalar(@_) != 1; + + my $template = shift; + + my ($tmpname, $junk, $errstr); + croak "Error getting name to temp file from template $template: $errstr" + unless (($junk, $tmpname) = _gettemp($template, + "open" => 0, + "mkdir"=> 0 , + "suffixlen" => 0, + "ErrStr" => \$errstr, + ) ); + + return $tmpname; +} + +=back + +=head1 POSIX FUNCTIONS + +This section describes the re-implementation of the tmpnam() +and tmpfile() functions described in L +using the mkstemp() from this module. + +Unlike the L implementations, the directory used +for the temporary file is not specified in a system include +file (C) but simply depends on the choice of tmpdir() +returned by L. On some implementations this +location can be set using the C environment variable, which +may not be secure. +If this is a problem, simply use mkstemp() and specify a template. + +=over 4 + +=item B + +When called in scalar context, returns the full name (including path) +of a temporary file (uses mktemp()). The only check is that the file does +not already exist, but there is no guarantee that that condition will +continue to apply. + + $file = tmpnam(); + +When called in list context, a filehandle to the open file and +a filename are returned. This is achieved by calling mkstemp() +after constructing a suitable template. + + ($fh, $file) = tmpnam(); + +If possible, this form should be used to prevent possible +race conditions. + +See L for information on the choice of temporary +directory for a particular operating system. + +=cut + +sub tmpnam { + + # Retrieve the temporary directory name + my $tmpdir = File::Spec->tmpdir; + + croak "Error temporary directory is not writable" + if $tmpdir eq ''; + + # Use a ten character template and append to tmpdir + my $template = File::Spec->catfile($tmpdir, TEMPXXX); + + if (wantarray() ) { + return mkstemp($template); + } else { + return mktemp($template); + } + +} + +=item B + +Returns the filehandle of a temporary file. + + $fh = tmpfile(); + +The file is removed when the filehandle is closed or when the program +exits. No access to the filename is provided. + +If the temporary file can not be created undef is returned. +Currently this command will probably not work when the temporary +directory is on an NFS file system. + +=cut + +sub tmpfile { + + # Simply call tmpnam() in a list context + my ($fh, $file) = tmpnam(); + + # Make sure file is removed when filehandle is closed + # This will fail on NFS + unlink0($fh, $file) + or return undef; + + return $fh; + +} + +=back + +=head1 ADDITIONAL FUNCTIONS + +These functions are provided for backwards compatibility +with common tempfile generation C library functions. + +They are not exported and must be addressed using the full package +name. + +=over 4 + +=item B + +Return the name of a temporary file in the specified directory +using a prefix. The file is guaranteed not to exist at the time +the function was called, but such guarantees are good for one +clock tick only. Always use the proper form of C +with C if you must open such a filename. + + $filename = File::Temp::tempnam( $dir, $prefix ); + +Equivalent to running mktemp() with $dir/$prefixXXXXXXXX +(using unix file convention as an example) + +Because this function uses mktemp(), it can suffer from race conditions. + +=cut + +sub tempnam { + + croak 'Usage tempnam($dir, $prefix)' unless scalar(@_) == 2; + + my ($dir, $prefix) = @_; + + # Add a string to the prefix + $prefix .= 'XXXXXXXX'; + + # Concatenate the directory to the file + my $template = File::Spec->catfile($dir, $prefix); + + return mktemp($template); + +} + +=back + +=head1 UTILITY FUNCTIONS + +Useful functions for dealing with the filehandle and filename. + +=over 4 + +=item B + +Given an open filehandle and the associated filename, make a safe +unlink. This is achieved by first checking that the filename and +filehandle initially point to the same file and that the number of +links to the file is 1 (all fields returned by stat() are compared). +Then the filename is unlinked and the filehandle checked once again to +verify that the number of links on that file is now 0. This is the +closest you can come to making sure that the filename unlinked was the +same as the file whose descriptor you hold. + + unlink0($fh, $path) or die "Error unlinking file $path safely"; + +Returns false on error. The filehandle is not closed since on some +occasions this is not required. + +On some platforms, for example Windows NT, it is not possible to +unlink an open file (the file must be closed first). On those +platforms, the actual unlinking is deferred until the program ends and +good status is returned. A check is still performed to make sure that +the filehandle and filename are pointing to the same thing (but not at +the time the end block is executed since the deferred removal may not +have access to the filehandle). + +Additionally, on Windows NT not all the fields returned by stat() can +be compared. For example, the C and C fields seem to be +different. Also, it seems that the size of the file returned by stat() +does not always agree, with C being more accurate than +C, presumably because of caching issues even when +using autoflush (this is usually overcome by waiting a while after +writing to the tempfile before attempting to C it). + +Finally, on NFS file systems the link count of the file handle does +not always go to zero immediately after unlinking. Currently, this +command is expected to fail on NFS disks. + +=cut + +sub unlink0 { + + croak 'Usage: unlink0(filehandle, filename)' + unless scalar(@_) == 2; + + # Read args + my ($fh, $path) = @_; + + warn "Unlinking $path using unlink0\n" + if $DEBUG; + + # Stat the filehandle + my @fh = stat $fh; + + if ($fh[3] > 1 && $^W) { + carp "unlink0: fstat found too many links; SB=@fh" if $^W; + } + + # Stat the path + my @path = stat $path; + + unless (@path) { + carp "unlink0: $path is gone already" if $^W; + return; + } + + # this is no longer a file, but may be a directory, or worse + unless (-f _) { + confess "panic: $path is no longer a file: SB=@fh"; + } + + # Do comparison of each member of the array + # On WinNT dev and rdev seem to be different + # depending on whether it is a file or a handle. + # Cannot simply compare all members of the stat return + # Select the ones we can use + my @okstat = (0..$#fh); # Use all by default + if ($^O eq 'MSWin32') { + @okstat = (1,2,3,4,5,7,8,9,10); + } elsif ($^O eq 'os2') { + @okstat = (0, 2..$#fh); + } elsif ($^O eq 'VMS') { # device and file ID are sufficient + @okstat = (0, 1); + } elsif ($^O eq 'dos') { + @okstat = (0,2..7,11..$#fh); + } + + # Now compare each entry explicitly by number + for (@okstat) { + print "Comparing: $_ : $fh[$_] and $path[$_]\n" if $DEBUG; + # Use eq rather than == since rdev, blksize, and blocks (6, 11, + # and 12) will be '' on platforms that do not support them. This + # is fine since we are only comparing integers. + unless ($fh[$_] eq $path[$_]) { + warn "Did not match $_ element of stat\n" if $DEBUG; + return 0; + } + } + + # attempt remove the file (does not work on some platforms) + if (_can_unlink_opened_file()) { + # XXX: do *not* call this on a directory; possible race + # resulting in recursive removal + croak "unlink0: $path has become a directory!" if -d $path; + unlink($path) or return 0; + + # Stat the filehandle + @fh = stat $fh; + + print "Link count = $fh[3] \n" if $DEBUG; + + # Make sure that the link count is zero + # - Cygwin provides deferred unlinking, however, + # on Win9x the link count remains 1 + # On NFS the link count may still be 1 but we cant know that + # we are on NFS + return ( $fh[3] == 0 or $^O eq 'cygwin' ? 1 : 0); + + } else { + _deferred_unlink($fh, $path, 0); + return 1; + } + +} + +=back + +=head1 PACKAGE VARIABLES + +These functions control the global state of the package. + +=over 4 + +=item B + +Controls the lengths to which the module will go to check the safety of the +temporary file or directory before proceeding. +Options are: + +=over 8 + +=item STANDARD + +Do the basic security measures to ensure the directory exists and +is writable, that the umask() is fixed before opening of the file, +that temporary files are opened only if they do not already exist, and +that possible race conditions are avoided. Finally the L +function is used to remove files safely. + +=item MEDIUM + +In addition to the STANDARD security, the output directory is checked +to make sure that it is owned either by root or the user running the +program. If the directory is writable by group or by other, it is then +checked to make sure that the sticky bit is set. + +Will not work on platforms that do not support the C<-k> test +for sticky bit. + +=item HIGH + +In addition to the MEDIUM security checks, also check for the +possibility of ``chown() giveaway'' using the L +sysconf() function. If this is a possibility, each directory in the +path is checked in turn for safeness, recursively walking back to the +root directory. + +For platforms that do not support the L +C<_PC_CHOWN_RESTRICTED> symbol (for example, Windows NT) it is +assumed that ``chown() giveaway'' is possible and the recursive test +is performed. + +=back + +The level can be changed as follows: + + File::Temp->safe_level( File::Temp::HIGH ); + +The level constants are not exported by the module. + +Currently, you must be running at least perl v5.6.0 in order to +run with MEDIUM or HIGH security. This is simply because the +safety tests use functions from L that are not +available in older versions of perl. The problem is that the version +number for Fcntl is the same in perl 5.6.0 and in 5.005_03 even though +they are different versions. + +On systems that do not support the HIGH or MEDIUM safety levels +(for example Win NT or OS/2) any attempt to change the level will +be ignored. The decision to ignore rather than raise an exception +allows portable programs to be written with high security in mind +for the systems that can support this without those programs failing +on systems where the extra tests are irrelevant. + +If you really need to see whether the change has been accepted +simply examine the return value of C. + + $newlevel = File::Temp->safe_level( File::Temp::HIGH ); + die "Could not change to high security" + if $newlevel != File::Temp::HIGH; + +=cut + +{ + # protect from using the variable itself + my $LEVEL = STANDARD; + sub safe_level { + my $self = shift; + if (@_) { + my $level = shift; + if (($level != STANDARD) && ($level != MEDIUM) && ($level != HIGH)) { + carp "safe_level: Specified level ($level) not STANDARD, MEDIUM or HIGH - ignoring\n" if $^W; + } else { + # Dont allow this on perl 5.005 or earlier + if ($] < 5.006 && $level != STANDARD) { + # Cant do MEDIUM or HIGH checks + croak "Currently requires perl 5.006 or newer to do the safe checks"; + } + # Check that we are allowed to change level + # Silently ignore if we can not. + $LEVEL = $level if _can_do_level($level); + } + } + return $LEVEL; + } +} + +=item TopSystemUID + +This is the highest UID on the current system that refers to a root +UID. This is used to make sure that the temporary directory is +owned by a system UID (C, C, C etc) rather than +simply by root. + +This is required since on many unix systems C is not owned +by root. + +Default is to assume that any UID less than or equal to 10 is a root +UID. + + File::Temp->top_system_uid(10); + my $topid = File::Temp->top_system_uid; + +This value can be adjusted to reduce security checking if required. +The value is only relevant when C is set to MEDIUM or higher. + +=back + +=cut + +{ + my $TopSystemUID = 10; + sub top_system_uid { + my $self = shift; + if (@_) { + my $newuid = shift; + croak "top_system_uid: UIDs should be numeric" + unless $newuid =~ /^\d+$/s; + $TopSystemUID = $newuid; + } + return $TopSystemUID; + } +} + +=head1 WARNING + +For maximum security, endeavour always to avoid ever looking at, +touching, or even imputing the existence of the filename. You do not +know that that filename is connected to the same file as the handle +you have, and attempts to check this can only trigger more race +conditions. It's far more secure to use the filehandle alone and +dispense with the filename altogether. + +If you need to pass the handle to something that expects a filename +then, on a unix system, use C<"/dev/fd/" . fileno($fh)> for arbitrary +programs, or more generally C<< "+<=&" . fileno($fh) >> for Perl +programs. You will have to clear the close-on-exec bit on that file +descriptor before passing it to another process. + + use Fcntl qw/F_SETFD F_GETFD/; + fcntl($tmpfh, F_SETFD, 0) + or die "Can't clear close-on-exec flag on temp fh: $!\n"; + +=head2 Temporary files and NFS + +Some problems are associated with using temporary files that reside +on NFS file systems and it is recommended that a local filesystem +is used whenever possible. Some of the security tests will most probably +fail when the temp file is not local. Additionally, be aware that +the performance of I/O operations over NFS will not be as good as for +a local disk. + +=head1 HISTORY + +Originally began life in May 1999 as an XS interface to the system +mkstemp() function. In March 2000, the OpenBSD mkstemp() code was +translated to Perl for total control of the code's +security checking, to ensure the presence of the function regardless of +operating system and to help with portability. + +=head1 SEE ALSO + +L, L, L, L + +See L and L for different implementations of +temporary file handling. + +=head1 AUTHOR + +Tim Jenness Et.jenness@jach.hawaii.eduE + +Copyright (C) 1999-2001 Tim Jenness and the UK Particle Physics and +Astronomy Research Council. All Rights Reserved. This program is free +software; you can redistribute it and/or modify it under the same +terms as Perl itself. + +Original Perl implementation loosely based on the OpenBSD C code for +mkstemp(). Thanks to Tom Christiansen for suggesting that this module +should be written and providing ideas for code improvements and +security enhancements. + +=cut + + +1; diff --git a/lib/Milkbone/AllHooks.pm b/lib/Milkbone/AllHooks.pm new file mode 100644 index 0000000..90a9442 --- /dev/null +++ b/lib/Milkbone/AllHooks.pm @@ -0,0 +1,15 @@ +# allows you to call hooks like subs - +# use base 'Milkbone::AllHooks'; +# hookname(@args); + +package Milkbone::AllHooks; + +use Milkbone; + +sub AUTOLOAD +{ + my $hook = $AUTOLOAD; + hook($hook, @_); +} + +1; diff --git a/lib/Milkbone/HookEntry.pm b/lib/Milkbone/HookEntry.pm new file mode 100644 index 0000000..cca3298 --- /dev/null +++ b/lib/Milkbone/HookEntry.pm @@ -0,0 +1,44 @@ +# ----------------------------------------------------------------------------- +# Author(s) : Bill Atkins +# Title : MOS hook info +# Date : 1.22.02 +# Desc : tracks information about registered hooks +# Notes : for more information see the plugin documentation +# License : under the same terms as mos.pl +# ----------------------------------------------------------------------------- + +package Milkbone::HookEntry; + +use strict; +use warnings; + +require Exporter; + +our @ISA = qw( Exporter ); +our @EXPORT = qw( ); +our @EXPORT_OK = qw( ); + +sub new +{ + my $self = {}; + my $class = shift; + bless $self, $class; + + my ($hook_name, $callback, $args, $package) = @_; + + $self->{hook_name} = $hook_name; + $self->{callback} = $callback; + $self->{args} = $args; + $self->{package} = $package; + + return $self; +} + +sub call +{ + my ($self, %args) = @_; + + $self->{callback}->(%args, %{$self->{args}}); +} + +1; \ No newline at end of file diff --git a/lib/Tk/Text.pm b/lib/Tk/Text.pm new file mode 100644 index 0000000..cdba615 --- /dev/null +++ b/lib/Tk/Text.pm @@ -0,0 +1,1599 @@ +# text.tcl -- +# +# This file defines the default bindings for Tk text widgets. +# +# @(#) text.tcl 1.18 94/12/17 16:05:26 +# +# Copyright (c) 1992-1994 The Regents of the University of California. +# Copyright (c) 1994 Sun Microsystems, Inc. +# perl/Tk version: +# Copyright (c) 1995-1999 Nick Ing-Simmons +# Copyright (c) 1999 Greg London +# +# See the file "license.terms" for information on usage and redistribution +# of this file, and for a DISCLAIMER OF ALL WARRANTIES. +package Tk::Text; +use Carp; +use strict; + +use Text::Tabs; + +use vars qw($VERSION); +$VERSION = '3.044'; # $Id$ + +use Tk qw(Ev $XS_VERSION); +use base qw(Tk::Clipboard Tk::Widget); + +Construct Tk::Widget 'Text'; + +bootstrap Tk::Text; + +sub Tk_cmd { \&Tk::text } + +sub Tk::Widget::ScrlText { shift->Scrolled('Text' => @_) } + +Tk::Methods('bbox','compare','debug','delete','dlineinfo','dump', + 'get','image','index','insert','mark','scan','search', + 'see','tag','window','xview','yview'); + +use Tk::Submethods ( 'mark' => [qw(gravity names next previous set unset)], + 'scan' => [qw(mark dragto)], + 'tag' => [qw(add bind cget configure delete lower + names nextrange prevrange raise ranges remove)], + 'window' => [qw(cget configure create names)], + 'image' => [qw(cget configure create names)], + 'xview' => [qw(moveto scroll)], + 'yview' => [qw(moveto scroll)], + ); + +sub Tag; +sub Tags; + +sub bindRdOnly +{ + + my ($class,$mw) = @_; + + # Standard Motif bindings: + $mw->bind($class,'','NoOp'); + $mw->bind($class,'','NoOp'); + $mw->bind($class,'','NoOp'); + $mw->bind($class,'','unselectAll'); + + $mw->bind($class,'<1>',['Button1',Ev('x'),Ev('y')]); + $mw->bind($class,'','B1_Motion' ) ; + $mw->bind($class,'','B1_Leave' ) ; + $mw->bind($class,'','CancelRepeat'); + $mw->bind($class,'','CancelRepeat'); + $mw->bind($class,'',['markSet','insert',Ev('@')]); + + $mw->bind($class,'','selectWord' ) ; + $mw->bind($class,'','selectLine' ) ; + $mw->bind($class,'','adjustSelect' ) ; + $mw->bind($class,'',['SelectTo',Ev('@'),'word']); + $mw->bind($class,'',['SelectTo',Ev('@'),'line']); + + $mw->bind($class,'',['SetCursor',Ev('index','insert-1c')]); + $mw->bind($class,'',['KeySelect',Ev('index','insert-1c')]); + $mw->bind($class,'',['SetCursor',Ev('index','insert-1c wordstart')]); + $mw->bind($class,'',['KeySelect',Ev('index','insert-1c wordstart')]); + + $mw->bind($class,'',['SetCursor',Ev('index','insert+1c')]); + $mw->bind($class,'',['KeySelect',Ev('index','insert+1c')]); + $mw->bind($class,'',['SetCursor',Ev('index','insert+1c wordend')]); + $mw->bind($class,'',['KeySelect',Ev('index','insert wordend')]); + + $mw->bind($class,'',['SetCursor',Ev('UpDownLine',-1)]); + $mw->bind($class,'',['KeySelect',Ev('UpDownLine',-1)]); + $mw->bind($class,'',['SetCursor',Ev('PrevPara','insert')]); + $mw->bind($class,'',['KeySelect',Ev('PrevPara','insert')]); + + $mw->bind($class,'',['SetCursor',Ev('UpDownLine',1)]); + $mw->bind($class,'',['KeySelect',Ev('UpDownLine',1)]); + $mw->bind($class,'',['SetCursor',Ev('NextPara','insert')]); + $mw->bind($class,'',['KeySelect',Ev('NextPara','insert')]); + + $mw->bind($class,'',['SetCursor','insert linestart']); + $mw->bind($class,'',['KeySelect','insert linestart']); + $mw->bind($class,'',['SetCursor','1.0']); + $mw->bind($class,'',['KeySelect','1.0']); + + $mw->bind($class,'',['SetCursor','insert lineend']); + $mw->bind($class,'',['KeySelect','insert lineend']); + $mw->bind($class,'',['SetCursor','end-1char']); + $mw->bind($class,'',['KeySelect','end-1char']); + + $mw->bind($class,'',['SetCursor',Ev('ScrollPages',-1)]); + $mw->bind($class,'',['KeySelect',Ev('ScrollPages',-1)]); + $mw->bind($class,'',['xview','scroll',-1,'page']); + + $mw->bind($class,'',['SetCursor',Ev('ScrollPages',1)]); + $mw->bind($class,'',['KeySelect',Ev('ScrollPages',1)]); + $mw->bind($class,'',['xview','scroll',1,'page']); + + $mw->bind($class,'', 'NoOp'); # Needed only to keep binding from triggering; does not have to actually do anything. + $mw->bind($class,'','focusNext'); + $mw->bind($class,'','focusPrev'); + + $mw->bind($class,'',['markSet','anchor','insert']); + $mw->bind($class,' call on all the object's +sockets and reads incoming commands from the OSCAR server on any connections which +have them. The C-based event loop, especially one where you +have many C objects. Simply call the L<"process_connections"> method +with references to the lists of readers, writers, and errors given to you by +C lists +so that you can use the lists for your own purposes. Here is an example that +demonstrates how to use this method with multiple C objects: + + my($rin, $win) = (0, 0); + foreach my $oscar(@oscars) { + my($thisrin, $thiswin) = $oscar->selector_filenos; + $rin |= $thisrin; + $win |= $thiswin; + } + # Add in any other file descriptors you care about using vec(). + my $ein = $rin | $win; + select($rin, $win, $ein, 0.01); + foreach my $oscar(@oscars) { + $oscar->process_connections(\$rin, \$win, \$ein); + } + + # Now $rin, $win, and $ein only have the file descriptors not + # associated with any of the OSCAR objects in them - we can + # process our events. + +The third way of doing connection processing uses the L<"connection_changed"> +callback in conjunction with C's L<"process_one"> method. +This method, in conjunction with C, probably offers the highest performance +in situations where you have a long-lived application which creates and destroys many +C sessions; that is, an application whose list of file descriptors to +monitor will likely be sparse. However, this method is the most complicated. +What you need to do is call C inside of the L<"connection_changed"> +callback. That part's simple. The tricky bit is figuring out which +C's to call and how to call them. My recommendation +for doing this is to use a hashmap whose keys are the file descriptors of everything +you're monitoring in the C - the FDs can be retrieved by doing +Cget_filehandle)> inside of the L<"connection_changed"> - +and then calling C<@handles = $poll-Ehandles(POLLIN | POLLOUT | POLLERR | POLLHUP)> +and walking through the handles. + + +=head1 FUNCTIONALITY + +C pretends to be WinAIM 4.7.2480. It supports remote buddylists +including permit and deny settings. It also supports chat. At the present +time, setting and retrieving of directory information is not supported; nor +are email privacy settings, buddy icons, voice chat, stock ticker, and +many other of the official AOL Instant Messenger client's features. + +=head1 TERMINOLOGY/METHODOLOGY + +When you sign on with the OSCAR service, you are establishing an OSCAR session. +C connects to the login server and requests a random challenge +string. It then sends the MD5 sum of the challenge string, +C, and your password to the server. If the login +is successful, the login server gives you an IP address and an authorization +cookie to use to connect with the BOS (Basic OSCAR Services) server. + +C proceeds to disconnect from the login server and connect to the +BOS server. The two go through a handshaking process which includes the +server sending us our buddylist. + +C supports privacy controls. Our visibility setting, along +with the contents of the permit and deny lists, determines who can +contact us. Visibility can be set to permit or deny everyone, permit only +those on the permit list, deny only those on the deny list, or permit +everyone on our buddylist. + +=head1 METHODS + +=over 4 + +=cut + +use strict; +use vars qw($VERSION @ISA @EXPORT_OK %EXPORT_TAGS); +use Carp; +use Scalar::Util; +use Net::OSCAR::Common qw(:all); +use Net::OSCAR::Connection; +use Net::OSCAR::Callbacks; +use Net::OSCAR::TLV; +use Net::OSCAR::Buddylist; +use Net::OSCAR::Screenname; +use Net::OSCAR::Chat; +use Net::OSCAR::_BLInternal; +use Net::OSCAR::OldPerl; + +require Exporter; +@ISA = qw(Exporter); +@EXPORT_OK = @Net::OSCAR::Common::EXPORT_OK; +%EXPORT_TAGS = %Net::OSCAR::Common::EXPORT_TAGS; + +=pod + +=item new + +Creates a new C object. + +=cut + +sub new($) { + my $class = ref($_[0]) || $_[0] || "Net::OSCAR"; + shift; + my $self = { }; + bless $self, $class; + $self->{LOGLEVEL} = 0; + $self->{SNDEBUG} = 0; + $self->{description} = "OSCAR session"; + + $self->{timeout} = 0.01; + + return $self; +} + + +=pod + +=item timeout ([NEW TIMEOUT]) + +Gets or sets the timeout value used by the L method. +The default timeout is 0.01 seconds. + +=cut + +sub timeout($;$) { + my($self, $timeout) = @_; + return $self->{timeout} unless $timeout; + $self->{timeout} = $timeout; +} + +=pod + +=item signon (HASH) + +=item signon (SCREENNAME, PASSWORD[, HOST, PORT]) + +Sign on to the OSCAR service. Using a hash to +pass the parameters to this function is preferred - +the old method is deprecated. You can specify an +alternate host/port to connect to. The default is +login.oscar.aol.com port 5190. + +If you use a hash to pass parameters to this function, +here are the valid keys: + +=over 4 + +=item screenname + +=item password + +Screenname and password are mandatory. The other keys are optional. +In the special case of password being present but undefined, the +auth_challenge callback will be used - see L<"auth_challenge"> for details. + +=item host + +=item port + +=back + +There are some other data that can be passed to this method. +These data are used to sign on to an OSCAR-using service other than the default of +AOL Instant Messenger, such as ICQ. You should not attempt to specify +these data directly - instead, use one of the following constants: + +=over 4 + +=item OSCAR_SVC_AIM + +=item OSCAR_SVC_ICQ + +=back + +Example of signing on to ICQ: + + $oscar->signon(screenname => "123456", password => "password", OSCAR_SVC_ICQ); + +=cut + +sub signon($@) { + my($self, $password, $host, %args); + $self = shift; + + # Determine whether caller is using hash-method or old method of passing parms. + # Note that this breaks if caller passes in both a host and a port using the old way. + # But hey, that's why it's deprecated! + if(@_ < 3) { + $args{screenname} = shift @_ or return $self->crapout($self->{bos}, "You must specify a username to sign on with!"); + $args{password} = shift @_ or return $self->crapout($self->{bos}, "You must specify a password to sign on with!");; + $args{host} = shift @_ if @_; + $args{port} = shift @_ if @_; + } else { + %args = @_; + return $self->crapout($self->{bos}, "You must specify a username and password to sign on with!") unless $args{screenname} and exists($args{password}); + } + + my %defaults = OSCAR_SVC_AIM; + foreach my $key(keys %defaults) { + $args{$key} ||= $defaults{$key}; + } + return $self->crapout($self->{bos}, "MD5 authentication not available for this service (you must define a password.)") if !defined($args{password}) and $args{hashlogin}; + $self->{screenname} = new Net::OSCAR::Screenname $args{screenname}; + + # We set BOS to the login connection so that our error handlers pick up errors on this connection as fatal. + $args{host} ||= "login.oscar.aol.com"; + $args{port} ||= 5190; + + + ($self->{screenname}, $password, $host, $self->{port}) = + delete @args{qw(screenname password host port)}; + + $self->{svcdata} = \%args; + $self->{bos} = $self->addconn($password, CONNTYPE_LOGIN, "login", $host); +} + +=pod + +=item auth_response (MD5_DIGEST) + +Provide a response to an authentication challenge - see the L<"auth_challenge"> +callback for details. + +=cut + +sub auth_response($$) { + my($self, $digest) = @_; + $self->log_print(OSCAR_DBG_SIGNON, "Got authentication response - proceeding with signon"); + $self->{auth_response} = $digest; + $self->{bos}->snac_put(family => 0x17, subtype => 0x2, data => tlv(signon_tlv($self))); +} + +=pod + +=item signoff + +Sign off from the OSCAR service. + +=cut + +sub signoff($) { + my $self = shift; + foreach my $connection(@{$self->{connections}}) { + $self->delconn($connection); + } + my $screenname = $self->{screenname}; + %$self = (); + $self->{screename} = $screenname; # Useful for post-mortem processing in multiconnection apps +} + +=pod + +=item loglevel ([LOGLEVEL[, SCREENNAME DEBUG]]) + +Gets or sets the loglevel. If this is non-zero, varing amounts of information will be printed +to standard error (unless you have a L<"log"> callback defined). Higher loglevels will give you more information. +If the optional screenname debug parameter is non-zero, +debug messages will be prepended with the screenname of the OSCAR session which is generating +the message (but only if you don't have a L<"log"> callback defined). This is useful when you have multiple C objects. + +See the L<"log"> callback for more information. + +=cut + +sub loglevel($;$$) { + my $self = shift; + return $self->{LOGLEVEL} unless @_; + $self->{LOGLEVEL} = shift; + $self->{SNDEBUG} = shift if @_; +} + +sub addconn($$$$$) { + my $self = shift; + my $conntype = $_[1]; + + my $connection = ($conntype == CONNTYPE_CHAT) ? Net::OSCAR::Chat->new($self, @_) : Net::OSCAR::Connection->new($self, @_); + if($_[1] == CONNTYPE_BOS) { + $self->{bos} = $connection; + } elsif($_[1] == CONNTYPE_ADMIN) { + $self->{admin} = 1; # We're not quite ready yet - add to queue but don't send svcreq + } elsif($_[1] == CONNTYPE_CHATNAV) { + $self->{chatnav} = 1; + } + push @{$self->{connections}}, $connection; + $self->callback_connection_changed($connection, "write"); + return $connection; +} + +sub delconn($$) { + my($self, $connection) = @_; + + return unless $self->{connections}; + $self->callback_connection_changed($connection, "deleted"); + for(my $i = scalar @{$self->{connections}} - 1; $i >= 0; $i--) { + next unless $self->{connections}->[$i] == $connection; + $connection->log_print(OSCAR_DBG_NOTICE, "Closing."); + splice @{$self->{connections}}, $i, 1; + if(!$connection->{sockerr}) { + eval { + $connection->flap_put("", FLAP_CHAN_CLOSE) if $connection->{socket}; + close $connection->{socket} if $connection->{socket}; + }; + } else { + if($connection->{conntype} == CONNTYPE_BOS or ($connection->{conntype} == CONNTYPE_LOGIN and !$connection->{closing})) { + delete $connection->{socket}; + print $connection->{sockerr}; + return $self->crapout($connection, "Lost connection to BOS"); + } elsif($connection->{conntype} == CONNTYPE_CHATNAV) { + delete $self->{chatnav}; + } elsif($connection->{conntype} == CONNTYPE_ADMIN) { + delete $self->{admin}; + $self->callback_admin_error("all", ADMIN_ERROR_CONNREF, undef) if scalar(keys(%{$self->{adminreq}})); + } elsif($connection->{conntype} == CONNTYPE_CHAT) { + $self->callback_chat_closed($connection, "Lost connection to chat"); + } + } + delete $connection->{socket}; + return 1; + } + return 0; +} + +=pod + +=item findconn (FILENO) + +Finds the connection that is using the specified file number, or undef +if the connection could not be found. Returns a C +object. + +=cut + +sub findconn($$) { + my($self, $target) = @_; + my($conn) = grep { fileno($_->{socket}) == $target } @{$self->{connections}}; + return $conn; +} + +sub DESTROY { + my $self = shift; + + foreach my $connection(@{$self->{connections}}) { + next unless $connection->{socket} and not $connection->{sockerr}; + $connection->flap_put("", FLAP_CHAN_CLOSE); + close $connection->{socket} if $connection->{socket}; + } +} + +=pod + +=item process_connections (READERSREF, WRITERSREF, ERRORSREF) + +Use this method when you want to implement your own C-based event loop that you are also +using for other purposes. + +See the L method for a way to get the necessary +bit vectors to use in your C indicates that +the connection is ready for reading, "write" if we should call +L<"process_one"> when the connection is ready for writing, "readwrite" if L<"process_one"> +should be called in both cases, or "deleted" if the connection has been deleted. + +C is a C object. + +Users of this callback may also be interested in the L<"get_filehandle"> +method of C. + +=cut + +sub do_callback($@) { + my $callback = shift; + return unless $_[0]->{callbacks}->{$callback}; + &{$_[0]->{callbacks}->{$callback}}(@_); +} +sub set_callback { $_[1]->{callbacks}->{$_[0]} = $_[2]; } + +sub callback_error(@) { do_callback("error", @_); } +sub callback_buddy_in(@) { do_callback("buddy_in", @_); } +sub callback_buddy_out(@) { do_callback("buddy_out", @_); } +sub callback_typing_status(@) { do_callback("typing_status", @_); } +sub callback_im_in(@) { do_callback("im_in", @_); } +sub callback_typing_changed(@) { do_callback("typing_changed", @_); } +sub callback_chat_joined(@) { do_callback("chat_joined", @_); } +sub callback_chat_buddy_in(@) { do_callback("chat_buddy_in", @_); } +sub callback_chat_buddy_out(@) { do_callback("chat_buddy_out", @_); } +sub callback_chat_im_in(@) { do_callback("chat_im_in", @_); } +sub callback_chat_invite(@) { do_callback("chat_invite", @_); } +sub callback_buddy_info(@) { do_callback("buddy_info", @_); } +sub callback_evil(@) { do_callback("evil", @_); } +sub callback_chat_closed(@) { do_callback("chat_closed", @_); } +sub callback_buddylist_error(@) { do_callback("buddylist_error", @_); } +sub callback_buddylist_ok(@) { do_callback("buddylist_ok", @_); } +sub callback_admin_error(@) { do_callback("admin_error", @_); } +sub callback_admin_ok(@) { do_callback("admin_ok", @_); } +sub callback_rate_alert(@) { do_callback("rate_alert", @_); } +sub callback_signon_done(@) { do_callback("signon_done", @_); } +sub callback_log(@) { do_callback("log", @_); } +sub callback_im_ok(@) { do_callback("im_ok", @_); } +sub callback_connection_changed(@) { do_callback("connection_changed", @_); } +sub callback_auth_challenge(@) { do_callback("auth_challenge", @_); } + +sub set_callback_error($\&) { set_callback("error", @_); } +sub set_callback_buddy_in($\&) { set_callback("buddy_in", @_); } +sub set_callback_buddy_out($\&) { set_callback("buddy_out", @_); } +sub set_callback_typing_status($\&) { set_callback("typing_status", @_); } +sub set_callback_im_in($\&) { set_callback("im_in", @_); } +sub set_callback_chat_joined($\&) { set_callback("chat_joined", @_); } +sub set_callback_chat_buddy_in($\&) { set_callback("chat_buddy_in", @_); } +sub set_callback_chat_buddy_out($\&) { set_callback("chat_buddy_out", @_); } +sub set_callback_chat_im_in($\&) { set_callback("chat_im_in", @_); } +sub set_callback_chat_invite($\&) { set_callback("chat_invite", @_); } +sub set_callback_buddy_info($\&) { set_callback("buddy_info", @_); } +sub set_callback_evil($\&) { set_callback("evil", @_); } +sub set_callback_chat_closed($\&) { set_callback("chat_closed", @_); } +sub set_callback_buddylist_error($\&) { set_callback("buddylist_error", @_); } +sub set_callback_buddylist_ok($\&) { set_callback("buddylist_ok", @_); } +sub set_callback_admin_error($\&) { set_callback("admin_error", @_); } +sub set_callback_admin_ok($\&) { set_callback("admin_ok", @_); } +sub set_callback_rate_alert($\&) { set_callback("rate_alert", @_); } +sub set_callback_signon_done($\&) { set_callback("signon_done", @_); } +sub set_callback_log($\&) { set_callback("log", @_); } +sub set_callback_im_ok($\&) { set_callback("im_ok", @_); } +sub set_callback_connection_changed($\&) { set_callback("connection_changed", @_); } +sub set_callback_auth_challenge($\&) { set_callback("auth_challenge", @_); } + +=pod + +=back + +=head1 CHATS + +Aside from the methods listed here, there are a couple of methods of the +C object that are important for implementing chat +functionality. C is a descendent of C. + +=over 4 + +=item invite (WHO, MESSAGE) + +Invite somebody into the chatroom. + +=item chat_send (MESSAGE[, NOREFLECT[, AWAY]]) + +Sends a message to the chatroom. If the NOREFLECT parameter is +present, you will not receive the message as an incoming message +from the chatroom. If AWAY is present, the message was generated +as an automatic reply, perhaps because you have an away message set. + +=item part + +Leave the chatroom. + +=item url + +Returns the URL for the chatroom. Use this to associate a chat invitation +with the chat_joined that C sends when you've join the chatroom. + +=item name + +Returns the name of the chatroom. + +=item exchange + +Returns the exchange of the chatroom. +This is normally 4 but can be 5 for certain chatrooms. + +=back + +=head1 ICQ + +ICQ support is very preliminary. A patch enabling us to sign on to +ICQ was provided by Sam Wong. No further work beyond the ability +to sign on has been done on ICQ at this time. See the C method +for details on signing on via ICQ. + +=head1 CONSTANTS + +The following constants are defined when C is imported with the +C<:standard> tag. Unless indicated otherwise, the constants are magical +scalars - they return different values in string and numeric contexts (for +instance, an error message and an error number.) + +=over 4 + +=item ADMIN_TYPE_PASSWORD_CHANGE + +=item ADMIN_TYPE_EMAIL_CHANGE + +=item ADMIN_TYPE_SCREENNAME_FORMAT + +=item ADMIN_TYPE_ACCOUNT_CONFIRM + +=item ADMIN_ERROR_UNKNOWN + +=item ADMIN_ERROR_BADPASS + +=item ADMIN_ERROR_BADINPUT + +=item ADMIN_ERROR_BADLENGTH + +=item ADMIN_ERROR_TRYLATER + +=item ADMIN_ERROR_REQPENDING + +=item ADMIN_ERROR_CONNREF + +=item VISMODE_PERMITALL + +=item VISMODE_DENYALL + +=item VISMODE_PERMITSOME + +=item VISMODE_DENYSOME + +=item VISMODE_PERMITBUDS + +=item RATE_CLEAR + +=item RATE_ALERT + +=item RATE_LIMIT + +=item RATE_DISCONNECT + +=item GROUPPERM_OSCAR + +=item GROUPPERM_AOL + +=back + +=head1 Net::AIM Compatibility + +Here are the major differences between the C interface +and the C interface: + +=over 4 + +=item * + +No get/set method. + +=item * + +No newconn/getconn method. + +=item * + +No group parameter for add_permit or add_deny. + +=item * + +Many differences in chat handling. + +=item * + +No chat_whisper. + +=item * + +No encode method - it isn't needed. + +=item * + +No send_config method - it isn't needed. + +=item * + +No send_buddies method - we don't keep a separate local buddylist. + +=item * + +No normalize method - it isn't needed. Okay, there is a normalize +function in C, but I can't think of any reason +why it would need to be used outside of the module internals. + +=item * + +Different callbacks with different parameters. + +=back + +=head1 MISCELLANEOUS + +There are two programs included with the C distribution. +oscartest is a minimalist implementation of a C client. +snacsnatcher is a tool designed for analyzing the OSCAR protocol from +libpcap-format packet captures. + +There is a class C. OSCAR screennames +are case and whitespace insensitive, and if you do something like +C<$buddy = new Net::OSCAR::Screenname "Matt Sachs"> instead of +C<$buddy = "Matt Sachs">, this will be taken care of for you when +you use the string comparison operators (eq, ne, cmp, etc.) + +C, the class used for connection objects, +has some methods that may or may not be useful to you. + +=over 4 + +=item get_filehandle + +Returns the filehandle used for the connection. Note that this is a method +of C, not C. + +=item process_one (CAN_READ, CAN_WRITE, HAS_ERROR) + +Call this when a C is ready for reading and/or +writing. You might call this yourself instead of using L<"process_connections"> +when, for instance, using the L<"connection_changed"> callback in conjunction with +C instead of C