ljr/local/bin/ljrimport/Simple.pm

4597 lines
128 KiB
Perl
Raw Normal View History

2019-02-05 21:49:12 +00:00
package LJ::Simple;
use strict;
use vars qw($VERSION @ISA @EXPORT @EXPORT_OK);
require Exporter;
require AutoLoader;
@ISA = qw(Exporter AutoLoader);
@EXPORT_OK = qw();
@EXPORT = qw();
$VERSION = '0.01';
## Bring in modules we use
use strict; # Silly not to be strict
use Socket; # Required for talking to the LJ server
use POSIX; # For errno values and other POSIX functions
## Helper function prototypes
sub Debug(@);
sub EncVal($$);
sub DecVal($);
sub SendRequest($$$$);
sub dump_list($$);
sub dump_hash($$);
=pod
=head1 NAME
LJ::Simple - A perl module to access LiveJournal via its flat protocol
=head1 SYNOPSIS
C<LJ::Simple> is an object based C<perl> module which is used to access
LiveJournal-based web logs. This module implements most of the
flat protocol LiveJournal uses; for details of this protocol please
see: L<http://www.livejournal.com/developer/protocol.bml>
=head1 REQUIREMENTS
This module requires nothing other than the modules which come with the
standard perl 5.6.1 distribution. The only modules it B<requires> are
C<POSIX> and C<Socket>.
If you have the C<Digest::MD5> module available then the code will make use of
encrypted passwords automatically. However C<Digest::MD5> is not required for
this module to work.
=head1 DESCRIPTION
C<LJ::Simple> is a trival API to access LiveJournal. Currently it
allows you to:
=over 2
=item Login
Log into the LiveJournal system
=item Post
Post a new journal entry in the LiveJournal system
=item Synchronise
Returns a list of journal entries created or modified from a given
date.
=item Edit
Edit the contents of an existing entry within the LiveJournal system
=item Delete
Delete an existing post from the LiveJournal system
=back
=head1 EXAMPLE
The following simple examples shows you how to use the module to post a
simple LiveJournal entry.
=head2 Using LJ::Simple::QuickPost()
C<LJ::Simple::QuickPost()> is a routine which allows you to quickly post an entry into
LiveJournal; as such it lacks a lot of the abilities which using the object-based
interface provides. The C<LJ::Simple::QuickPost()> routine is explained in depth below, however
the following example shows how it can be used to easily post to LiveJournal:
use LJ::Simple;
LJ::Simple::QuickPost(
user => "test",
pass => "test",
entry => "Just a simple entry",
) || die "$0: Failed to post entry: $LJ::Simple::error\n";
=head2 Using the standard calls
use LJ::Simple;
# Log into the server
my $lj = new LJ::Simple ({
user => "test",
pass => "test",
site => undef,
});
(defined $lj)
|| die "$0: Failed to log into LiveJournal: $LJ::Simple::error\n";
# Prepare the event
my %Event=();
$lj->NewEntry(\%Event) ||
die "$0: Failed to create new entry: $LJ::Simple::error\n";
# Put in the entry
my $entry=<<EOF;
A simple entry made using <tt>LJ::Simple</tt> version $LJ::Simple::VERSION
EOF
$lj->SetEntry(\%Event,$entry)
|| die "$0: Failed to set entry: $LJ::Simple::error\n";
# Say we are happy
$lj->SetMood(\%Event,"happy")
|| die "$0: Failed to set mood: $LJ::Simple::error\n";
# Post the event
my ($item_id,$anum,$html_id)=$lj->PostEntry(\%Event);
(defined $item_id)
|| die "$0: Failed to post journal entry: $LJ::Simple::error\n";
=head1 VARIABLES
There are various variables which can be used to control certain
aspects of the module. It is generally recommended that if you
wish to change these variables that you do so B<before> you
create the initial object.
The variable you are most likely to use is C<$LJ::Simple::error>
which holds error messages if any of the C<LJ::Simple> calls
fail.
=over 4
=item $LJ::Simple::error
Holds error messages, is set with a blank string at the
start of each method. Whilst the messages are relatively free-form,
there are some prefixes which are sometimes used:
CODE: An error in the code calling the API
INTERNAL: An internal error in this module
=item $LJ::Simple::debug
If set to C<1>, debugging messages are sent to stderr.
=item $LJ::Simple::protocol
If set to C<1> the protocol used to talk to the remote server is sent to stderr.
=item $LJ::Simple::raw_protocol
If set to C<1> the raw protocol used to talk to the remote server is sent to stderr;
this is only useful if you are doing debugging on C<LJ::Simple> itself as the protocol
is shown as the module gets it from the server; non-printable characters are converted
to their octal presentation form, I<ie> a newline becomes C<\012>.
It should be noted that if C<$LJ::Simple::raw_protocol> is set along with
C<$LJ::Simple::protocol> then the raw protocol display takes precedence for data
returning from the LJ server.
=item $LJ::Simple::UTF
If set to C<1> the LiveJournal server is told to expect UTF-8 encoded characters.
If you enable this the module will attempt to use the utf8 perl module.
The default is see if we have a version of Perl with UTF-8 support and use
it if its available.
=item $LJ::Simple::challenge
If set to C<1> we make use of the challenge-response system instead of using
plain or hashed passwords. This does add some overhead into processing requests
since every action has to be preceeded by a request for a challenge value from
the server.
The default is to see if we have the C<Digest::MD5> module available and if
so we make use of the challenge-response system. This can be disabled by
setting the variable to C<0>.
=item $LJ::Simple::timeout
The time - specified in seconds - to wait for data from the server. If
given a value of C<undef> the API will block until data is avaiable.
=item $LJ::Simple::NonBlock
By default this is set to C<undef>. When given a reference to a sub-routine this
module will call the given sub-routine at various stages of processing the responses
to the LiveJournal server. This is intended for GUI applications which need to process
event queues, update progress bars, I<etc>. When called the sub-routine is passed a
number of variables which maybe useful; the calling method is:
&{sub}($mode,$status,$action,$bytes_in,$bytes_out,$time,$waiting)
$mode - The mode sent to the LJ server
$status - The status of the request; ranges from 0 to 1
$action - The action performed
$bytes_in - The number of bytes read from the remote server
$bytes_out - The number of bytes written to the remote server
$time - The time taken so far in seconds
$waiting - Are we waiting for a response from the server ?
It should be noted that if C<$waiting> is set to C<1> then it is B<highly> recommended
that the sub-routine calls C<select()> itself to provide at least some time delay. If
this is not done it is likely that this module will consume far more CPU than necessary.
An example sub-routine follows:
sub LJStatus {
my ($mode,$status,$action,$bytes_in,$bytes_out,$time,$waiting) = @_;
print "\$mode = $mode\n";
print "\$status = $status\n";
print "\$action = $action\n";
print "\$bytes_in = $bytes_in\n";
print "\$bytes_out = $bytes_out\n";
print "\$time = $time\n";
print "\$waiting = $waiting\n";
print "\n";
($waiting) && select(undef,undef,undef,0.5);
}
$LJ::Simple::NonBlock=\&LJStatus;
=item $LJ::Simple::ProtoSub
By default this points to a sub-routine within the module; this is called when
the protocol between the module and LiveJournal server is to be shown, in other
words when C<$LJ::Simple::protocol> is set to C<1>. The sub-routine called must
take two variables; it is called in the following way:
&{sub}($direction,$data,$server,$ip_addr)
$direction - The direction of the flow; 0 means from client to server
and 1 means from server to client
$data - The data which has flowed; there should not be any newlines
with the data, but do not rely on this.
$server - The name of the LJ server we are talking to
$ip_addr - The IP address of the LJ server we are talking to
If both variables are C<undef> then data is about to flow. If just C<$direction> is
C<undef> then C<$data> holds an informational message.
The standard sub-routine which is called is:
sub DefaultProtoSub {
my ($direct,$data,$server,$ip_addr)=@_;
my $arrow="--> ";
if (!defined $direct) {
if (!defined $data) {
print STDERR "Connecting to $server [$ip_addr]\n";
print STDERR "Lines starting with \"-->\" is data SENT to the server\n";
print STDERR "Lines starting with \"<--\" is data RECEIVED from the server\n";
return;
}
$arrow="";
} else {
($direct) && ($arrow="<-- ");
}
print STDERR "$arrow$data\n";
}
$LJ::Simple::ProtoSub=\&DefaultProtoSub;
=item $LJ::Simple::buffer
The number of bytes to try and read in on each C<sysread()> call.
=back
=cut
sub DefaultProtoSub {
my ($direct,$data,$server,$ip_addr)=@_;
my $arrow="--> ";
if (!defined $direct) {
if (!defined $data) {
print STDERR "Connecting to $server [$ip_addr]\n";
print STDERR "Lines starting with \"-->\" is data SENT to the server\n";
print STDERR "Lines starting with \"<--\" is data RECEIVED from the server\n";
return;
}
$arrow="";
} else {
($direct) && ($arrow="<-- ");
}
print STDERR "$arrow$data\n";
}
## Global variables - documented
# Debug ?
$LJ::Simple::debug=0;
# Show protocol ?
$LJ::Simple::protocol=0;
# Protocol handling code
$LJ::Simple::ProtoSub=\&DefaultProtoSub;
# Show raw protocol ?
$LJ::Simple::raw_protocol=0;
# Use UTF-8 ?
$LJ::Simple::UTF = undef;
# Use challenge-response ?
$LJ::Simple::challenge = undef;
# Use non-block sub-routine
$LJ::Simple::NonBlock = undef;
# Errors
$LJ::Simple::error="";
# Timeout for reading from sockets - default is 5 minutes
$LJ::Simple::timeout = 300;
# How much data to read from the socket in one read()
$LJ::Simple::buffer = 8192;
# How much times to retry if some network error occurs
$LJ::Simple::network_retries = 10;
# How much seconds to wait before each retry
$LJ::Simple::network_sleep = 2;
# LJ Client string
$LJ::Simple::LJ_Client = "LJ::Simple/$VERSION";
# UserAgent string
$LJ::Simple::UserAgent = "LJ::Simple/$VERSION; http://www.bpfh.net/computing/software/LJ::Simple/; lj-simple\@bpfh.net";
## Global variables - internal and undocumented
# Should we not fully run the QuickPost routine ?
$LJ::Simple::TestStopQuickPost = 0;
## Internal variables - private to this module
# Standard ports
my %StdPort = (
http => 80,
http_proxy => 3128,
);
=pod
=head1 AVAILABLE METHODS
=head2 LJ::Simple::QuickPost()
C<LJ::Simple::QuickPost()> is a routine which allows you to quick post to LiveJournal.
However it does this by hiding a lot of the details involved in using
C<LJ::Simple> to do this. This routine will do all of the work involved in
logging into the LiveJournal server, preparing the entry and then posting it.
If at any stage there is a failure then C<0> is returned and C<$LJ::Simple::error>
will contain the reason why. If the entry was successfully posted to the LiveJournal
server then the routine will return C<1>.
There are a number of options to the C<LJ::Simple::QuickPost()> routine:
LJ::Simple::QuickPost(
user => Username
pass => Password
entry => Contents of the entry
subject => Subject line of the entry
mood => Current mood
music => Current music
html => HTML content ?
protect => Security settings of the entry
groups => Friends groups list
);
Of these, only the C<user>, C<pass> and C<entry> options are required; all of the other
options are optional. The option names are all case insensitive.
=over 4
=item user
The username who owns the journal the entry should be posted to;
this option is B<required>.
=item pass
The password of the C<user>;
this option is B<required>.
=item entry
The actual entry itself;
this option is B<required>.
=item subject
The subject line of the post.
=item mood
The mood to associate with the post; the value is given to the C<SetMood()> method
for processing.
=item music
The music to associate with the post.
=item html
This is a boolean value of either C<1> or C<0>. If you want to say that the entry
contains HTML and thus should be considered to be preformatted then set C<html> to
C<1>. Otherwise you can either set it to C<0> or not give the option.
=item protect
By default the new entry will be public unless you give the C<protect> option. This
option should be given the protection level required for the post and can be one of
the following:
public - The entry is public
friends - Entry is friends-only
groups - Entry is restricted to friends groups
private - Entry is restricted to the journal's owner
If you set the C<protect> option to C<groups> you must also include the C<groups>
option - see below for details.
=item groups
If the C<protect> option is set to C<groups> then this option should contain a
list reference which contains the list of groups the entry should be restricted to.
This option is B<required> if the C<protect> option is set to C<groups>.
=back
Example code:
# Simple test post
LJ::Simple::QuickPost(
user => "test",
pass => "test",
entry => "Just a simple entry",
) || die "$0: Failed to post entry: $LJ::Simple::error\n";
# A friends-only preformatted entry
LJ::Simple::QuickPost(
user => "test",
pass => "test",
entry => "<p>Friends-only, preformatted, entry</p>",
html => 1,
protect => "friends",
) || die "$0: Failed to post entry: $LJ::Simple::error\n";
# A entry restricted to several friends groups
LJ::Simple::QuickPost(
user => "test",
pass => "test",
entry => "Entry limited to friends groups",
protect => "groups",
groups => [qw( one_group another_group )],
) || die "$0: Failed to post entry: $LJ::Simple::error\n";
=cut
sub QuickPost(@) {
my %opts=();
my @prot_opts=();
while($#_>-1) {
my $k=lc(shift(@_));
my $v=shift(@_);
(defined $v) || next;
$opts{$k}=$v;
}
foreach (qw( user pass entry )) {
(exists $opts{$_}) && next;
$LJ::Simple::error="CODE: QuickPost() called without the required $_ option";
return 0;
}
if ((exists $opts{html}) && ($opts{html}!~/^[01]$/)) {
$LJ::Simple::error="CODE: QuickPost() not given either 0 or 1 for html option";
return 0;
}
if ((exists $opts{protect}) && ($opts{protect} eq "groups")) {
if (!exists $opts{groups}) {
$LJ::Simple::error="CODE: QuickPost() given protect=groups, but no groups option";
return 0;
}
if (ref($opts{groups}) ne "ARRAY") {
$LJ::Simple::error="CODE: QuickPost() not given a list reference for the groups option";
return 0;
}
@prot_opts=@{$opts{groups}};
}
# Kludge so we can test the input validation
($LJ::Simple::TestStopQuickPost) && return 1;
my $lj = new LJ::Simple({
user => $opts{user},
pass => $opts{pass},
});
(defined $lj) || return 0;
my %Event=();
$lj->NewEntry(\%Event) || return 0;
$lj->SetEntry(\%Event,$opts{entry}) || return 0;
(exists $opts{subject}) &&
($lj->SetSubject(\%Event,$opts{subject}) || return 0);
(exists $opts{mood}) &&
($lj->SetMood(\%Event,$opts{mood}) || return 0);
(exists $opts{music}) &&
($lj->Setprop_current_music(\%Event,$opts{music}) || return 0);
(exists $opts{html}) &&
($lj->Setprop_preformatted(\%Event,$opts{html}) || return 0);
(exists $opts{protect}) &&
($lj->SetProtect(\%Event,$opts{protect},@prot_opts) || return 0);
my ($item_id,$anum,$html_id)=$lj->PostEntry(\%Event);
(defined $item_id) || return 0;
return 1;
}
=pod
=head2 Object creation
=over 4
=item login
Logs into the LiveJournal system.
## Simplest logon method
my $lj = new LJ::Simple ( {
user => "username",
pass => "password",
} );
## Login with options
my $lj = new LJ::Simple ( {
user => "username",
pass => "password",
site => "hostname[:port]",
proxy => "hostname[:port]",
moods => 0 | 1,
pics => 0 | 1,
fast => 0 | 1,
} );
## Login by using login()
my $lj = LJ::Simple->login ( {
user => "username",
pass => "password",
site => "hostname[:port]",
proxy => "hostname[:port]",
moods => 0 | 1,
pics => 0 | 1,
fast => 0 | 1,
} );
Where:
user is the username to use
pass is the password associated with the username
site is the remote site to use
proxy is the HTTP proxy site to use; see below.
moods is set to 0 if we do not want to download the mood
list. Defaults to 1
pics is set to 0 if we do not want to download the user
picture information. Defaults to 1
fast is set to 1 if we want to perform a fast login.
Default is 0. See below for details of this.
Sites defined in C<site> or C<proxy> are a hostname with an
optional port number, separated by a C<:>, i.e.:
www.livejournal.com
www.livejournal.com:80
If C<site> is given C<undef> then the code assumes that you wish to
connect to C<www.livejournal.com:80>. If no port is given then port
C<80> is the default.
If C<proxy> is given C<undef> then the code will go directly to the
C<$site> unless a suitable environment variable is set.
If no port is given then port C<3128> is the default.
C<LJ::Simple> also supports the use the environment variables C<http_proxy>
and C<HTTP_PROXY> to store the HTTP proxy server details. The format of these
environment variables is assumed to be:
http://server[:port]/
Where C<server> is the name of the proxy server and the optional C<port> the
proxy server is on - port C<3128> is used if no port is explicitly given.
It should be noted that the proxy environment variables are B<only> checked
if the C<proxy> value is B<NOT> given to the C<LJ::Simple> object creation.
Thus to disable looking at the proxy environment variables use
C<proxy=E<gt>undef> in C<new()> or C<login()>.
If C<moods> is set to C<0> then the mood list will not be pulled from
the LiveJournal server and the following functions will be affected:
o moods() will always return undef (error)
o Setprop_current_mood_id() will not validate the mood_id
given to it.
o SetMood() will not attempt to convert the string it is
given into a given mood_id
If C<pics> is set to C<0> then the data on the user pictures will
not be pulled from the LiveJournal server and the following
functions will be affected:
o pictures() will always return undef (error)
o Setprop_picture_keyword() will blindly set the picture keyword
you give it - no validation will be performed.
o DefaultPicURL() will always return undef (error)
If C<fast> is set to C<1> then we will perform a I<fast login>. Essentially
all this does is to set up the various entries in the object hash which
the routines called after C<login> expect to see; at no time does it talk to
the LiveJournal servers. What this means is that it is very fast. However it
also means that when you use parts of the API which B<do> talk to the LiveJournal
servers its quite possible that you will get back errors associated with
authentication errors, network outages, I<etc>. In other words, in C<fast> mode
the login will always succeed, no matter what the state the LiveJournal
server we're talking is in. It should be noted that the following functions
will be affected if you enable the I<fast login>:
o moods() will always return undef (error)
o Setprop_current_mood_id() will not validate the mood_id
given to it
o SetMood() will not attempt to convert the string it is
given into a given mood_id
o pictures() will always return undef (error)
o Setprop_picture_keyword() will blindly set the picture keyword
you give it - no validation will be performed
o communities() will always return an empty list
o MemberOf() will always return 0 (error)
o UseJournal() will not validate the shared journal name you
give it
o groups() will always return undef (error)
o MapGroupToId() will always undef (error)
o MapIdToGroup() will always undef (error)
o SetProtectGroups() will always 0 (error)
o message() will always return undef (error)
o The key of "groups" in the list of hashes returned by
GetFriends() will always point to an empty list
o CheckFriends() will return undef (error) if you give it a
list of groups
On success this sub-routine returns an C<LJ::Simple> object. On
failure it returns C<undef> with the reason for the failure being
placed in C<$LJ::Simple::error>.
Example code:
## Simple example, going direct to www.livejournal.com:80
my $lj = new LJ::Simple ({ user => "someuser", pass => "somepass" });
(defined $lj) ||
die "$0: Failed to access LiveJournal - $LJ::Simple::error\n";
## More complex example, going via a proxy server on port 3000 to a
## a LiveJournal system available on port 8080 on the machine
## www.somesite.com.
my $lj = new LJ::Simple ({
user => "someuser",
pass => "somepass",
site => "www.somesite.com:8080",
proxy => "proxy.internal:3000",
});
(defined $lj) ||
die "$0: Failed to access LiveJournal - $LJ::Simple::error\n";
## Another complex example, this time saying that we do not want
## the mood list or user pictures downloaded
my $lj = new LJ::Simple ({
user => "someuser",
pass => "somepass",
pics => 0,
moods => 0,
});
(defined $lj) ||
die "$0: Failed to access LiveJournal - $LJ::Simple::error\n";
## Final example - this one shows the use of the fast logon
my $lj = new LJ::Simple ({
user => "someuser",
pass => "somepass",
fast => 1,
});
(defined $lj) ||
die "$0: Failed to access LiveJournal - $LJ::Simple::error\n";
=cut
##
## Log into the LiveJournal system. Given that the LJ stuff is just
## layered over HTTP, its not essential to do this. However it does
## mean that we can check the auth details, get some useful info for
## later, etc.
##
sub login($$) {
# Handle the OOP stuff
my $this=shift;
$LJ::Simple::error="";
if ($#_ != 0) {
$LJ::Simple::error="CODE: Incorrect usage of login() for argv - see docs";
return undef;
}
# Get the hash
my $hr = shift;
my $class = ref($this) || $this;
my $self = {};
bless $self,$class;
if ((!exists $hr->{user})||($hr->{user} eq "") ||
(!exists $hr->{pass})||($hr->{pass} eq "")) {
$LJ::Simple::error="CODE: Incorrect usage of login() - see docs";
return undef;
}
$self->{auth}={
user => $hr->{user},
pass => $hr->{pass},
challenge => {},
};
if (! defined $LJ::Simple::UTF) {
eval { require utf8 };
if (!$@) {
$LJ::Simple::UTF=1;
Debug("UTF-8 support found");
} else {
$LJ::Simple::UTF=0;
Debug("No UTF-8 support found");
}
} elsif ($LJ::Simple::UTF) {
eval { require utf8 };
if (!$@) {
Debug("Using UTF-8 as requested");
} else {
$LJ::Simple::error="CODE: no UTF-8 support in your version of perl";
return undef;
}
}
eval { require Digest::MD5 };
if (!$@) {
Debug("Using Digest::MD5");
my $md5=Digest::MD5->new;
$md5->add($hr->{pass});
$self->{auth}->{hash}=$md5->hexdigest;
delete $self->{auth}->{pass};
(!defined $LJ::Simple::challenge) && ($LJ::Simple::challenge=1);
} else {
if ((defined $LJ::Simple::challenge)&&($LJ::Simple::challenge)) {
$LJ::Simple::error="Challenge-response auth requested, no Digest::MD5 found";
return undef;
}
$LJ::Simple::challenge=0;
}
if ((exists $hr->{site})&&(defined $hr->{site})&&($hr->{site} ne "")) {
my $site_port=$StdPort{http};
if ($hr->{site}=~/\s*(.*?):([0-9]+)\s*$/) {
$hr->{site} = $1;
$site_port = $2;
}
$self->{lj}={
host => $hr->{site},
port => $site_port,
}
} else {
$self->{lj}={
host => "www.livejournal.com",
port => $StdPort{http},
}
}
if ((exists $hr->{proxy})&&(defined $hr->{proxy})&&($hr->{proxy} ne "")) {
my $proxy_port=$StdPort{http_proxy};
if ($hr->{proxy}=~/\s*(.*?):([0-9]+)\s*$/) {
$hr->{proxy} = $1;
$proxy_port = $2;
}
$self->{proxy}={
host => $hr->{proxy},
port => $proxy_port,
};
} elsif (!exists $hr->{proxy}) {
# Getting proxy details from the environment; assumes that the proxy is
# given as http://site[:port]/
# The first matching env is used.
foreach my $env (qw( http_proxy HTTP_PROXY )) {
(exists $ENV{$env}) || next;
($ENV{$env}=~/^(?:http:\/\/)([^:\/]+)(?::([0-9]+)){0,1}/o) || next;
$self->{proxy}={
host => $1,
port => $2,
};
(defined $self->{proxy}->{port}) || ($self->{proxy}->{port}=$StdPort{http_proxy});
}
} else {
$self->{proxy}=undef;
}
# Set fastserver to 0 until we know better
$self->{fastserver}=0;
if ((exists $hr->{fast}) && ($hr->{fast}==1)) {
## Doing fast login, so return object
Debug(dump_hash($self,""));
return $self;
}
my $GetMoods=1;
if ((exists $hr->{moods}) && ($hr->{moods}==0)) {
$GetMoods=0;
}
my $GetPics=1;
if ((exists $hr->{pics}) && ($hr->{pics}==0)) {
$GetPics=0;
}
# Perform the actual login
$self->SendRequest("login", {
"moods" => $GetMoods,
"getpickws" => $GetPics,
"getpickurls" => $GetPics,
},undef) || return undef;
# Now see if we can set fastserver
if ( (exists $self->{request}->{lj}->{fastserver}) &&
($self->{request}->{lj}->{fastserver} == 1) ) {
$self->{fastserver}=1;
}
# Moods
$self->{moods}=undef;
$self->{mood_map}=undef;
# Shared access journals
$self->{access}=undef;
# User groups
$self->{groups}=undef;
# Images defined
$self->{pictures}=undef;
# Default URL
$self->{defaultpicurl}=undef;
# Message from LJ
$self->{message}=undef;
# Handle moods, etc.
my ($k,$v)=(undef,undef);
while(($k,$v) = each %{$self->{request}->{lj}}) {
# Message from LJ
if ($k eq "message") {
$self->{message}=$v;
# Moods
} elsif ($k=~/^mood_([0-9]+)_([a-z]+)/o) {
my ($id,$type)=($1,$2);
if (!defined $self->{moods}) {
$self->{moods}={};
}
if (!exists $self->{moods}->{$id}) {
$self->{moods}->{$id}={};
}
if ($type eq "id") {
$self->{moods}->{$id}->{id}=$v;
} elsif ($type eq "name") {
$self->{moods}->{$id}->{name}=$v
}
# Picture key words
} elsif ($k=~/^(pickw_count)/o) {
if (!defined $self->{pictures}) {
$self->{pictures}={};
}
} elsif ($k eq "defaultpicurl") {
$self->{defaultpicurl}=$v;
} elsif ($k=~/^(pickw[^_]*)_([0-9]+)/o) {
my ($type,$id)=($1,$2);
if (!defined $self->{pictures}) {
$self->{pictures}={};
}
if (!exists $self->{pictures}->{$id}) {
$self->{pictures}->{$id}={};
}
if ($type eq "pickwurl") {
$self->{pictures}->{$id}->{url}=$v;
} elsif ($type eq "pickw") {
$self->{pictures}->{$id}->{name}=$v
}
# Shared access journals
} elsif ($k=~/^access_([0-9]+)/) {
if (!defined $self->{access}) {
$self->{access}={};
}
$self->{access}->{$v}=1;
# Groups
} elsif ($k=~/^frgrp_([0-9]+)_(.*)/) {
my ($id,$type)=($1,$2);
if (!defined $self->{groups}) {
$self->{groups}={
src => {}, # Source data
id => {}, # Id -> name mapping
name => {}, # Real data, name keyed
};
}
if (!exists $self->{groups}->{src}->{$id}) {
$self->{groups}->{src}->{$id}={};
}
if ($type eq "sortorder") {
$self->{groups}->{src}->{$id}->{sort}=$v;
} elsif ($type eq "name") {
$self->{groups}->{src}->{$id}->{name}=$v
}
}
}
## We now handle the group hash fully. Note in the case
## of groups having the same name, only the first will
## go into the name hash.
($k,$v)=(undef,undef);
while(($k,$v)=each %{$self->{groups}->{src}}) {
$self->{groups}->{id}->{$k}=$v->{name};
if (!exists $self->{groups}->{name}->{$v->{name}}) {
$self->{groups}->{name}->{$v->{name}} = {
id => $k,
name => $v->{name},
sort => $v->{sort},
};
}
}
##
## And now we handle the mood map fully
##
if ($GetMoods) {
$self->{mood_map}={};
foreach (values %{$self->{moods}}) {
$self->{mood_map}->{lc($_->{name})}=$_->{id};
}
}
Debug(dump_hash($self,""));
## Logged in, so return self.
return $self;
}
## Define reference from new to login
*new="";
*new=\&login;
=pod
=back
=head2 Getting data from the LiveJournal login
=over 4
=item $lj->message()
Returns back a message set in the LiveJournal system. Either
returns back the message or C<undef> if no message is set.
Example code:
my $msg = $lj->message();
(defined $msg) &&
print "LJ Message: $msg\n";
=cut
sub message($) {
my $self=shift;
return $self->{message};
}
=pod
=item $lj->moods($hash_ref)
Takes a reference to a hash and fills it with information about
the moods returned back by the server. Either returns back the
same hash reference or C<undef> on error.
Note that if the LiveJournal
object was created with either C<moods> set to C<0> or
with C<fast> set to C<1> then this function will always return
an error.
The hash the given reference is pointed to is emptied before
it is used and after a successful call the hash given will
contain:
%hash = (
list => [ list of mood names, alphabetical ]
moods => {
mood_name => mood_id
}
idents => {
mood_id => mood_name
}
)
Example code:
my %Moods=();
if (!defined $lj->moods(\%Moods)) {
die "$0: LJ error - $LJ::Simple::error";
}
foreach (@{$Moods{list}}) {
print "$_ -> $Moods{moods}->{$_}\n";
}
=cut
sub moods($$) {
my $self=shift;
my ($hr) = @_;
$LJ::Simple::error="";
if (ref($hr) ne "HASH") {
$LJ::Simple::error="CODE: moods() not given a hash reference";
return undef;
}
if (!defined $self->{moods}) {
$LJ::Simple::error="Unable to return moods - not requested at login";
return undef;
}
%{$hr}=(
list => [],
moods => {},
idents => {},
);
my ($k,$v);
while(($k,$v)=each %{$self->{moods}}) {
push(@{$hr->{list}},$v->{name});
$hr->{moods}->{$v->{name}}=$v->{id};
$hr->{idents}->{$v->{id}}=$v->{name};
}
$hr->{list} = [ (sort { $a cmp $b } @{$hr->{list}}) ];
return $hr;
}
=pod
=item $lj->communities()
Returns a list of shared access communities the user logged in can
post to. Returns an empty list if no communities are available
Example code:
my @communities = $lj->communities();
print join("\n",@communities),"\n";
=cut
sub communities($) {
my $self=shift;
$LJ::Simple::error="";
(defined $self->{access}) || return ();
return sort {$a cmp $b} (keys %{$self->{access}});
}
=pod
=item $lj->MemberOf($community)
Returns C<1> if the user is a member of the named community. Returns
C<0> otherwise.
Example code:
if ($lj->MemberOf("some_community")) {
:
:
:
}
=cut
sub MemberOf($$) {
my $self=shift;
my ($community)=@_;
$LJ::Simple::error="";
(defined $self->{access}) || return 0;
return (exists $self->{access}->{$community});
}
=pod
=item $lj->groups($hash_ref)
Takes a reference to a hash and fills it with information about
the friends groups the user has configured for themselves. Either
returns back the hash reference or C<undef> on error.
The hash the given reference points to is emptied before it is
used and after a successful call the hash given will contain
the following:
%hash = (
"name" => {
"Group name" => {
id => "Number of the group",
sort => "Sort order",
name => "Group name (copy of key)",
},
},
"id" => {
"Id" => "Group name",
},
);
Example code:
my %Groups=();
if (!defined $lj->groups(\%Groups)) {
die "$0: LJ error - $LJ::Simple::error";
}
my ($id,$name)=(undef,undef);
while(($id,$name)=each %{$Groups{id}}) {
my $srt=$Groups{name}->{$name}->{sort};
print "$id\t=> $name [$srt]\n";
}
=cut
sub groups($$) {
my $self=shift;
my ($hr) = @_;
$LJ::Simple::error="";
if (ref($hr) ne "HASH") {
$LJ::Simple::error="CODE: groups() not given a hash reference";
return undef;
}
if (!defined $self->{groups}) {
$LJ::Simple::error="Unable to return groups - none defined";
return undef;
}
%{$hr}=(
name => {},
id => {},
);
my ($k,$v);
while(($k,$v)=each %{$self->{groups}->{id}}) {
$hr->{id}->{$k}=$v;
}
while(($k,$v)=each %{$self->{groups}->{name}}) {
$hr->{name}->{$k}={};
my ($lk,$lv);
while(($lk,$lv)=each %{$self->{groups}->{name}->{$k}}) {
$hr->{name}->{$k}->{$lk}=$lv;
}
}
return $hr;
}
=pod
=item $lj->MapGroupToId($group_name)
Used to map a given group name to its identity. On
success returns the identity for the group name. On
failure it returns C<undef> and sets
C<$LJ::Simple::error>.
=cut
sub MapGroupToId($$) {
my $self=shift;
my ($grp)=@_;
$LJ::Simple::error="";
if (!defined $self->{groups}) {
$LJ::Simple::error="Unable to map group to id - none defined";
return undef;
}
if (!exists $self->{groups}->{name}->{$grp}) {
$LJ::Simple::error="No such group";
return undef;
}
return $self->{groups}->{name}->{$grp}->{id};
}
=pod
=item $lj->MapIdToGroup($id)
Used to map a given identity to its group name. On
success returns the group name for the identity. On
failure it returns C<undef> and sets
C<$LJ::Simple::error>.
=cut
sub MapIdToGroup($$) {
my $self=shift;
my ($id)=@_;
$LJ::Simple::error="";
if (!defined $self->{groups}) {
$LJ::Simple::error="Unable to map group to id - none defined";
return undef;
}
if (!exists $self->{groups}->{id}->{$id}) {
$LJ::Simple::error="No such group ident";
return undef;
}
return $self->{groups}->{id}->{$id};
}
=pod
=item $lj->pictures($hash_ref)
Takes a reference to a hash and fills it with information about
the pictures the user has configured for themselves. Either
returns back the hash reference or C<undef> on error. Note that
the user has to have defined picture keywords for this to work.
Note that if the LiveJournal
object was created with either C<pics> set to C<0> or
with C<fast> set to C<1> then this function will always return
an error.
The hash the given reference points to is emptied before it is
used and after a successful call the hash given will contain
the following:
%hash = (
"keywords" => "URL of picture",
);
Example code:
my %pictures=();
if (!defined $lj->pictures(\%pictures)) {
die "$0: LJ error - $LJ::Simple::error";
}
my ($keywords,$url)=(undef,undef);
while(($keywords,$url)=each %pictures) {
print "\"$keywords\"\t=> $url\n";
}
=cut
sub pictures($$) {
my $self=shift;
my ($hr)=@_;
$LJ::Simple::error="";
if (!defined $self->{pictures}) {
$LJ::Simple::error="Unable to return pictures - none defined";
return undef;
}
if (ref($hr) ne "HASH") {
$LJ::Simple::error="CODE: pictures() not given a hash reference";
return undef;
}
%{$hr}=();
foreach (values %{$self->{pictures}}) {
$hr->{$_->{name}}=$_->{url};
}
return $hr;
}
=pod
=item $lj->DefaultPicURL()
Returns the URL of the default picture used by the user.
Note that if the LiveJournal
object was created with either C<pics> set to C<0> or
with C<fast> set to C<1> then this function will always return
an error.
Example code:
print $lj->DefaultPicURL(),"\n";
=cut
sub DefaultPicURL($) {
my $self=shift;
$LJ::Simple::error="";
if (!defined $self->{defaultpicurl}) {
$LJ::Simple::error="Unable to return default picture URL - none defined";
return undef;
}
return $self->{defaultpicurl};
}
=pod
=item $lj->user()
Returns the username used to log into LiveJournal
Example code:
my $user = $lj->user();
=cut
sub user($) {
my $self=shift;
$LJ::Simple::error="";
return $self->{auth}->{user};
}
=pod
=item $lj->fastserver()
Used to tell if the user which was logged into the LiveJournal system can use the
fast servers or not. Returns C<1> if the user can use the fast servers, C<0>
otherwise.
Example code:
if ($lj->fastserver()) {
print STDERR "Using fast server for ",$lj->user(),"\n";
}
=cut
sub fastserver($) {
my $self=shift;
$LJ::Simple::error="";
return $self->{fastserver};
}
=pod
=back
=head2 Dealing with friends
=over 4
=item $lj->GetFriendOf()
Returns a list of the other LiveJournal users who list the current
user as a friend. The list returned contains a least one entry, the
number of entries in the list. This value can range from 0 to however
many users are in the list. In the event of a failure this value is
undefined.
The list of friends is a list of hash references which contain data
about the users who list the current user as a friend. Each hash
referenced will contain the following:
{
user => The LiveJournal username
name => The full name of the user
fg => The foreground colour which represents the user
bg => The background colour which represents the user
status => The status of the user
type => The type of the user
}
Both the C<bg> and C<fg> values are stored in the format of "C<#>I<RR>I<GG>I<BB>"
where the I<RR>, I<GG>, I<BB> values are given as two digit hexadecimal numbers which
range from C<00> to C<ff>.
The C<status> of a user can be one of C<active>, C<deleted>, C<suspended> or C<purged>.
The C<type> of a user can either be C<user> which means that the user is a normal
LiveJournal user or it can be C<community> which means that the user is actually a
community which the current LJ user is a member of.
It should be noted that any of the values in the hash above can be undefined if
that value was not returned from the LiveJournal server.
The returned list is ordered by the LiveJournal login names of the users.
Example code:
my ($num_friends_of,@FriendOf)=$lj->GetFriendOf();
(defined $num_friends_of) ||
die "$0: Failed to get friends of user - $LJ::Simple::error\n";
print "LJ login\tReal name\tfg\tbg\tStatus\tType\n";
foreach (@FriendOf) {
print "$_->{user}\t",
"$_->{name}\t",
"$_->{fg}\t",
"$_->{bg}\t",
"$_->{status}\t",
"$_->{type}\n";
}
=cut
sub GetFriendOf($) {
my $self=shift;
$LJ::Simple::error="";
my %Event=();
my %Resp=();
$self->SendRequest("friendof",\%Event,\%Resp) || return undef;
my %Friends=();
my ($k,$v);
while(($k,$v)=each %Resp) {
($k=~/^friendof_([0-9]+)_(.*)/) || next;
my ($id,$type)=($1,$2);
if (!exists $Friends{$id}) {
$Friends{$id}={
user => undef,
name => undef,
bg => undef,
fg => undef,
status => "active",
type => "user",
};
}
$Friends{$id}->{$type}=$v;
}
my @lst=sort {$a->{user} cmp $b->{user}} (values %Friends);
return ($#lst+1,@lst);
}
=pod
=item $lj->GetFriends()
Returns a list of the other LiveJournal user who are listed as friends of
the current user. The list returned contains a least one entry, the
number of entries in the list. This value can range from 0 to however
many users are in the list. In the event of a failure this value is
undefined.
The list of friends is a list of hash references which contain data
about the users who list the current user as a friend. Each hash
referenced will contain the following:
{
user => The LiveJournal username
name => The full name of the user
fg => The foreground colour which represents the user
bg => The background colour which represents the user
dob => The date of birth for the user
birthday => The birthday of the user
groups => The list of friends groups this user is in
groupmask => The actual group mask for this user
status => The status of the user
type => The type of the user
}
Both the C<bg> and C<fg> values are stored in the format of "C<#>I<RR>I<GG>I<BB>"
where the I<RR>, I<GG>, I<BB> values are given as two digit hexadecimal numbers which
range from C<00> to C<ff>.
The C<dob> value is stored as a Unix timestamp; that is seconds since epoch. If the
user has no date of birth defined B<or> they have only given their birthday then this
value will be C<undef>.
The C<birthday> value is the date of the user's next birthday given as a Unix timestamp.
The C<groups> value is a reference to a list of the friends group this user is a member
of. It should be noted that to have any items in the list the user must be a
member of a friends group and the C<login()> method must B<not> have been called
with the fast login option.
The C<groupmask> value is the actual group mask for the user. This is used to build
the C<groups> list. It is a 32-bit number where each bit represents membership of a
given friends group. Bits 0 and 31 are reserved; all other bits can be used. The bit
a group corresponds to is taken by bit-shifting 1 by the group id number.
The C<status> of a user can be one of C<active>, C<deleted>, C<suspended> or C<purged>.
The C<type> of a user can either be C<user> which means that the user is a normal
LiveJournal user or it can be C<community> which means that the user is actually a
community which the current LJ user is a member of.
It should be noted that any of the values in the hash above can be undefined if
that value was not returned from the LiveJournal server.
The returned list is ordered by the LiveJournal login names of the users.
Example code:
use POSIX;
my ($num_friends,@Friends)=$lj->GetFriends();
(defined $num_friends) ||
die "$0: Failed to get friends - $LJ::Simple::error\n";
my $f=undef;
foreach $f (@Friends) {
foreach (qw(dob birthday)) {
(defined $f->{$_}) || next;
$f->{$_}=strftime("%Y/%m/%d",gmtime($f->{$_}));
}
my ($k,$v)=(undef,undef);
while(($k,$v)=each %{$f}) {
(!defined $v) && ($f->{$k}="[undefined]");
}
print "$f->{user}\n";
print " Name : $f->{name}\n";
print " Colors : fg->$f->{fg} bg->$f->{bg}\n";
print " DOB : $f->{dob}\n";
print " Next birthday: $f->{birthday}\n";
print " Status : $f->{status}\n";
print " Type : $f->{type}\n";
if ($#{$f->{groups}}>-1) {
print " Friend groups:\n";
print " + ",join("\n + ",@{$f->{groups}}),"\n";
} else {
print " Friend groups: [none]\n";
}
print "\n";
}
=cut
sub GetFriends($) {
my $self=shift;
$LJ::Simple::error="";
my %Event=(
includegroups => 1,
includebdays => 1,
);
my %Resp=();
$self->SendRequest("getfriends",\%Event,\%Resp) || return undef;
my %Friends=();
my ($k,$v);
while(($k,$v)=each %Resp) {
($k=~/^friend_([0-9]+)_(.*)/) || next;
my ($id,$type)=($1,$2);
if (!exists $Friends{$id}) {
$Friends{$id}={
user => undef,
name => undef,
bg => undef,
fg => undef,
dob => undef,
birthday => undef,
groups => [],
groupmask => undef,
status => "active",
type => "user",
};
}
if ($type eq "birthday") {
($v=~/([0-9]+)-([0-9]{2})-([0-9]{2})/o) || next;
my @tm=(0,0,0,$3,$2,$1-1900);
if ($tm[5]>0) {
$Friends{$id}->{dob}=mktime(@tm);
if (!defined $Friends{$id}->{dob}) {
$LJ::Simple::error="Failed to convert time $v into Unix timestamp";
return undef;
}
}
$tm[5]=(gmtime(time()))[5];
$Friends{$id}->{birthday}=mktime(@tm);
if (!defined $Friends{$id}->{birthday}) {
$LJ::Simple::error="Failed to convert time $v into Unix timestamp";
return undef;
}
} else {
$Friends{$id}->{$type}=$v;
}
}
if (defined $self->{groups}) {
my $id=undef;
foreach $id (values %Friends) {
(defined $id->{groupmask}) || next;
foreach (values %{$self->{groups}->{name}}) {
my $bit=1 << $_->{id};
if (($id->{groupmask} & $bit) == $bit) {
push(@{$id->{groups}},$_->{name});
}
}
}
}
my @lst=sort {$a->{user} cmp $b->{user}} (values %Friends);
return ($#lst+1,@lst);
}
=pod
=item $lj->CheckFriends(@groups)
This routine is used to poll the LiveJournal server to see if your friends list
has been updated or not. This routine returns a list. The first item in the
list is a value which holds C<1> if there has been an update
to your friends list and C<0> if not. The second item in the list holds the number
of seconds you must wait before calling C<CheckFriends()> again.
In the event of an error C<undef> is returned in the first item of the list.
The routine can be given an optional list of friends group to check instead of
just looking at all of the friends for the user.
Example code:
while(1) {
my ($new_friends,$next_update)=$lj->CheckFriends();
(defined $new_friends) ||
die "$0: Failed to check friends - $LJ::Simple::error\n";
($new_friends) && print "Friends list updated\n";
sleep($next_update+1);
}
=cut
sub CheckFriends($$) {
my $self=shift;
my (@groups)=@_;
my %Event=();
my %Resp=();
if ($#groups>-1) {
if (!defined $self->{groups}) {
$LJ::Simple::error="Groups not requested at login";
return 0;
}
my $g;
my $mask=0;
foreach $g (@groups) {
if (!exists $self->{groups}->{name}->{$g}) {
$LJ::Simple::error="Group \"$g\" does not exist";
return 0;
}
$mask=$mask | (1 << $self->{groups}->{name}->{$g}->{id});
}
$Event{mask}=$mask;
}
if (exists $self->{checkfriends}) {
$Event{lastupdate}=$self->{checkfriends}->{lastupdate};
my $currtime=time();
if ($currtime<$self->{checkfriends}->{interval}) {
$LJ::Simple::error="Insufficent time left between CheckFriends() call";
return undef;
}
} else {
$self->{checkfriends}={};
}
$self->SendRequest("checkfriends",\%Event,\%Resp) || return undef;
$self->{checkfriends}->{lastupdate}=$Resp{lastupdate};
$self->{checkfriends}->{interval}=time() + $Resp{interval};
return ($Resp{new},$Resp{interval});
}
=pod
=item $lj->GetDayCounts($hash_ref,$journal)
This routine is given a reference to hash which it fills with information
on the journal entries posted to the LiveJournal we are currently associated
with. On success the reference to the hash will be returned. On error
C<undef> is returned.
There is an optional argument - C<$journal> - which can be used to gather this
data for a shared journal the user has access to. If not required then this
value should be C<undef> or an empty string.
The key to the hash is a date, given as seconds since epoch (I<i.e.> C<time_t>)
and the value is the number of entries made on that day. Only dates which have
journal entries made against them will have values in the hash; thus it can be
assumed that if a date is B<not> in the hash then no journal entries were made
on that day.
The hash will be emptied before use.
Example code:
use POSIX;
(defined $lj->GetDayCounts(\%gdc_hr,undef))
|| die "$0: Failed to get day counts - $LJ::Simple::error\n";
foreach (sort {$a<=>$b} keys %gdc_hr) {
printf("%s %03d\n",strftime("%Y/%m/%d",gmtime($_)),$gdc_hr{$_});
}
=cut
sub GetDayCounts($$$) {
my $self=shift;
my ($hr,$journal)=@_;
$LJ::Simple::error="";
if (ref($hr) ne "HASH") {
my $r=ref($hr);
$LJ::Simple::error="CODE: GetDayCounts() given \"$r\", not a hash reference";
return undef;
}
%{$hr}=();
my %Event=();
my %Resp=();
if ((defined $journal) && ($journal ne "")) {
$Event{usejournal}=$journal;
}
$self->SendRequest("getdaycounts",\%Event,\%Resp) || return undef;
my ($k,$v);
while(($k,$v)=each %Resp) {
($k=~/([0-9]+)-([0-9]+)-([0-9]+)/o) || next;
($v==0) && next;
my $timet=mktime(0,0,0,$3,$2-1,$1-1900);
if (!defined $timet) {
$LJ::Simple::error="Failed to convert date $k into Unix timestamp";
return undef;
}
if (exists $hr->{$timet}) {
$hr->{$timet}=$hr->{$timet}+$v;
} else {
$hr->{$timet}=$v;
}
}
return $hr;
}
=pod
=item $lj->GetFriendGroups($hash_ref)
This routine is given a reference to a hash which it fills with information
on the friends groups the user has defined. On success the reference to the
hash will be returned. On error C<undef> is returned.
The hash key is the id number of the friends group as it is possible to
have multiple friends groups with the same name. Each hash value is a hash
reference which points to the following hash:
{
id => Id of the group; used to create permission masks
name => Name of the group
sort => Sort order number from 0 to 255
public => Public group ? 1 for yes, 0 for no
}
The hash given will be emptied before use.
Example code:
my %fg=();
(defined $lj->GetFriendGroups(\%fg)) ||
die "$0: Failed to get groups - $LJ::Simple::error\n";
my $format="| %-4s | %-2s | %-6s | %-40s |\n";
my $line=sprintf($format,"","","","");
$line=~s/\|/+/go;
$line=~s/ /-/go;
print $line;
printf($format,"Sort","Id","Public","Group");
print $line;
foreach (sort {$fg{$a}->{sort}<=>$fg{$b}->{sort}} keys %fg) {
my $hr=$fg{$_};
my $pub="No";
$hr->{public} && ($pub="Yes");
printf($format,$hr->{sort},$hr->{id},$pub,$hr->{name});
}
print $line;
In case you're wondering, the above code outputs something similar to
the following:
+------+----+--------+------------------------------------------+
| Sort | Id | Public | Group |
+------+----+--------+------------------------------------------+
| 5 | 1 | Yes | Good Friends |
| 10 | 2 | No | Communities |
+------+----+--------+------------------------------------------+
=cut
sub GetFriendGroups($$) {
my $self=shift;
my ($hr)=@_;
$LJ::Simple::error="";
if (ref($hr) ne "HASH") {
my $r=ref($hr);
$LJ::Simple::error="CODE: GetFriendGroups() given \"$r\", not a hash reference";
return undef;
}
%{$hr}=();
my %Event=();
my %Resp=();
$self->SendRequest("getfriendgroups",\%Event,\%Resp) || return undef;
my ($k,$v);
while(($k,$v)=each %Resp) {
$k=~/^frgrp_([0-9]+)_(.*)$/o || next;
my ($id,$name)=($1,$2);
if (!exists $hr->{$id}) {
$hr->{$id}={
id => $id,
public => 0,
};
}
($name eq "sortorder") && ($name="sort");
$hr->{$id}->{$name}=$v;
}
return $hr;
}
=pod
=item $lj->CreateFriendGroups($hash_ref)
This routine modifies user friend groups according to $hash_ref
=cut
sub CreateFriendGroups($$) {
my $self=shift;
my ($hr)=@_;
$LJ::Simple::error="";
if (ref($hr) ne "HASH") {
my $r=ref($hr);
$LJ::Simple::error="CODE: CreateFriendGroups() given \"$r\", not a hash reference";
return undef;
}
my %Event=();
my %Resp=();
my $i = 0;
foreach my $grpid (keys %{$hr}) {
if ($grpid > 0 && $grpid < 31) {
my $pname = "efg_set_" . $grpid . "_name";
$Event{$pname} = $hr->{$grpid}->{name};
$i++;
}
}
$self->SendRequest("editfriendgroups",\%Event,\%Resp) || return undef;
return $i;
}
=pod
=back
=head2 The creation and editing of entries
=over 4
=item $lj->NewEntry($event)
Prepares for a new journal entry to be sent into the LiveJournal system.
Takes a reference to a hash which will be emptied and prepared for use
by the other routines used to prepare a journal entry for posting.
On success returns C<1>, on failure returns C<0>
Example code:
my %Entry=();
$lj->NewEntry(\%Entry)
|| die "$0: Failed to prepare new post - $LJ::Simple::error\n";
=cut
sub NewEntry($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
## Build the event hash - put defaults in
my $ltime=time();
my @ltime=localtime($ltime);
%{$event}=(
__new_entry => 1,
event => undef,
lineenddings => "unix",
subject => undef,
year => $ltime[5]+1900,
mon => $ltime[4]+1,
day => $ltime[3],
hour => $ltime[2],
min => $ltime[1],
__timet => $ltime,
);
return 1;
}
=pod
=item $lj->SetDate($event,$time_t)
Sets the date for the event being built from the given C<time_t> (i.e. seconds
since epoch) value. Bare in mind that you may need to call
C<$lj-E<gt>Setprop_backdate(\%Event,1)> to backdate the journal entry if the journal being
posted to has events more recent than the date being set here. Returns C<1> on
success, C<0> on failure.
If the value given for C<time_t> is C<undef> then the current time is used.
If the value given for C<time_t> is negative then it is taken to be relative
to the current time, i.e. a value of C<-3600> is an hour earlier than the
current time.
Note that C<localtime()> is called to convert the C<time_t> value into
the year, month, day, hours and minute values required by LiveJournal.
Thus the time given to LiveJournal will be the local time as shown on
the machine the code is running on.
Example code:
## Set date to current time
$lj->SetDate(\%Event,undef)
|| die "$0: Failed to set date of entry - $LJ::Simple::error\n";
## Set date to Wed Aug 14 11:56:42 2002 GMT
$lj->SetDate(\%Event,1029326202)
|| die "$0: Failed to set date of entry - $LJ::Simple::error\n";
## Set date to an hour ago
$lj->SetDate(\%Event,-3600)
|| die "$0: Failed to set date of entry - $LJ::Simple::error\n";
=cut
sub SetDate($$$) {
my $self=shift;
my ($event,$timet)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
(defined $timet) || ($timet=time());
if ($timet<0) {
$timet=time() + $timet;
}
my @ltime=localtime($timet);
$event->{__timet}=$timet;
$event->{year}=$ltime[5]+1900;
$event->{mon}=$ltime[4]+1;
$event->{day}=$ltime[3];
$event->{hour}=$ltime[2];
$event->{min}=$ltime[1];
return 1;
}
=pod
=item $lj->SetMood($event,$mood)
Given a mood this routine sets the mood for the journal entry. Unlike the
more direct C<$lj-E<gt>Setprop_current_mood()> and C<$lj-E<gt>Setprop_current_mood_id(\%Event,)>
routines, this routine will attempt to first attempt to find the mood given
to it in the mood list returned by the LiveJournal server. If it is unable to
find a suitable mood then it uses the text given.
Note that if the LiveJournal
object was created with either C<moods> set to C<0> or
with C<fast> set to C<1> then this function will not attempt to find the
mood name given in C<$mood> in the mood list.
Returns C<1> on success, C<0> otherwise.
Example code:
$lj->SetMood(\%Event,"happy")
|| die "$0: Failed to set mood - $LJ::Simple::error\n";
=cut
sub SetMood($$$) {
my $self=shift;
my ($event,$mood) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
## Simple opt - none of the mood names have a space in them
if (($mood!~/\s/)&&(defined $self->{mood_map})) {
my $lc_mood=lc($mood);
if (exists $self->{mood_map}->{$lc_mood}) {
return $self->Setprop_current_mood_id($event,$self->{mood_map}->{$lc_mood})
}
}
return $self->Setprop_current_mood($event,$mood);
}
=pod
=item $lj->UseJournal($event,$journal)
The journal entry will be posted into the shared journal given
as an argument rather than the default journal for the user.
Returns C<1> on success, C<0> otherwise.
Example code:
$lj->UseJournal(\%Event,"some_community")
|| die "$0: Failed to - $LJ::Simple::error\n";
=cut
sub UseJournal($$$) {
my $self=shift;
my ($event,$journal) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if ((defined $self->{access})&&(!exists $self->{access}->{$journal})) {
$LJ::Simple::error="user unable to post to journal \"$journal\"";
return 0;
}
$event->{usejournal}=$journal;
return 1;
}
=pod
=item $lj->SetSubject($event,$subject)
Sets the subject for the journal entry. The subject has the following
limitations:
o Limited to a length of 255 characters
o No newlines are allowed
Returns C<1> on success, C<0> otherwise.
Example code:
$lj->SetSubject(\%Event,"Some subject")
|| die "$0: Failed to set subject - $LJ::Simple::error\n";
=cut
sub SetSubject($$$) {
my $self=shift;
my ($event,$subject) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if ($subject) {
if (length($subject)>255) {
my $len=length($subject);
$LJ::Simple::error="Subject length limited to 255 characters [given $len]";
return 0;
}
if ($subject=~/[\r\n]/) {
$LJ::Simple::error="New lines not allowed in subject";
return 0;
}
$event->{subject}=$subject;
}
else {
$event->{subject}="";
}
return 1;
}
=pod
=item $lj->SetEntry($event,@entry)
Sets the entry for the journal; takes a list of strings. It should be noted
that this list will be C<join()>ed together with a newline between each
list entry.
If the list is null or C<undef> then any existing entry is removed.
Returns C<1> on success, C<0> otherwise.
Example code:
# Single line entry
$lj->SetEntry(\%Event,"Just a simple entry")
|| die "$0: Failed to set entry - $LJ::Simple::error\n";
# Three lines of text
my @stuff=(
"Line 1",
"Line 2",
"Line 3",
);
$lj->SetEntry(\%Event,@stuff)
|| die "$0: Failed to set entry - $LJ::Simple::error\n";
# Clear the entry
$lj->SetEntry(\%Event,undef)
|| die "$0: Failed to set entry - $LJ::Simple::error\n";
$lj->SetEntry(\%Event)
|| die "$0: Failed to set entry - $LJ::Simple::error\n";
=cut
sub SetEntry($$@) {
my $self=shift;
my ($event,@entry) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if ((!defined $entry[0]) || ($#entry == -1)) {
$event->{event}=undef;
} else {
$event->{event}=join("\n",@entry);
}
return 1;
}
=pod
=item $lj->AddToEntry($event,@entry)
Adds a string to the existing journal entry being worked on. The new data
will be appended to the existing entry with a newline separating them.
It should be noted that as with C<$lj-E<gt>SetEntry()> the list given to
this routine will be C<join()>ed together with a newline between each
list entry.
If C<$lj-E<gt>SetEntry()> has not been called then C<$lj-E<gt>AddToEntry()> acts
in the same way as C<$lj-E<gt>SetEntry()>.
If C<$lj-E<gt>SetEntry()> has already been called then calling C<$lj-E<gt>AddToEntry()>
with a null list or a list which starts with C<undef> is a NOP.
Returns C<1> on success, C<0> otherwise.
Example code:
# Single line entry
$lj->AddToEntry(\%Event,"Some more text")
|| die "$0: Failed to set entry - $LJ::Simple::error\n";
# Three lines of text
my @stuff=(
"Line 5",
"Line 6",
"Line 7",
);
$lj->AddToEntry(\%Event,@stuff)
|| die "$0: Failed to set entry - $LJ::Simple::error\n";
=cut
sub AddToEntry($$@) {
my $self=shift;
my ($event,@entry) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if (!defined $event->{event}) {
if ((!defined $entry[0]) || ($#entry == -1)) {
$event->{event}=undef;
} else {
$event->{event}=join("\n",@entry);
}
} else {
if ((!defined $entry[0]) || ($#entry == -1)) {
return 1;
}
$event->{event}=join("\n",$event->{event},@entry);
}
return 1;
}
=pod
=back
=head2 Setting of journal entry security levels
=over 4
=item $lj->SetProtect($event,$type,@args)
A wrapper function which calls the underlying C<SetProtect*()> routines
for the caller. This takes two or more arguments; the first argument is
the hash reference of the current event. The second argument is the
type of security we are setting. Subsequent arguments are related to
the security type. Available types and their arguments are:
+---------+------------------+------------------------------------+
| Type | Additional args | Security |
+---------+------------------+------------------------------------+
| public | None | Public - the default |
| friends | None | Friends only |
| groups | A list of groups | Restricted to groups of friends |
| private | None | Private - only the user can access |
+---------+------------------+------------------------------------+
On success this routine returns C<1>; otherwise it returns C<0> and
sets C<$LJ::Simple::error> to the reason why.
Example code:
## Make entry public (the default)
$lj->SetProtect(\%Event,"public")
|| die "$0: Failed to make entry public - $LJ::Simple::error\n";
## Make entry friends only
$lj->SetProtect(\%Event,"friends")
|| die "$0: Failed to make entry friends only - $LJ::Simple::error\n";
## Make entry only readable by friends in the groups "close" and "others"
$lj->SetProtect(\%Event,"groups","close","others")
|| die "$0: Failed to make entry public - $LJ::Simple::error\n";
## Make entry private so only the journal owner can view it
$lj->SetProtect(\%Event,"private")
|| die "$0: Failed to make entry private - $LJ::Simple::error\n";
=cut
sub SetProtect($$$@) {
my $self=shift;
my ($event,$type,@args)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if ($type eq "public") {
return $self->SetProtectPublic($event);
} elsif ($type eq "friends") {
return $self->SetProtectFriends($event);
} elsif ($type eq "groups") {
return $self->SetProtectGroups($event,@args);
} elsif ($type eq "private") {
return $self->SetProtectPrivate($event);
} else {
$LJ::Simple::error="CODE: type \"$type\" not recognised by SetProtect()";
return 0;
}
};
=pod
=item $lj->SetProtectPublic($event)
Sets the current post so that anyone can read the journal entry. Note that this
is the default for a new post created by C<LJ::Simple> - this method is most
useful when working with an existing post. Returns C<1> on success, C<0>
otherwise.
Example code:
$lj->SetProtectPublic(\%Event)
|| die "$0: Failed to make entry public - $LJ::Simple::error\n";
=cut
sub SetProtectPublic($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
(exists $event->{security}) && delete $event->{security};
(exists $event->{allowmask}) && delete $event->{allowmask};
return 1;
}
=pod
=pod
=item $lj->SetProtectFriends($event)
Sets the current post so that only friends can read the journal entry. Returns
C<1> on success, C<0> otherwise.
Example code:
$lj->SetProtectFriends(\%Event)
|| die "$0: Failed to protect via friends - $LJ::Simple::error\n";
=cut
sub SetProtectFriends($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
$event->{security}="usemask";
$event->{allowmask}=1;
return 1;
}
=pod
=item $lj->SetProtectGroups($event,$group1, $group2, ... $groupN)
Takes a list of group names and sets the current entry so that only those
groups can read the journal entry. Returns
C<1> on success, C<0> otherwise.
Example code:
$lj->SetProtectGroups(\%Event,"foo","bar")
|| die "$0: Failed to protect via group - $LJ::Simple::error\n";
=cut
sub SetProtectGroups($$@) {
my $self=shift;
my ($event,@grps) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if (!defined $self->{groups}) {
$LJ::Simple::error="Groups not requested at login";
return 0;
}
if ($#grps==-1) {
$LJ::Simple::error="No group names given";
return 0;
}
$event->{security}="usemask";
my $g;
my $mask=0;
foreach $g (@grps) {
if (!exists $self->{groups}->{name}->{$g}) {
$LJ::Simple::error="Group \"$g\" does not exist";
return 0;
}
$mask=$mask | (1 << $self->{groups}->{name}->{$g}->{id});
}
$event->{allowmask}=$mask;
return 1;
}
=pod
=item $lj->SetProtectPrivate($event)
Sets the current post so that the owner of the journal only can read the
journal entry. Returns C<1> on success, C<0> otherwise.
Example code:
$lj->SetProtectPrivate(\%Event)
|| die "$0: Failed to protect via private - $LJ::Simple::error\n";
=cut
sub SetProtectPrivate($$) {
my $self=shift;
my ($event) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
$event->{security}="private";
(exists $event->{allowmask}) &&
delete $event->{allowmask};
return 1;
}
##
## Helper function used to set meta data
##
sub Setprop_general($$$$$$) {
my ($self,$event,$prop,$caller,$type,$data)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
my $nd=undef;
if ($type eq "bool") {
if (($data == 1)||($data == 0)) {
$nd=$data;
} else {
$LJ::Simple::error="INTERNAL: Invalid value [$data] for type bool [from $caller]";
return 0;
}
} elsif ($type eq "char") {
$nd=$data;
} elsif ($type eq "num") {
if ($data!~/^[0-9]+$/o) {
$LJ::Simple::error="INTERNAL: Invalid value [$data] for type num [from $caller]";
return 0;
}
$nd=$data;
} else {
$LJ::Simple::error="INTERNAL: Unknown type \"$type\" [from $caller]";
return 0;
}
if (!defined $nd) {
$LJ::Simple::error="INTERNAL: Setprop_general did not set \$nd [from $caller]";
return 0;
}
$event->{"prop_$prop"}=$nd;
return 1;
}
=pod
=back
=head2 Setting journal entry properties
=over 4
=item $lj->Setprop_taglist($event,@tags)
Set the tags for the entry; C<@tags> is a list of the tags to give the
entry.
Example code:
$lj->Setprop_taglist(\%Event,qw( gabe pets whatever )) ||
die "$0: Failed to set back date property - $LJ::Simple::error\n";
=cut
sub Setprop_taglist($$@) {
my ($self,$event,@tags)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"taglist","Setprop_taglist","char",join(", ",@tags));
}
=pod
=item $lj->Setprop_backdate($event,$onoff)
Used to indicate if the journal entry being written should be back dated or not. Back dated
entries do not appear on the friends view of your journal entries. The C<$onoff>
value takes either C<1> for switching the property on or C<0> for switching the
property off. Returns C<1> on success, C<0> on failure.
You will need to set this value if the journal entry you are sending has a
date earlier than other entries in your journal.
Example code:
$lj->Setprop_backdate(\%Event,1) ||
die "$0: Failed to set back date property - $LJ::Simple::error\n";
=cut
sub Setprop_backdate($$$) {
my ($self,$event,$onoff)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"opt_backdated","Setprop_backdate","bool",$onoff);
}
=pod
=item $lj->Setprop_current_mood($event,$mood)
Used to set the current mood for the journal being written. This takes a string which
describes the mood.
It is better to use C<$lj-E<gt>SetMood()> as that will automatically use a
mood known to the LiveJournal server if it can.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_current_mood(\%Event,"Happy, but tired") ||
die "$0: Failed to set current_mood property - $LJ::Simple::error\n";
=cut
sub Setprop_current_mood($$$) {
my ($self,$event,$mood)=@_;
$LJ::Simple::error="";
if ($mood=~/[\r\n]/) {
$LJ::Simple::error="Mood may not contain a new line";
return 0;
}
return $self->Setprop_general($event,"current_mood","Setprop_current_mood","char",$mood);
}
=pod
=item $lj->Setprop_current_mood_id($event,$id)
Used to set the current mood_id for the journal being written. This takes a number which
refers to a mood_id the LiveJournal server knows about.
Note that if the LiveJournal
object was created with either C<moods> set to C<0> or
with C<fast> set to C<1> then this function will not attempt to validate
the C<mood_id> given to it.
It is better to use C<$lj-E<gt>SetMood()> as that will automatically use a
mood known to the LiveJournal server if it can.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_current_mood_id(\%Event,15) ||
die "$0: Failed to set current_mood_id property - $LJ::Simple::error\n";
=cut
sub Setprop_current_mood_id($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
if (defined $self->{moods}) {
if (!exists $self->{moods}->{$data}) {
$LJ::Simple::error="The mood_id $data is not known by the LiveJournal server";
return 0;
}
}
return $self->Setprop_general($event,"current_moodid","Setprop_current_mood_id","num",$data);
}
=pod
=item $lj->Setprop_current_music($event,$music)
Used to set the current music for the journal entry being written. This takes
a string.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_current_music(\%Event,"Collected euphoric dance") ||
die "$0: Failed to set current_music property - $LJ::Simple::error\n";
=cut
sub Setprop_current_music($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"current_music","Setprop_current_music","char",$data);
}
=pod
=item $lj->Setprop_preformatted($event,$onoff)
Used to set if the text for the journal entry being written is preformatted in HTML
or not. This takes a boolean value of C<1> for true and C<0> for false.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_preformatted(\%Event,1) ||
die "$0: Failed to set property - $LJ::Simple::error\n";
=cut
sub Setprop_preformatted($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"opt_preformatted","Setprop_preformatted","bool",$data);
}
=pod
=item $lj->Setprop_nocomments($event,$onoff)
Used to set if the journal entry being written can be commented on or not. This takes
a boolean value of C<1> for true and C<0> for false. Thus if you use a value
of C<1> (true) then comments will not be allowed.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_nocomments(\%Event,1) ||
die "$0: Failed to set property - $LJ::Simple::error\n";
=cut
sub Setprop_nocomments($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"opt_nocomments","Setprop_nocomments","bool",$data);
}
=pod
=item $lj->Setprop_picture_keyword($event,$keyword)
Used to set the picture keyword for the journal entry being written. This takes
a string. We check to make sure that the picture keyword exists.
Note that if the LiveJournal
object was created with either C<pics> set to C<0> or
with C<fast> set to C<1> then this function will B<not> validate
the picture keyword before setting it.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_picture_keyword(\%Event,"Some photo") ||
die "$0: Failed to set property - $LJ::Simple::error\n";
=cut
sub Setprop_picture_keyword($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
if (defined $self->{pictures}) {
my $match=0;
foreach (values %{$self->{pictures}}) {
if ($_->{name} eq $data) {
$match=1;
last;
}
}
if (!$match) {
$LJ::Simple::error="Picture keyword not associated with journal";
return 0;
}
}
return $self->Setprop_general($event,"picture_keyword","Setprop_picture_keyword","char",$data);
}
=pod
=item $lj->Setprop_noemail($event,$onoff)
Used to say that comments on the journal entry being written should not be emailed.
This takes boolean value of C<1> for true and C<0> for false.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_noemail(\%Event,1) ||
die "$0: Failed to set property - $LJ::Simple::error\n";
=cut
sub Setprop_noemail($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"opt_noemail","Setprop_noemail","bool",$data);
}
=pod
=item $lj->Setprop_unknown8bit($event,$onoff)
Used say that there is 8-bit data which is not in UTF-8 in the journal entry
being written. This takes a boolean value of C<1> for true and C<0> for false.
Returns C<1> on success, C<0> on failure.
Example code:
$lj->Setprop_unknown8bit(\%Event,1) ||
die "$0: Failed to set property - $LJ::Simple::error\n";
=cut
sub Setprop_unknown8bit($$$) {
my ($self,$event,$data)=@_;
$LJ::Simple::error="";
return $self->Setprop_general($event,"unknown8bit","Setprop_unknown8bit","bool",$data);
}
=pod
=back
=head2 Posting, editing and deleting journal entries
=over 4
=item $lj->PostEntry($event)
Submit a journal entry into the LiveJournal system. This requires you to have
set up the journal entry with C<$lj-E<gt>NewEntry()> and to have at least called
C<$lj-E<gt>SetEntry()>.
On success a list containing the following is returned:
o The item_id as returned by the LiveJournal server
o The anum as returned by the LiveJournal server
o The item_id of the posted entry as used in HTML - that is the
value of C<($item_id * 256) + $anum)>
On failure C<undef> is returned.
# Build the new entry
my %Event;
$lj->NewEntry(\%Event) ||
die "$0: Failed to create new journal entry - $LJ::Simple::error\n";
# Set the journal entry
$lj->SetEntry(\%Event,"foo") ||
die "$0: Failed set journal entry - $LJ::Simple::error\n";
# And post it
my ($item_id,$anum,$html_id)=$lj->PostEntry(\%Event);
defined $item_id ||
die "$0: Failed to submit new journal entry - $LJ::Simple::error\n";
=cut
##
## PostEntry - actually submit a journal entry.
##
sub PostEntry($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if (!exists $event->{"__new_entry"}) {
$LJ::Simple::error="CODE: NewEntry not called";
return undef;
}
## Blat any key in $event which starts with a double underscore
map {/^__/ && delete $event->{$_}} (keys %{$event});
if (!defined $event->{event}) {
$LJ::Simple::error="CODE: No journal entry set - call SetEntry() or AddToEntry() first";
return undef;
}
## Blat any entry in $self->{event} with an undef value
map {defined $event->{$_} || delete $event->{$_}} (keys %{$event});
## Finally send the actual request
my %Resp=();
$self->SendRequest("postevent",$event,\%Resp) || return undef;
if (!exists $Resp{itemid}) {
$LJ::Simple::error="LJ server did not return itemid";
return undef;
}
if (!exists $Resp{anum}) {
$LJ::Simple::error="LJ server did not return anum";
return undef;
}
return ($Resp{itemid},$Resp{anum},($Resp{itemid} * 256) + $Resp{anum});
}
=pod
=item $lj->EditEntry($event)
Edit an entry from the LiveJournal system which has the givem C<item_id>.
The entry should have been fetched from LiveJournal using the
C<$lj-E<gt>GetEntries()> function and then adjusted using the various
C<$lj-E<gt>Set...()> functions.
It should be noted that this function can be used to delete a journal entry
by setting the entry to a blank string, I<i.e.> by using
C<$lj-E<gt>SetEntry(\%Event,undef)>
Returns C<1> on success, C<0> on failure.
Example:
# Fetch the most recent event
my %Events = ();
(defined $lj->GetEntries(\%Events,undef,"one",-1)) ||
die "$0: Failed to get entries - $LJ::Simple::error\n";
# Mark it as private
foreach (values %Entries) {
$lj->SetProtectPrivate($_);
$lj->EditEntry($_) ||
die "$0: Failed to edit entry - $LJ::Simple::error\n";
}
# Alternatively we could just delete it...
my $event=(values %Entries)[0];
$lj->SetEntry($event,undef);
$lj->EditEntry($event) ||
die "$0: Failed to edit entry - $LJ::Simple::error\n";
=cut
sub EditEntry($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return 0;
}
if (!exists $event->{"__itemid"}) {
$LJ::Simple::error="CODE: Not an existing entry; use GetEntry()";
return 0;
}
$event->{itemid}=$event->{"__itemid"};
## Blat any key in $event which starts with a double underscore
map {/^__/ && delete $event->{$_}} (keys %{$event});
if (!defined $event->{event}) {
$LJ::Simple::error="CODE: No journal entry set";
return 0;
}
## Blat any entry in $event with an undef value
map {defined $event->{$_} || delete $event->{$_}} (keys %{$event});
## Make the request
return $self->SendRequest("editevent",$event,undef);
}
=pod
=item $lj->DeleteEntry($item_id)
Delete an entry from the LiveJournal system which has the given C<item_id>.
On success C<1> is returned; on failure C<0> is returned.
Example:
$lj->DeleteEntry($some_item_id) ||
die "$0: Failed to delete journal entry - $LJ::Simple::error\n";
=cut
sub DeleteEntry($$) {
my $self=shift;
my ($item_id) = @_;
$LJ::Simple::error="";
if (!defined $item_id) {
$LJ::Simple::error="CODE: DeleteEntry() given undefined item_id";
return 0;
}
if ($item_id!~/^[0-9]+$/) {
$LJ::Simple::error="CODE: DeleteEntry() given invalid item_id";
return 0;
}
my %Event=(
itemid => $item_id,
event => "",
);
return $self->SendRequest("editevent",\%Event,undef);
}
=pod
=back
=head2 Retriving journal entries
=over 4
=item $lj->SyncItems($timestamp)
This routine returns a list of all of the items (journal entries, to-do items,
comments) which have been created or updated on LiveJournal. There is an optional
timestamp value for specifying the time you last synchronised with the server.
This timestamp value can either be a Unix-style C<time_t> value or a previously
returned timestamp from this routine. If not used specify the undefined value
C<undef>.
When specifying the time you must take into account the fact that the modification
or creation times of the entries in the LiveJournal database are stored as the
time local to the computer running the database rather than GMT. Due to this
it is safest to use the time from the latest item downloaded from the LiveJournal
from a previous C<SyncItems()> call.
On success this routine will return a list which contains first the number of
valid items in the list and then a list of hashes which contain the details
of the items found. This routine can return an empty list which signifies that
no new items could be found. On failure C<undef> is returned.
The format of the returned list is as follows. The list of hashes is ordered
by the timestamps of the entries, oldest to newest.
@list = (
number of items returned,
{
item_id => Item_id of the entry changed
type => Type of entry
action => What happened to the entry
time_t => Time of change in Unix time (see note below)
timestamp => Timestamp from server
},
);
The C<type> of entry can be one of the following letters:
L: Journal entries
C: Comments
T: To-do items
It should be noted that currently the LiveJournal system will only ever
return C<L> types due to the C<C> and C<T> types not having been implemented
in the LiveJournal code yet.
The C<action> of the entry can be either C<create> for a new entry,
C<update> for an entry which has been modified or C<del> for a deleted entry.
The C<time_t> value is probably going to be wrong; as far as the author of
this code can tell, you can not get the timezone of the server which is
serving out the request. This means that converting the timestamps
returned by the server from their format of C<YYYY-MM-DD hh:mm:ss> into
a Unix C<time_t> value is inaccurate at best since C<time_t> is defined
as the number of seconds since 00:00 1st January 1970 B<GMT>. Functions
like C<mktime()> which can be used to create C<time_t> values have to
assume that the data they are being given is valid for the timezone the
machine it is running on is actually in. Given the nature of the net
this is rarely the case. I<sigh> I wish that the LJ developers had stored
timestamps in pure C<time_t> in the database... and if they have done they
should provide a way for developers to get access to this as its B<much>
more useful IMHO.
Given the above you're probably wondering why I included the C<time_t>
value. Well, whilst the value isn't much use when it really comes down
to it, it B<is> useful when it comes to sorting the list of entries as
all of the entries from the same server will be inaccurate to the same
amount.
The C<timestamp> from server takes the format of C<YYYY-MM-DD hh:mm:ss>
It should be noted that this routine can take a long time to return
if there are large numbers of entries to be returned. This is especially
true if you give C<undef> as the timestamp.
Example code:
# All entries in the last day or so; this is fudged due to timezone
# differences (WTF didn't they store stuff in GMT ?)
my ($num_of_items,@lst)=$lj->SyncItems(time() - (86400 * 2));
(defined $num_of_items) ||
die "$0: Failed to sync - $LJ::Simple::error\n";
my $hr=undef;
print "Number of items: $num_of_items\n";
print "Item_id\tType\tAction\tTime_t\t\tTimestamp\n";
foreach $hr (@lst) {
print "$hr->{item_id}\t" .
"$hr->{type}\t" .
"$hr->{action}\t" .
"$hr->{time_t}\t" .
"$hr->{timestamp}\n";
}
There is also an example of how to work with all of the entries of a LiveJournal
shown in the C<examples/friends-only> script which accompanies the C<LJ::Simple>
distribution. This example script looks at a LiveJournal and makes sure that every
journal entry is at the very least marked as being friends-only.
=cut
sub SyncItems($$) {
my $self=shift;
my ($timet)=@_;
$LJ::Simple::error="";
if ($LJ::Simple::debug) {
my $ts=undef;
if (defined $timet) {
$ts="\"$timet\"";
} else {
$ts="undef";
}
Debug "SyncItems($ts)";
}
my %Event=();
my %Resp=();
if (defined $timet) {
if ($timet=~/^[0-9]+$/) {
my @tm=gmtime($timet);
if ($#tm==-1) {
$LJ::Simple::error="CODE: Invalid timestamp";
return undef;
}
$Event{lastsync}=strftime("%Y-%m-%d %H:%M:%S",@tm);
} else {
$Event{lastsync}=$timet;
}
}
$self->SendRequest("syncitems",\%Event,\%Resp) || return undef;
my %Mh=();
my $sync_count;
my $sync_total;
my $latest=0;
my $latest_ts;
my ($key,$val);
while(($key,$val)=each %Resp) {
if ($key=~/sync_([0-9]+)_(.*)$/o) {
my ($id,$name)=($1,$2);
(exists $Mh{$id}) || ($Mh{$id}={});
if ($name eq "item") {
my ($type,$item_id)=split(/-/,$val,2);
$Mh{$id}->{item_id}=$item_id;
$Mh{$id}->{type}=$type;
} elsif ($name eq "action") {
$Mh{$id}->{action}=$val;
} elsif ($name eq "time") {
$Mh{$id}->{timestamp}=$val;
if ($val!~/([0-9]+)-([0-9]+)-([0-9]+)\s([0-9]+):([0-9]+):([0-9]+)/io) {
$LJ::Simple::error="INTERNAL: failed to parse timestamp \"$val\"";
return undef;
}
$Mh{$id}->{time_t}=mktime($6,$5,$4,$3,$2-1,$1-1900,0,0,0);
if (!defined $Mh{$id}->{time_t}) {
$LJ::Simple::error="INTERNAL: failed to create time_t from \"$val\"";
return undef;
}
if ($Mh{$id}->{time_t}>$latest) {
$latest_ts=$val;
$latest=$Mh{$id}->{time_t};
}
} else {
$LJ::Simple::error="INTERNAL: Unrecognised sync_[0-9]_* \"$key\"";
return undef;
}
} elsif ($key eq "sync_total") {
$sync_total=$val;
} elsif ($key eq "sync_count") {
$sync_count=$val;
}
}
Debug "sync_count=$sync_count\n";
Debug "sync_total=$sync_total\n";
my @lst=();
push(@lst,values %Mh);
if ($sync_count != $sync_total) {
my ($num,@nl)=$self->SyncItems($latest_ts);
(defined $num) || return undef;
push(@lst,@nl);
}
@lst=sort { $a->{time_t} <=> $b->{time_t} } @lst;
map { $_->{kv}=join(":",$_->{item_id},$_->{type},$_->{action},$_->{time_t}) } @lst;
my %seen=();
@lst=grep((!exists $seen{$_->{kv}}) && ($seen{$_->{kv}}=1),@lst);
my $tot=$#lst+1;
return ($tot,@lst);
}
=pod
=item $lj->GetEntries($hash_ref,$journal,$type,@opt)
This routine allows you to pull events from the user's LiveJournal. There are
several different ways this routine can work depending on the value given in
the C<$type> argument.
This routine will currently only allow you to get a B<maximum of 50 journal entries>
thanks to restrictions imposed by LiveJournal servers. If you want to perform work
on I<every> journal entry within a LiveJournal account then you should look at the
C<SyncItems()> routine documented above.
The first argument - C<$hash_ref> is a reference to a hash which will be filled
with the details of the journal entries downloaded. The key to this hash is the
C<item_id> of the journal entries. The value is a hash reference which points to
a hash of the same type created by C<NewPost()> and used by C<PostEntry()> and
C<EditEntry()>. The most sensible way to access this hash is to use the various
C<Get*()> routines.
The second argument - C<$journal> - is an optional argument set if the journal
to be accessed is a shared journal. If this is set then the name of shared journal
will be propogated into the entries returned in the hash reference C<$hash_ref> as
if C<$lj->UseJournal($event,$journal)> was called. If not required set this to C<undef>.
The third argument - C<$type> - specifies how the journal entries are to be
pulled down. The contents of the fourth argument - C<@opt> - will depend on the
value in the C<$type> variable. Thus:
+-------+------------+------------------------------------------+
| $type | @opt | Comments |
+-------+------------+------------------------------------------+
| day | $timestamp | Download a single day. $timestamp is a |
| | | Unix timestamp for the required day |
+-------+------------+------------------------------------------+
| lastn |$num,$before| Download a number of entries. $num has a |
| | | maximum value of 50. If $num is undef |
| | | then the default of 20 is used. $before |
| | | is an optional value which specifies a |
| | | date before which all entries must occur.|
| | | The date is specified as a Unix |
| | | timestamp. If not specified the value |
| | | should be undef. |
+-------+------------+------------------------------------------+
| one | $item_id | The unique ItemID for the entry to be |
| | | downloaded. A value of -1 means to |
| | | download the most recent entry |
+-------+------------+------------------------------------------+
| sync | $date | Get journal entries since the given date.|
| | | The date should be specified as a Unix |
| | | timestamp. |
+-------+------------+------------------------------------------+
If the operation is successful then C<$hash_ref> is returned. On failure
C<undef> is returned and C<$LJ::Simple::error> is updated with the
reason for the error.
Example code:
The following code only uses a single C<$type> from the above list; C<one>.
However the hash of hashes returned is the same in every C<$type> used. The
code below shows how to pull down the last journal entry posted and then uses
all of the various C<Get*()> routines to decode the hash returned.
use POSIX;
my %Entries=();
(defined $lj->GetEntries(\%Entries,undef,"one",-1)) ||
die "$0: Failed to get entries - $LJ::Simple::error\n";
my $Entry=undef;
my $Format="%-20s: %s\n";
foreach $Entry (values %Entries) {
# Get URL
my $url=$lj->GetURL($Entry);
(defined $url) && print "$url\n";
# Get ItemId
my ($item_id,$anum,$html_id)=$lj->GetItemId($Entry);
(defined $item_id) && printf($Format,"Item_id",$item_id);
# Get the subject
my $subj=$lj->GetSubject($Entry);
(defined $subj) && printf($Format,"Subject",$subj);
# Get the date entry was posted
my $timet=$lj->GetDate($Entry);
if (defined $timet) {
printf($Format,"Date",
strftime("%Y-%m-%d %H:%M:%S",gmtime($timet)));
}
# Is entry protected ?
my $EntProt="";
my ($protect,@prot_opt)=$lj->GetProtect($Entry);
if (defined $protect) {
if ($protect eq "public") {
$EntProt="public";
} elsif ($protect eq "friends") {
$EntProt="friends only";
} elsif ($protect eq "groups") {
$EntProt=join("","only groups - ",join(", ",@prot_opt));
} elsif ($protect eq "private") {
$EntProt="private";
}
printf($Format,"Journal access",$EntProt);
}
## Properties
# Backdated ?
my $word="no";
my $prop=$lj->Getprop_backdate($Entry);
if ((defined $prop) && ($prop==1)) { $word="yes" }
printf($Format,"Backdated",$word);
# Preformatted ?
$word="no";
$prop=$lj->Getprop_preformatted($Entry);
if ((defined $prop) && ($prop==1)) { $word="yes" }
printf($Format,"Preformatted",$word);
# No comments allowed ?
$word="no";
$prop=$lj->Getprop_nocomments($Entry);
if ((defined $prop) && ($prop==1)) { $word="yes" }
printf($Format,"No comments",$word);
# Do not email comments ?
$word="no";
$prop=$lj->Getprop_noemail($Entry);
if ((defined $prop) && ($prop==1)) { $word="yes" }
printf($Format,"No emailed comments",$word);
# Unknown 8-bit ?
$word="no";
$prop=$lj->Getprop_unknown8bit($Entry);
if ((defined $prop) && ($prop==1)) { $word="yes" }
printf($Format,"Any 8 bit, non UTF-8",$word);
# Current music
$word="[None]";
$prop=$lj->Getprop_current_music($Entry);
if ((defined $prop) && ($prop ne "")) { $word=$prop }
printf($Format,"Current music",$word);
# Current mood [text]
$word="[None]";
$prop=$lj->Getprop_current_mood($Entry);
if ((defined $prop) && ($prop ne "")) { $word=$prop }
printf($Format,"Current mood",$word);
# Current mood [id]
$word="[None]";
$prop=$lj->Getprop_current_mood_id($Entry);
if ((defined $prop) && ($prop ne "")) { $word=$prop }
printf($Format,"Current mood_id",$word);
# Picture keyword
$word="[None]";
$prop=$lj->Getprop_picture_keyword($Entry);
if ((defined $prop) && ($prop ne "")) { $word=$prop }
printf($Format,"Picture keyword",$word);
# Finally output the actual journal entry
printf($Format,"Journal entry","");
my $text=$lj->GetEntry($Entry);
(defined $text) &&
print " ",join("\n ",split(/\n/,$text)),"\n\n";
}
=cut
sub GetEntries($$@) {
my $self=shift;
my ($hr,$journal,$type,@opts)=@_;
$LJ::Simple::error="";
if (ref($hr) ne "HASH") {
$LJ::Simple::error="CODE: GetEntries() not given a hash reference";
return undef;
}
%{$hr}=();
my %Event=();
my %Resp=();
if (defined $journal) {
$Event{usejournal}=$journal;
}
my $ctype=lc($type);
if ($ctype eq "day") {
if ($#opts<0) {
$LJ::Simple::error="CODE: GetEntries($type) requires year,month,day in \@opts";
return undef;
}
my ($timestamp)=@opts;
if ($timestamp!~/^[0-9]+$/) {
$LJ::Simple::error="CODE: GetEntries($type) given invalid timestamp";
return undef;
}
my @tm=localtime($timestamp);
if ($#tm==-1) {
$LJ::Simple::error="CODE: GetEntries($type) given invalid timestamp";
return undef;
}
$Event{selecttype}=$ctype;
$Event{year}=$tm[5]+1900;
$Event{month}=$tm[4]+1;
$Event{day}=$tm[3];
} elsif ($ctype eq "lastn") {
if ($#opts<1) {
$LJ::Simple::error="CODE: GetEntries($type) requires num and beforedate in \@opts";
return undef;
}
$Event{selecttype}=$ctype;
my ($num,$beforedate)=@opts;
if (defined $num) {
if ($num!~/^[0-9]{1,2}$/) {
$LJ::Simple::error="CODE: GetEntries($type) requires valid number for num";
return undef;
}
if ($num>50) {
$LJ::Simple::error="Maximum number of journal entries returned is 50";
return undef;
}
} else {
$num=20;
}
$Event{howmany}=$num;
if (defined $beforedate) {
if ($beforedate!~/^[0-9]+$/) {
$LJ::Simple::error="Invalid Unix timestamp";
return undef;
}
my @tm=gmtime($beforedate);
if ($#tm==-1) {
$LJ::Simple::error="CODE: GetEntries($type) given invalid timestamp";
return undef;
}
$Event{beforedate}=strftime("%Y-%m-%d %H:%M:%S",@tm);
}
} elsif ($ctype eq "one") {
if ($#opts<0) {
$LJ::Simple::error="CODE: GetEntries($type) requires item_id in \@opts";
return undef;
}
my ($item_id)=@opts;
if ($item_id!~/^-*[0-9]+$/) {
$LJ::Simple::error="Invalid item_id";
return undef;
}
if ($item_id<-1) {
$LJ::Simple::error="Invalid item_id";
return undef;
}
$Event{selecttype}=$ctype;
$Event{itemid}=$item_id;
} elsif ($ctype eq "sync") {
if ($#opts<0) {
$LJ::Simple::error="CODE: GetEntries($type) requires timestamp in \@opts";
return undef;
}
my ($lastsync)=@opts;
if ($lastsync!~/^[0-9]+$/) {
$LJ::Simple::error="Invalid Unix timestamp";
return undef;
}
my @tm=gmtime($lastsync);
if ($#tm==-1) {
$LJ::Simple::error="CODE: GetEntries($type) given invalid timestamp";
return undef;
}
$Event{lastsync}=strftime("%Y-%m-%d %H:%M:%S",@tm);
$Event{selecttype}="syncitems";
} else {
$LJ::Simple::error="CODE: GetEntries() does not understand type $type\n";
return undef;
}
$self->SendRequest("getevents",\%Event,\%Resp) || return undef;
my %Ev=();
my %Pr=();
my ($k,$v);
while(($k,$v)=each %Resp) {
my ($num,$key,$hash)=(undef,undef,undef);
if ($k=~/^events_([0-9]+)_(.*)$/) {
($num,$key,$hash)=($1,$2,\%Ev);
} elsif ($k=~/^prop_([0-9]+)_(.*)$/) {
($num,$key,$hash)=($1,$2,\%Pr);
}
if (defined $hash) {
(exists $hash->{$num}) || ($hash->{$num}={});
$hash->{$num}->{$key}=$v;
}
}
my $ehr=undef;
foreach $ehr (values %Ev) {
my $itemid=$ehr->{itemid};
$hr->{$itemid}={};
my $nhr=$hr->{$itemid};
%{$nhr}=(
__htmlid => ($ehr->{itemid} * 256) + $ehr->{anum},
__anum => $ehr->{anum},
__itemid => $itemid,
event => $ehr->{event},
lineenddings => "unix",
);
(defined $journal) && ($nhr->{usejournal}=$journal);
(exists $ehr->{subject}) && ($nhr->{subject}=$ehr->{subject});
(exists $ehr->{allowmask}) && ($nhr->{allowmask}=$ehr->{allowmask});
(exists $ehr->{security}) && ($nhr->{security}=$ehr->{security});
if ($ehr->{eventtime}=~/([0-9]+)-([0-9]+)-([0-9]+)\s([0-9]+):([0-9]+):([0-9]+)/o) {
$nhr->{year}=int($1);
$nhr->{mon}=int($2);
$nhr->{day}=int($3);
$nhr->{hour}=int($4);
$nhr->{min}=int($5);
my $timet=mktime($6,$5,$4,$3,$2-1,$1-1900);
if (!defined $timet) {
$LJ::Simple::error="Failed to mktime() from \"$ehr->{eventtime}\" for itemid $hr->{$ehr->{itemid}}->{__htmlid}";
return undef;
}
$nhr->{__timet}=$timet;
} else {
$LJ::Simple::error="Failed to parse eventtime \"$ehr->{eventtime}\" for itemid $hr->{$ehr->{itemid}}->{__htmlid}";
return undef;
}
}
my $phr=undef;
foreach $phr (values %Pr) {
if (!exists $hr->{$phr->{itemid}}) {
$LJ::Simple::error="Protocol error: properties returned for itemid not seen";
return undef;
}
my $nhr=$hr->{$phr->{itemid}};
my $k=join("_","prop",$phr->{name});
if (!exists $nhr->{$k}) {
$nhr->{$k}=$phr->{value};
}
}
return $hr;
}
=pod
=back
=head2 Getting information from an entry
=over 4
=item $lj->GetDate($event)
Gets the date for the event given. The date is returned as a C<time_t> (i.e. seconds
since epoch) value. Returns C<undef> on failure.
Example code:
use POSIX; # For strftime()
## Get date
my $timet=$lj->GetDate(\%Event);
(defined $timet)
|| die "$0: Failed to set date of entry - $LJ::Simple::error\n";
# Get time list using gmtime()
my @tm=gmtime($timet);
($#tm<0) &&
die "$0: Failed to run gmtime() on time_t $timet\n";
# Format date in the normal way used by LJ "YYYY-MM-DD hh:mm:ss"
my $jtime=strftime("%Y-%m-%d %H:%M:%S",@tm);
=cut
sub GetDate($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if (!exists $event->{__timet}) {
$LJ::Simple::error="No time value stored";
return undef;
}
return $event->{__timet};
}
=pod
=item $lj->GetItemId($event)
Returns a list which contains the real C<item_id>, C<anum> and HTMLised C<item_id> which
can be used to contruct a URL suitable for accessing the item via the web.
Returns C<undef> on failure. Note that you must only use this
routine on entries which have been returned by the C<GetEntries()>
routine.
Example code:
my ($item_id,$anum,$html_id)=$lj->GetItemId(\%Event);
(defined $item_id)
|| die "$0: Failed to get item id - $LJ::Simple::error\n";
=cut
sub GetItemId($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if (!exists $event->{__itemid}) {
$LJ::Simple::error="item_id does not exist - must use GetEntries()";
return undef;
}
if (!exists $event->{__anum}) {
$LJ::Simple::error="anum does not exist - must use GetEntries()";
return undef;
}
if (!exists $event->{__htmlid}) {
$LJ::Simple::error="HTML id does not exist - must use GetEntries()";
return undef;
}
return ($event->{__itemid},$event->{__anum},$event->{__htmlid});
}
=pod
=item $lj->GetURL($event)
Returns the URL which can be used to access the journal entry via a web
browser. Returns C<undef> on failure. Note that you must only use this
routine on entries which have been returned by the C<GetEntries()>
routine.
Example code:
my $url=$lj->GetURL(\%Event);
(defined $url)
|| die "$0: Failed to get URL - $LJ::Simple::error\n";
system("netscape -remote 'openURL($url)'");
=cut
sub GetURL($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if (!exists $event->{__htmlid}) {
$LJ::Simple::error="HTML id does not exist - must use GetEntries()";
return undef;
}
my $user=$self->user();
my $server=$self->{lj}->{host};
my $port=$self->{lj}->{port};
my $htmlid=$event->{__htmlid};
return "http://$server:$port/talkpost.bml\?journal=$user\&itemid=$htmlid";
}
=pod
=item $lj->GetSubject($event)
Gets the subject for the journal entry. Returns the subject if it is
available, C<undef> otherwise.
Example code:
my $subj=$lj->GetSubject(\%Event)
if (defined $subj) {
print "Subject: $subj\n";
}
=cut
sub GetSubject($$) {
my $self=shift;
my ($event) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if (!exists $event->{subject}) {
$LJ::Simple::error="No subject set";
return undef;
}
return $event->{subject};
}
=pod
=item $lj->GetEntry($event)
Gets the entry for the journal. Returns either a single string which contains
the entire journal entry or C<undef> on failure.
Example code:
my $ent = $lj->GetEntry(\%Event);
(defined $ent)
|| die "$0: Failed to get entry - $LJ::Simple::error\n";
print "Entry: $ent\n";
=cut
sub GetEntry($$) {
my $self=shift;
my ($event) = @_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if (!exists $event->{event}) {
$LJ::Simple::error="No journal entry set";
return undef;
}
return $event->{event};
}
=pod
=item $lj->GetProtect($event)
Gets the protection information on the event given. Returns a list with
details of the protection set on the post. On failure C<undef> is returned.
There are several different types of protection which can be returned for a
journal entry. These include public, friends only, specific friends groups
and private. The list returned will always have the type of protection listed
first followed by any details of that protection. Thus the list can contain:
("public")
A publically accessable journal entry
("friends")
Only friends may read the entry
("groups","group1" ...)
Only users listed in the friends groups given after the "groups"
may read the entry
("private")
Only the owner of the journal may read the entry
Example code:
my ($protect,@prot_opt)=$lj->GetProtect(\%Event);
(defined $protect) ||
die "$0: Failed to get entry protection type - $LJ::Simple::error\n";
if ($protect eq "public") {
print "Journal entry is public\n";
} elsif ($protect eq "friends") {
print "Journal entry only viewable by friends\n";
} elsif ($protect eq "groups") {
print "Journal entry only viewable by friends in the following groups:\n";
print join(", ",@prot_opt),"\n";
} elsif ($protect eq "private") {
print "Journal entry only viewable by the journal owner\n";
}
=cut
sub GetProtect($$) {
my $self=shift;
my ($event)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
if ((!exists $event->{security})||($event->{security} eq "")) {
return "public";
}
if ($event->{security} eq "private") {
return "private";
}
if ($event->{security} ne "usemask") {
$LJ::Simple::error="INTERNAL: security contains unknown value \"$event->{security}\"";
return undef;
}
if (($event->{allowmask} & 1) == 1) {
return "friends";
}
my @lst=("groups");
my $g=undef;
foreach $g (keys %{$self->{groups}->{name}}) {
my $bit=1 << $self->{groups}->{name}->{$g}->{id};
if (($event->{allowmask} & $bit) == $bit) {
push(@lst,$g);
}
}
return @lst;
}
##
## Helper function used to get meta data
##
sub Getprop_general($$$$$) {
my ($self,$event,$prop,$caller,$type)=@_;
$LJ::Simple::error="";
if (ref($event) ne "HASH") {
$LJ::Simple::error="CODE: Not given a hash reference";
return undef;
}
my $key=join("_","prop",$prop);
if (!exists $event->{$key}) {
if ($type eq "bool") {
return 0;
}
return "";
}
return $event->{$key};
}
=pod
=item $lj->Getprop_taglist($event)
Get tags for given entry.
=cut
sub Getprop_taglist($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"taglist","Getprop_taglist","char");
}
=pod
=item $lj->Getprop_backdate($event)
Indicates if the journal entry is back dated or not. Back dated
entries do not appear on the friends view of your journal entries. Returns
C<1> if the entry is backdated, C<0> if it is not. C<undef> is returned in the
event of an error.
Example code:
my $prop=$lj->Getprop_backdate(\%Event);
(defined $prop) ||
die "$0: Failed to get property - $LJ::Simple::error\n";
if ($prop) {
print STDERR "Journal is backdated\n";
} else {
print STDERR "Journal is not backdated\n";
}
=cut
sub Getprop_backdate($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"opt_backdated","Getprop_backdate","bool");
}
=pod
=item $lj->Getprop_current_mood($event)
Used to get the current mood for the journal being written. This returns the
mood if one exists, an empty string if none exists or C<undef> in the event
of an error.
Example code:
my $prop=$lj->Getprop_current_mood(\%Event);
(defined $prop) ||
die "$0: Failed to get property - $LJ::Simple::error\n";
if ($prop ne "") {
print STDERR "Journal has mood of $prop\n";
} else {
print STDERR "Journal has no mood set\n";
}
=cut
sub Getprop_current_mood($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"current_mood","Getprop_current_mood","char");
}
=pod
=item $lj->Getprop_current_mood_id($event)
Used to get the current mood_id for the journal being written. Will return
the mood_id if one is set, a null string is one is not set and C<undef> in
the event of an error.
Example code:
my $prop=$lj->Getprop_current_mood_id(\%Event);
(defined $prop) ||
die "$0: Failed to get property - $LJ::Simple::error\n";
if ($prop ne "") {
print STDERR "Journal has mood_id of $prop\n";
} else {
print STDERR "Journal has no mood_id set\n";
}
=cut
sub Getprop_current_mood_id($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"current_moodid","Getprop_current_mood_id","num");
}
=pod
=item $lj->Getprop_current_music($event)
Used to get the current music for the journal entry being written. Returns
the music if one is set, a null string is one is not set and C<undef> in
the event of an error.
Example code:
my $prop=$lj->Getprop_current_music(\%Event);
(defined $prop) ||
die "$0: Failed to get property - $LJ::Simple::error\n";
if ($prop) {
print STDERR "Journal has the following music: $prop\n";
} else {
print STDERR "Journal has no music set for it\n";
}
=cut
sub Getprop_current_music($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"current_music","Getprop_current_music","char");
}
=pod
=item $lj->Getprop_preformatted($event)
Used to see if the text for the journal entry being written is preformatted in HTML
or not. This returns true (C<1>) if so, false (C<0>) if not.
Example code:
$lj->Getprop_preformatted(\%Event) &&
print "Journal entry is preformatted\n";
=cut
sub Getprop_preformatted($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"opt_preformatted","Getprop_preformatted","bool");
}
=pod
=item $lj->Getprop_nocomments($event)
Used to see if the journal entry being written can be commented on or not.
This returns true (C<1>) if so, false (C<0>) if not.
Example code:
$lj->Getprop_nocomments(\%Event) &&
print "Journal entry set to disallow comments\n";
=cut
sub Getprop_nocomments($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"opt_nocomments","Getprop_nocomments","bool");
}
=pod
=item $lj->Getprop_picture_keyword($event)
Used to get the picture keyword for the journal entry being written. Returns
the picture keyword if one is set, a null string is one is not set and C<undef> in
the event of an error.
Example code:
my $prop=$lj->Getprop_picture_keyword(\%Event);
(defined $prop) ||
die "$0: Failed to get property - $LJ::Simple::error\n";
if ($prop) {
print STDERR "Journal has picture keyword $prop set\n";
} else {
print STDERR "Journal has no picture keyword set\n";
}
=cut
sub Getprop_picture_keyword($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"picture_keyword","Getprop_picture_keyword","char");
}
=pod
=item $lj->Getprop_noemail($event)
Used to see if comments on the journal entry being written should be emailed or
not. This returns true (C<1>) if so comments should B<not> be emailed and false
(C<0>) if they should be emailed.
Example code:
$lj->Getprop_noemail(\%Event) &&
print "Comments to journal entry not emailed\n";
=cut
sub Getprop_noemail($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"opt_noemail","Getprop_noemail","bool");
}
=pod
=item $lj->Getprop_unknown8bit($event)
Used see if there is 8-bit data which is not in UTF-8 in the journal entry
being written. This returns true (C<1>) if so, false (C<0>) if not.
Example code:
$lj->Getprop_unknown8bit(\%Event) &&
print "Journal entry contains 8-bit data not in UTF-8 format\n";
=cut
sub Getprop_unknown8bit($$) {
my ($self,$event)=@_;
$LJ::Simple::error="";
return $self->Getprop_general($event,"unknown8bit","Getprop_unknown8bit","bool");
}
##### Start of helper functions
##
## A helper function which takes a key and value pair;
## both are encoded for HTTP transit.
##
sub EncVal($$) {
my ($key,$val)=@_;
$key=~s/ /\+/go;
$key=~s/([^a-z0-9+])/sprintf("%%%x",ord($1))/egsi;
$val=~s/ /\+/go;
$val=~s/([^a-z0-9+])/sprintf("%%%02x",ord($1))/egsi;
return "$key=$val";
}
##
## A helper function which takes an encoded value from HTTP
## transit and decodes it
##
sub DecVal($) {
my ($val)=@_;
$val=~s/\+/ /go;
$val=~s/%([0-9A-F]{2})/pack("C", hex($1))/egsi;
return "$val";
}
##
## Actually make the LJ request; could be called directly, but isn't
## documented.
##
## The first argument is the the mode to use. The list of currently
## supported modes is:
## o login
## o postevent
##
## The second argument is a hash reference to arguments specific to the
## mode.
##
## The third argument is a reference to a hash which contain the response
## from the LJ server. This can be undef.
##
## Returns 1 on success, 0 on failure. On failure $LJ::Simple::error is
## populated.
##
sub SendRequest($$$$) {
my ($self,$mode,$args,$req_hash)=@_;
my ($oself, $omode, $oargs, $oreq_hash, $oretry) = @_;
$oretry = 0 unless defined($oretry);
$LJ::Simple::error="";
my $sub=$LJ::Simple::NonBlock;
my $bytes_in=0;
my $bytes_out=0;
my $timestart=time();
if ((defined $sub) && (ref($sub) ne "CODE")) {
my $reftype=ref($sub);
$LJ::Simple::error="\$LJ::Simple::NonBlock given a $reftype reference, not CODE";
return 0;
}
$self->{request}={};
if ((ref($args) ne "HASH")&&($mode ne "getchallenge")) {
$LJ::Simple::error="INTERNAL: SendRequest() not given hashref for arguments";
return 0;
}
if ((defined $req_hash) && (ref($req_hash) ne "HASH")) {
$LJ::Simple::error="INTERNAL: SendRequest() not given hashref for responses";
return 0;
}
$mode=lc($mode);
my @request=(
"mode=$mode",
);
if ($mode ne "getchallenge") {
push(@request,
EncVal("user",$self->{auth}->{user}),
);
# Much fun here - see if we use the challenge-response stuff
if ($LJ::Simple::challenge) {
Debug("Trying to use challenge-response system");
Debug(" Getting new challenge");
my %chall=();
$self->SendRequest("getchallenge",undef,\%chall) || return 0;
if ($chall{auth_scheme} ne "c0") {
$LJ::Simple::error="Server returned unsupported auth_scheme \"$chall{auth_scheme}\"";
return 0;
}
Debug(" Got challenge from server:");
Debug(" challenge: $chall{challenge}");
Debug(" expire_time: $chall{expire_time}");
Debug(" server_time: $chall{server_time}");
# Work out our own timeout point, basically the livetime of the
# challenge less 10 seconds of fudge factor.
my $chall_livetime=$chall{expire_time} - $chall{server_time} - 10;
my $ctime=time();
$self->{auth}->{challenge}->{timeout}=$ctime + $chall_livetime;
Debug(" Challenge lifetime is $chall_livetime seconds");
Debug(" Current: $ctime");
Debug(" Expire: $self->{auth}->{challenge}->{timeout}");
$self->{auth}->{challenge}->{challenge}=$chall{challenge};
# We assume that the Digest::MD5 module is loaded already; also
# means that we have an MD5 hash of the password to hand.
my $md5=Digest::MD5->new;
$md5->add($chall{challenge});
$md5->add($self->{auth}->{hash});
$self->{auth}->{challenge}->{hash}=$md5->hexdigest;
}
if (exists $self->{auth}->{challenge}->{hash}) {
push(@request,
EncVal("auth_method","challenge"),
EncVal("auth_challenge",$self->{auth}->{challenge}->{challenge}),
EncVal("auth_response",$self->{auth}->{challenge}->{hash}),
);
} else {
if (exists $self->{auth}->{hash}) {
push(@request,EncVal("hpassword",$self->{auth}->{hash}));
} else {
push(@request,EncVal("password",$self->{auth}->{pass}));
}
}
my $ljprotver=0;
if ($LJ::Simple::UTF) { $ljprotver=1; }
push(@request,
"ver=$ljprotver",
);
}
(defined $sub) && &{$sub}($mode,0.1,"Preparing request data",$bytes_in,$bytes_out,time()-$timestart,0);
if ($mode eq "login") {
push(@request,EncVal("clientversion",$LJ::Simple::LJ_Client));
if ((exists $args->{moods}) && ($args->{moods} == 1)) {
push(@request,EncVal("getmoods",0));
}
if ((exists $args->{getpickws}) && ($args->{getpickws} == 1)) {
push(@request,EncVal("getpickws",1));
push(@request,EncVal("getpickwurls",1));
}
} elsif ( ($mode eq "postevent")
|| ($mode eq "editevent")
|| ($mode eq "syncitems")
|| ($mode eq "getevents")
|| ($mode eq "getfriends")
|| ($mode eq "friendof")
|| ($mode eq "checkfriends")
|| ($mode eq "getdaycounts")
|| ($mode eq "getfriendgroups")
|| ($mode eq "editfriendgroups")
) {
if (defined $args) {
my ($k,$v);
while(($k,$v)=each %{$args}) {
if (!defined $k) {
$LJ::Simple::error="CODE: SendRequest() given undefined key value";
return 0;
}
if (!defined $v) {
$LJ::Simple::error="CODE: SendRequest() given undefined value for \"$k\"";
return 0;
}
push(@request,EncVal($k,$v));
}
}
} elsif ( ($mode eq "getchallenge")
|| ($mode eq "sessiongenerate")
) {
# NOP - nothing required
} elsif ($mode eq "getrawdata") {
if (! $args->{url}) {
$LJ::Simple::error="CODE: SendRequest() given undefined url value with getrawdata;";
return 0;
}
if (! $self->{auth}->{ljsession}) {
$LJ::Simple::error="CODE: SendRequest(getrawdata) called without previous sessiongenerate;";
return 0;
}
} else {
$LJ::Simple::error="INTERNAL: SendRequest() given unsupported mode \"$mode\"";
return 0;
}
my $req=join("&",@request);
my $ContLen=length($req);
(defined $sub) && &{$sub}($mode,0.2,"Preparing connection to server",$bytes_in,$bytes_out,time()-$timestart,0);
## Now we've got the request ready, time to start talking to the web
# Work out where we're talking to and the URI to do it with
my $server=$self->{lj}->{host};
my $host=$server;
my $port=$self->{lj}->{port};
my $uri="/interface/flat";
$uri = $args->{url} if $mode eq "getrawdata";
if (defined $self->{proxy}) {
$uri="http://$server:$port$uri";
$server=$self->{proxy}->{host};
$port=$self->{proxy}->{port};
}
# Prepare the HTTP request now we've got the URI
my @HTTP = ();
if ($mode eq "getrawdata") {
my $post_data = "user=" . $self->{auth}->{user} . "&auth_method=cookie";
if ($args->{"post-data"}) {
foreach (keys %{$args->{"post-data"}}) {
$post_data .= "&" . $_ . "=" . $args->{"post-data"}->{$_};
}
}
my $method = "POST";
if ($args->{"http_method"}) {
$method = $args->{"http_method"};
}
@HTTP=(
"$method $uri HTTP/1.0",
"Host: $host",
"Content-type: application/x-www-form-urlencoded",
"User-Agent: " . $LJ::Simple::UserAgent,
"Content-length: " . length($post_data),
"X-LJ-Auth: cookie",
"Cookie: ljsession=" . $self->{auth}->{ljsession},
);
push(@HTTP,
"",
$post_data,
"",
);
}
else {
@HTTP=(
"POST $uri HTTP/1.0",
"Host: $host",
"Content-type: application/x-www-form-urlencoded",
"User-Agent: " . $LJ::Simple::UserAgent,
"Content-length: $ContLen",
);
if ($self->{fastserver}) {
push(@HTTP,"Cookie: ljfastserver=1");
}
push(@HTTP,
"",
$req,
"",
);
}
# Prepare the socket
my $tcp_proto=getprotobyname("tcp");
socket(SOCK,PF_INET,SOCK_STREAM,$tcp_proto);
# Resolve the server name we're connecting to
(defined $sub) && &{$sub}($mode,0.3,"Starting to resolve $server to IP address",$bytes_in,$bytes_out,time()-$timestart,0);
my $addr=inet_aton($server);
if (!defined $addr) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Failed to resolve server $server";
return 0;
}
}
my $sin=sockaddr_in($port,$addr);
my $ip_addr=join(".",unpack("CCCC",$addr));
my $proto=$LJ::Simple::ProtoSub;
($LJ::Simple::protocol) && &{$proto}(undef,undef,$server,$ip_addr);
if ($LJ::Simple::raw_protocol) {
print STDERR "Connecting to $server [$ip_addr]\n";
print STDERR "Lines starting with \"-->\" is data SENT to the server\n";
print STDERR "Lines starting with \"<--\" is data RECEIVED from the server\n";
}
# Connect to the server
(defined $sub) && &{$sub}($mode,0.4,"Trying to connect to server $server",$bytes_in,$bytes_out,time()-$timestart,0);
if (!connect(SOCK,$sin)) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Failed to connect to $server - $!";
return 0;
}
}
($LJ::Simple::protocol) && &{$proto}(undef,"Connected to $server [$ip_addr]",$server,$ip_addr);
($LJ::Simple::raw_protocol) &&
print STDERR "Connected to $server [$ip_addr]\n";
# Send the HTTP request
(defined $sub) && &{$sub}($mode,0.5,"Starting to send HTTP request to $server",$bytes_in,$bytes_out,time()-$timestart,0);
my $cp=0.5;
foreach (@HTTP) {
my $line="$_\r\n";
my $len=length($line);
my $pos=0;
my $fail=0;
while($pos!=$len) {
my $nbytes=syswrite(SOCK,$line,$len,$pos);
if (!defined $nbytes) {
if ( ($! == EAGAIN) || ($! == EINTR) ) {
$fail++;
if ($fail>4) {
$LJ::Simple::error="Write to socket failed with EAGAIN/EINTR $fail times";
shutdown(SOCK,2);
close(SOCK);
return 0;
}
next;
} else {
$LJ::Simple::error="Write to socket failed - $!";
shutdown(SOCK,2);
close(SOCK);
return 0;
}
}
$pos+=$nbytes;
$bytes_out+=$nbytes;
$cp=$cp+0.001;
(defined $sub) && &{$sub}($mode,$cp,"Sending HTTP request to $server",$bytes_in,$bytes_out,time()-$timestart,0);
}
($LJ::Simple::protocol) && &{$proto}(0,$_,$server,$ip_addr);
($LJ::Simple::raw_protocol) && print STDERR "--> $_\n";
}
# Read the response from the server - use select()
(defined $sub) && &{$sub}($mode,0.6,"Getting HTTP response from $server",$bytes_in,$bytes_out,time()-$timestart,0);
$cp=0.6001;
my ($rin,$rout,$eout)=("","","");
vec($rin,fileno(SOCK),1) = 1;
my $ein = $rin;
my $response="";
my $done=0;
while (!$done) {
my $nfound;
if (defined $sub) {
$nfound = select($rout=$rin,undef,$eout=$ein,0);
my $ttaken=time()-$timestart;
if ($nfound!=1) {
if ($ttaken>$LJ::Simple::timeout) {
&{$sub}($mode,1,"Connection with server $server timed out",$bytes_in,$bytes_out,$ttaken,0);
$LJ::Simple::error="Failed to receive data from $server [$ip_addr]";
shutdown(SOCK,2);
close(SOCK);
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
return 0;
}
}
&{$sub}($mode,$cp,"Waiting for response from $server",$bytes_in,$bytes_out,time()-$timestart,1);
next;
}
} else {
$nfound = select($rout=$rin,undef,$eout=$ein,$LJ::Simple::timeout);
if ($nfound!=1) {
$LJ::Simple::error="Failed to receive data from $server [$ip_addr]";
shutdown(SOCK,2);
close(SOCK);
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
return 0;
}
}
}
my $resp="";
my $nbytes=sysread(SOCK,$resp,$LJ::Simple::buffer);
if (!defined $nbytes) {
$LJ::Simple::error="Error in getting data from $server [$ip_addr] - $!";
shutdown(SOCK,2);
close(SOCK);
(defined $sub) && &{$sub}($mode,1,$LJ::Simple::error,$bytes_in,$bytes_out,time()-$timestart,0);
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
return 0;
}
} elsif ($nbytes==0) {
$done=1;
} else {
$bytes_in=$bytes_in+$nbytes;
(defined $sub) && &{$sub}($mode,$cp,"Getting response from server $server",$bytes_in,$bytes_out,time()-$timestart,0);
$cp=$cp+0.001;
$response="$response$resp";
if ($LJ::Simple::raw_protocol) {
print STDERR "<-- ";
foreach (split(//,$resp)) {
s/([\x00-\x20\x7f-\xff])/sprintf("\\%o",ord($1))/ei;
print "$_";
}
print STDERR "\n";
} elsif ($LJ::Simple::protocol) {
foreach (split(/[\r\n]{1,2}/o,$resp)) {
&{$proto}(1,$_,$server,$ip_addr);
}
}
}
}
(defined $sub) && &{$sub}($mode,0.7,"Finished getting data from server $server",$bytes_in,$bytes_out,time()-$timestart,0);
# Shutdown the socket
if (!shutdown(SOCK,2)) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Failed to shutdown socket - $!";
(defined $sub) && &{$sub}($mode,1,$LJ::Simple::error,$bytes_in,$bytes_out,time()-$timestart,0);
return 0;
}
}
# Close the socket
close(SOCK);
(defined $sub) && &{$sub}($mode,0.8,"Parsing data from server $server",$bytes_in,$bytes_out,time()-$timestart,0);
## We've got the response from the server, so we now parse it
if (!defined $response) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Failed to get result from server";
return 0;
}
}
## Ensure that response isn't zero length
if (length($response) == 0) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Zero length response from server";
(defined $sub) && &{$sub}($mode,1,"$LJ::Simple::error $server",$bytes_in,$bytes_out,time()-$timestart,0);
return 0;
}
}
# Split into headers and body
my ($http,$body)=split(/\r\n\r\n/,$response,2);
if (!defined $http) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Failed to get HTTP headers from server";
(defined $sub) && &{$sub}($mode,1,"$LJ::Simple::error $server",$bytes_in,$bytes_out,time()-$timestart,0);
return 0;
}
}
if (!defined $body) {
if ($oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
$LJ::Simple::error="Failed to get HTTP body from server";
(defined $sub) && &{$sub}($mode,1,"$LJ::Simple::error $server",$bytes_in,$bytes_out,time()-$timestart,0);
return 0;
}
}
# First lets see if we got a valid response
$self->{request}->{http}={};
$self->{request}->{http}->{headers}=[(split(/\r\n/,$http))];
my $srv_resp=$self->{request}->{http}->{headers}->[0];
$srv_resp=~/^HTTP\/[^\s]+\s([0-9]+)\s+(.*)/;
my ($srv_code,$srv_msg)=($1,$2);
$self->{request}->{http}->{code}=$srv_code;
$self->{request}->{http}->{msg}=$srv_msg;
if ($srv_code != 200) {
$LJ::Simple::error="HTTP request failed with $srv_code $srv_msg";
return 0;
}
if ($mode eq "getrawdata") {
return {"content" => $body};
}
else {
# We did, so lets pull in the LJ stuff for processing
$self->{request}->{lj}={};
# The response from LJ takes the form of a key\nvalue\n
# Note that the value can be null tho
$done=0;
while (!$done) {
if ($body=~/^([^\n]+)\n([^\n]*)\n(.*)$/so) {
my ($k,$v)=(undef,undef);
($k,$v,$body)=(lc($1),DecVal($2),$3);
$v=~s/\r\n/\n/go;
$self->{request}->{lj}->{$k}=$v;
} else {
$done=1;
}
}
# Got it into a hash - lets see if we made a successful request
if ( (!exists $self->{request}->{lj}->{success}) ||
($self->{request}->{lj}->{success} ne "OK") ) {
if ($omode ne "login" && $oretry < $LJ::Simple::network_retries) {
$oretry++;
sleep $LJ::Simple::network_sleep;
return $oself->SendRequest($omode, $oargs, $oreq_hash, $oretry);
}
else {
my $errmsg="Server Error, try again later";
if (exists $self->{request}->{lj}->{errmsg}) {
$errmsg=$self->{request}->{lj}->{errmsg};
}
$LJ::Simple::error="($mode) LJ request failed: $errmsg";
(defined $sub) && &{$sub}($mode,1,"$LJ::Simple::error $server",$bytes_in,$bytes_out,time()-$timestart,0);
return 0;
}
}
# We did!
# Now to populate the hash we were given (if asked to)
if (defined $req_hash) {
%{$req_hash}=();
my ($k,$v);
while(($k,$v)=each %{$self->{request}->{lj}}) {
$req_hash->{$k}=$v;
}
}
}
(defined $sub) && &{$sub}($mode,1,"Finished processing request to server $server",$bytes_in,$bytes_out,time()-$timestart,0);
return 1;
}
sub GenerateCookie() {
my ($self) = @_;
$self->SendRequest("sessiongenerate", {}, undef) || return undef;
$self->{auth}->{ljsession} = $self->{request}->{lj}->{ljsession};
return undef unless $self->{auth}->{ljsession};
return $self->{auth}->{ljsession};
}
sub GetRawData() {
my ($self,$args)=@_;
return $self->SendRequest("getrawdata", $args, undef) || return undef;
}
##
## Output debugging info
##
sub Debug(@) {
($LJ::Simple::debug) || return;
my $msg=join("",@_);
foreach (split(/\n/,$msg)) {
print STDERR "DEBUG> $_\n";
}
}
##
## Dump out a list recursively. Will call dump_hash
## for any hash references in the list.
##
## Generally used for debugging
##
sub dump_list($$) {
my ($lr,$sp)=@_;
my $le="";
my $res="";
foreach $le (@{$lr}) {
if (ref($le) eq "HASH") {
$res="$res$sp\{\n";
$res=$res . dump_hash($le,"$sp ");
$res="$res$sp},\n";
} elsif (ref($le) eq "ARRAY") {
$res="$res$sp\[\n" . dump_list($le,"$sp ") . "$sp],\n";
} else {
my $lv=$le;
if (defined $lv) {
$lv=~s/\n/\\n/go;
$lv=quotemeta($lv);
$lv=~s/\\-/-/go;
$lv="\"$lv\"";
} else {
$lv="undef";
}
$res="$res$sp$lv,\n";
}
}
return $res;
}
##
## Dump out a hash recursively. Will call dump_list
## for any list references in the hash values.
##
## Generally used for debugging
##
sub dump_hash($$) {
my ($hr,$sp)=@_;
my ($k,$v)=();
my $res="";
while(($k,$v)=each %{$hr}) {
$k=quotemeta($k);
$k=~s/\\-/-/go;
if (ref($v) eq "HASH") {
$res="$res$sp\"$k\"\t=> {\n";
$res=$res . dump_hash($v,"$sp ");
$res="$res$sp},\n";
} elsif (ref($v) eq "ARRAY") {
$res="$res$sp\"$k\"\t=> \[\n" . dump_list($v,"$sp ") . "$sp],\n";
} else {
if (defined $v) {
$v=~s/\n/\\n/go;
$v=quotemeta($v);
$v=~s/\\\\n/\\n/go;
$v=~s/\\-/-/go;
$v="\"$v\"";
} else {
$v="undef";
}
my $out="$sp\"$k\"\t=> $v,";
$res="$res$out\n";
}
}
return $res;
}
1;
__END__
=pod
=back
=head1 AUTHOR
Simon Burr E<lt>simes@bpfh.netE<gt>
=head1 SEE ALSO
perl
L<http://www.livejournal.com/>
=head1 LICENSE
Copyright (c) 2002, Simon Burr E<lt>F<simes@bpfh.net>E<gt>
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
* Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
* Neither the name of the author nor the names of its contributors may
be used to endorse or promote products derived from this software
without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
=cut