The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
package AnyEvent::XMPP::Ext::Pubsub;
use strict;
use AnyEvent::XMPP::Util qw/simxml split_uri/;
use AnyEvent::XMPP::Namespaces qw/xmpp_ns/;
use AnyEvent::XMPP::Ext;

our @ISA = qw/AnyEvent::XMPP::Ext/;

=head1 NAME

AnyEvent::XMPP::Ext::Pubsub - Implements XEP-0060: Publish-Subscribe

=head1 SYNOPSIS

   my $con = AnyEvent::XMPP::Connection->new (...);
   $con->add_extension (my $ps = AnyEvent::XMPP::Ext::Pubsub->new);
   ...

=head1 DESCRIPTION

This module implements all tasks of handling the publish subscribe
mechanism. (partially implemented)

=cut

sub handle_incoming_pubsub_event {
    my ($self, $node) = @_;
    
    my (@items);
    if(my ($q) = $node->find_all ([qw/pubsub_ev items/])) {
        foreach($q->find_all ([qw/pubsub_ev item/])) {
            push @items, $_;
        }
    }
    $self->event(pubsub_recv => @items);
}

=head1 METHODS

=over 4

=item B<new>

This is the constructor for a pubsub object.
It takes no further arguments.

=cut

sub new {
    my $this = shift;
    my $class = ref($this) || $this;
    my $self = bless { @_ }, $class;
    $self->init;
    $self
}

sub init {
    my ($self) = @_;

    $self->reg_cb (
        ext_before_message_xml => sub {
            my ($self, $con, $node) = @_;

            my $handled = 0;
            for ($node->find_all ([qw/pubsub_ev event/])) {
                $self->stop_event;
                $self->handle_incoming_pubsub_event($_);
            }

            $handled
        }
    );
}

=item B<delete_node($con, $uri, $cb)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$cb> is the callback

Try to remove a node.

=cut

sub delete_node {
    my ($self, $con, $uri, $cb) = @_;

    my ($service, $node) = split_uri ($uri);
    
    $con->send_iq (
        set => sub {
            my ($w) = @_;
            simxml ($w, defns => 'pubsub_own', node => {
                    name => 'pubsub', childs => [
                    { name => 'delete', attrs => [ node => $node ] },
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=item B<create_node ($con, $uri, $cb)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$cb> is the callback

Try to create a node.

=cut

sub create_node {
    my ($self, $con, $uri, $cb) = @_;

    my ($service, $node) = split_uri ($uri);

    $con->send_iq (
        set => sub {
            my ($w) = @_;
            simxml ($w, defns => 'pubsub', node => {
                    name => 'pubsub', childs => [
                    { name => 'create', attrs => [ node => $node ] },
                    { name => 'configure' }
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=item B<subscribe_node($con, $uri, $cb)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$cb> is the callback

Try to retrieve items.

=cut

sub subscribe_node {
    my ($self, $con, $uri, $cb) = @_;
    my $jid = $con->jid;

    my ($service, $node) = split_uri ($uri);

    $con->send_iq (
        set => sub {
            my ($w) = @_;
            simxml ($w, defns => 'pubsub', node => {
                    name => 'pubsub', childs => [
                    { name => 'subscribe', attrs => [ 
                        node => $node,
                        jid => $jid ]
                    }
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=item B<unsubscribe_node>($con, $uri, $bc)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$cb> is the callback

Try to unsubscribe from a node.

=cut

sub unsubscribe_node {
    my ($self, $con, $uri, $cb) = @_;
    my $jid = $con->jid;

    my ($service, $node) = split_uri ($uri);

    $con->send_iq (
        set => sub {
            my ($w) = @_;
            simxml ($w, defns => 'pubsub', node => {
                    name => 'pubsub', childs => [
                    { name => 'unsubscribe', attrs => [
                        node => $node,
                        jid => $jid ]
                    }
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=item B<publish_item($con, $uri, $create_cb, $cb)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$create_cb> is the callback
C<$cb> is the callback

Try to publish an item.

=cut

sub publish_item {
    my ($self, $con, $uri, $create_cb, $cb) = @_;

    my ($service, $node) = split_uri ($uri);

    $con->send_iq (
        set => sub {
            my ($w) = @_;
            simxml ($w, defns => 'pubsub', node => {
                    name => 'pubsub', childs => [
                    { name => 'publish', attrs => [ node => $node ], childs => [
                        { name => 'item', childs => [ $create_cb ] }
                        ]
                    },
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            warn "OK $create_cb / $cb\n";
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=item B<retrive_items($con, $uri, $cb)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$cb> is the callback

Try to retrieve items.

=cut

sub retrieve_items {
    my ($self, $con, $uri, $cb) = @_;

    my($service, $node) = split_uri ($uri);

    $con->send_iq (
        get => sub {
            my ($w) = @_;
            simxml ($w, defns => 'pubsub', node => {
                    name => 'pubsub', childs => [
                    { name => 'items', attrs => [ node => $node ] }
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=item B<retrive_item($con, $uri, $id, $cb)>
C<$con> is the connection already established,
C<$uri> is the name of the node to be created
C<$id> is the id of the entry to be retrieved
C<$cb> is the cb

Try to retrieve item.

=cut

sub retrieve_item {
    my ($self, $con, $uri, $id, $cb) = @_;

    my($service, $node) = split_uri ($uri);
 
    $con->send_iq (
        get => sub {
            my ($w) = @_;
            simxml( $w, defns => 'pubsub', node => {
                    name => 'pubsub', childs => [
                    { name => 'items', attrs => [ node => $node ],
                        childs => [
                        { name => 'item', attrs => [ id => $id ] }]
                    }
                    ]
                });
        },
        sub {
            my ($node, $err) = @_;
            $cb->(defined $err ? $err : ()) if $cb;
        },
        (defined $service ? (to => $service) : ())
    );
}

=back

=head1 AUTHOR

Robin Redeker, C<< <elmex at ta-sa.org> >>, JID: C<< <elmex at jabber.org> >>

=head1 CONTRIBUTORS

Chris Miceli - additional work on the pubsub extension

=head1 COPYRIGHT & LICENSE

Copyright 2007, 2008 Robin Redeker, all rights reserved.

This program is free software; you can redistribute it and/or modify it
under the same terms as Perl itself.

=cut

1; # End of AnyEvent::XMPP::Ext::Pubsub