use XML::Simple;
use Data::Dumper;
use File::Spec;
+use List::Util qw(max);
use xmlrtorrent;
+my @talkers;
+my $talker;
my $conf;
my $conffile = File::Spec->catfile(Irssi::get_irssi_dir(), 'xmlrtorrent.xml');
my $scriptdir = File::Spec->catfile(Irssi::get_irssi_dir(), 'scripts');
+my $plugindir = File::Spec->catfile($scriptdir, 'xmlrtorrent');
my %torrentlist = ();
my $torrentindex = 1;
my $rtorrent;
my $PARAMS = {
'XMLURL' => 'http://localhost/RPC2',
+ 'USERNAME' => '',
+ 'PASSWORD' => '',
+ '_QUEUE' => {},
};
# activate debug here
-my $debug = 0;
+my $debug = 1;
# "message public", SERVER_REC, char *msg, char *nick, char *address, char *target
signal_add_last("message public" => sub {check_for_link(\@_,1,4,2,0);});
cmd_remote(@_);
},
+ 'talker' => sub {
+ cmd_talker(@_);
+ },
+
'debug' => sub {
$debug = 1;
write_irssi('Enabled debugging');
my @text = @_;
my $output = $outputstack[0];
- $text[0] = 'xmlrtorrent: ' . $text[0];
+ $text[0] = '%%mxmlrtorrent: %%n' . $text[0];
if (defined($output) and ref($output)) {
$output->print(sprintf(shift(@text), @text), MSGLEVEL_CLIENTCRAP);
push_output($witem);
# Look if there is a torrent link in there
- $message =~ m|(http://\S*\.torrent)|;
+ $message =~ m,(http://\S*\.(?:torrent|penis)),;
$m = $1;
while (defined($m)) {
write_debug('Torrent-URL: %s', $m);
$u = $torrentlist{$id}->{'URL'};
write_debug('Sending %s to rtorrent', $u);
- unless(defined($rtorrent->load_start($u))) {
- write_irssi('Error sending URL %s: %s', $u, $rtorrent->errstr());
+ unless(defined($rtorrent->load_start($talker, $u))) {
+ write_irssi('%%RError sending URL %s: %s', $u, $rtorrent->errstr());
} else {
+ write_irssi('%s enqueued', $u);
delete($torrentlist{$id});
}
+ } elsif ('add' eq $subcmd) {
+ unless(defined($id)) {
+ return;
+ }
+ $torrentlist{$torrentindex++} = {'CHANNEL' => '', 'NICK' => '', 'URL' => $id};
} elsif (('list' eq $subcmd) or !defined($subcmd)) {
+ my $l;
write_irssi('List of queued torrents');
- foreach (sort(keys(%torrentlist))) {
- write_irssi(' %d: %s@%s: %s', $_,
- $torrentlist{$_}->{'NICK'},
- $torrentlist{$_}->{'CHANNEL'},
- $torrentlist{$_}->{'URL'});
+ if (0 == scalar(keys(%torrentlist))) {
+ write_irssi(' (no torrents in local queue)');
+ } else {
+ foreach (sort(keys(%torrentlist))) {
+ write_irssi(' %3d: %s@%s: %s', $_,
+ $torrentlist{$_}->{'NICK'},
+ $torrentlist{$_}->{'CHANNEL'},
+ $torrentlist{$_}->{'URL'});
+ }
}
+ } else {
+ write_irssi('Unknown subcommand: %s', $subcmd);
}
}
my ($subcmd, $id, @params) = @_;
my $rqueue;
- if ('queue' eq $subcmd) {
- unless(defined($rqueue = $rtorrent->download_list())) {
+ if (('list' eq $subcmd) or !defined($subcmd)) {
+ unless(defined($rqueue = $rtorrent->download_list($talker))) {
write_irssi('Error getting list of downloads: %s', $rtorrent->errstr());
return;
}
- foreach (@{$rqueue}) {
- write_irssi('%s%s: %sB/%sB done, %sb/s up, %sb/s down',
- $_->[6]?'*':' ',
- $_->[0],
- format_number($_->[2]),
- format_number($_->[1]),
- format_number($_->[3]),
- format_number($_->[4]));
+ write_irssi('List of rempote torrents');
+ if (0 == scalar(@{$rqueue})) {
+ write_irssi(' (no torrents in remote queue)');
+ } else {
+ foreach (@{$rqueue}) {
+ write_irssi(' %s%s: %sB/%sB done (%d%%), %sB/s up, %sB/s down',
+ $_->[6]?'*':' ',
+ $_->[0],
+ format_number($_->[2]),
+ format_number($_->[1]),
+ ($_->[2]*100)/$_->[1],
+ format_number($_->[3]),
+ format_number($_->[4]));
+ }
}
}
}
sub cmd_save {
+
+ my %mappedqueue;
+
+ # XML::Simple has some problems with numbers as nodenames,
+ # so we have to modify our queue a bit.
+ %mappedqueue = map {("_$_" => $torrentlist{$_})} keys(%torrentlist);
eval {
open(CONF, '>'.$conffile) or die 'Could not open config file';
+ $conf->{'xmlrtorrent'}->{'_QUEUE'} = \%mappedqueue;
print CONF XML::Simple::XMLout($conf, KeepRoot => 1, KeyAttr => {'config' => 'module', 'option' => 'key'});
close(CONF);
};
my $target = shift;
my $key = shift;
my $val = shift;
+ my $p;
- if ('global' eq $target) {
- if(exists($PARAMS->{$key})) {
- $conf->{'xmlrtorrent'}->{$key} = $val;
- if ('XMLURL' eq $key) {
- unless(defined($rtorrent = xmlrtorrent->new('XMLURL' => $conf->{'xmlrtorrent'}->{'XMLURL'}))) {
- write_irssi('Could not initialize XMLRPC instance');
- return;
- }
- }
- } else {
- write_irssi('Key %s does not exist', $key);
+ foreach $p (@talkers) {
+ if ($p->{'NAME'} eq $target) {
+ $p->setval($key, $val);
+ return;
}
}
+ write_irssi(undef, 'No such module');
}
-
sub cmd_show {
my $target = shift;
my $p;
my $e;
+
+ if (defined($target)) {
+ foreach $p (@talkers) {
+ if ($p->{'NAME'} eq $target) {
+ write_irssi($p->getconfstr());
+ return;
+ }
+ }
+ write_irssi('No such module');
+ } else {
+ write_irssi('Loaded talkers:');
+ foreach $p (@talkers) {
+ write_irssi(' %s', $p->{'NAME'});
+ };
+ }
}
sub cmd_help {
my $target = shift;
my $p;
- write_irssi(<<'EOT');
+ if (defined($target)) {
+ foreach $p (@talkers) {
+ if ($p->{'NAME'} eq $target) {
+ write_irssi($p->gethelpstr());
+ return;
+ }
+ }
+ write_irssi('No such module');
+ } else {
+ write_irssi(<<'EOT');
Supported commands:
- save: Save the current configuration
- help: Display this help
+ save: save the current configuration
+ help [modulename]: display this help or module specific help
+ show [modulename]: show loaded modules or the current parameters of a module
+ talker [modulename]: display or set the talker to use
debug: enable debugging messages
nodebug: disable debugging messages
EOT
+;
+ }
}
+sub cmd_talker {
+ my $target = shift;
+ my $p;
+
+ if (defined($target)) {
+ foreach $p (@talkers) {
+ if (($p->{'NAME'} eq $target) && ($p->{'TYPE'} eq 'talker')) {
+ $talker = $p;
+ $conf->{'videosite'}->{'talker'} = $target;
+ return;
+ }
+ }
+ write_irssi('No such talker');
+ } else {
+ write_irssi('Current talker: %s', $conf->{'videosite'}->{'talker'});
+ }
+}
+
+
# save on unload
sub sig_command_script_unload {
}
}
+sub ploader {
+
+ my $dir = shift;
+ my $pattern = shift;
+ my $type = shift;
+ my @list;
+ my $p;
+ my $g;
+ my @g = ();
+
+ opendir(D, $dir) || return ();
+ @list = grep {/$pattern/ && -f File::Spec->catfile($dir, $_) } readdir(D);
+ closedir(D);
+
+ foreach $p (@list) {
+ write_debug('Trying to load %s:', $p);
+ $p =~ s/\.pm$//;
+ eval qq{ require xmlrtorrent::$p; };
+ if ($@) {
+ write_irssi('Failed to load plugin: %s', "$@");
+ next;
+ }
+
+ $g = eval qq{ xmlrtorrent::$p->new(); };
+ if ($@) {
+ write_irssi('Failed to instanciate: %s', "$@");
+ delete($INC{$p});
+ next;
+ }
+
+ write_debug('found %s %s', $g->{'TYPE'}, $g->{'NAME'});
+ if ($type eq $g->{'TYPE'}) {
+ push(@g, $g);
+ $g->setio(sub {Irssi::print(shift)});
+ } else {
+ write_irssi('%s has wrong type (got %s, expected %s)', $p, $g->{'TYPE'}, $type);
+ delete($INC{$p});
+ }
+ }
+
+ write_debug('Loaded %d plugins', $#g+1);
+
+ return @g;
+}
+
+sub _load_modules($) {
+
+ my $path = shift;
+
+ foreach (keys(%INC)) {
+ if ($INC{$_} =~ m|^$path|) {
+ write_debug('Removing %s from $INC', $_);
+ delete($INC{$_});
+ }
+ }
+ @talkers = ploader($path, '.*Talker\.pm$', 'talker');
+}
+
sub init_xmlrtorrent {
my $bindings = shift;
}
}
- unless(defined($rtorrent = xmlrtorrent->new('XMLURL' => $conf->{'xmlrtorrent'}->{'XMLURL'}))) {
+ _load_modules($plugindir);
+
+ unless (defined(@talkers)) {
+ write_irssi('No talkers found, can not proceed.');
+ return;
+ }
+
+ $talker = $talkers[0];
+ foreach $p (@talkers) {
+ if ($conf->{'xmlrtorrent'}->{'talker'} eq $p->{'NAME'}) {
+ $talker = $p;
+ }
+ }
+ write_debug(undef, 'Selected %s as talker', $talker->{'NAME'});
+ $conf->{'videosite'}->{'talker'} = $talker->{'NAME'};
+
+
+ # Restore the queue
+ %torrentlist = %{$conf->{'xmlrtorrent'}->{'_QUEUE'}};
+ %torrentlist = map { my $a = substr($_, 1); ("$a" => $torrentlist{$_}) } keys(%torrentlist);
+ $torrentindex = max(keys(%torrentlist)) + 1;
+
+ unless(defined($rtorrent = xmlrtorrent->new())) {
write_irssi('Could not initialize XMLRPC instance');
return;
}
if (exists($xmlrtorrent_commands->{$cmd})) {
$xmlrtorrent_commands->{$cmd}->(@params);
+ } else {
+ write_irssi('Unknown command: %s', $cmd);
}
pop_output();