NAME
Net::RGTP - Reverse Gossip client
SYNOPSIS
use Net::RGTP;
my $rgtp = Net::RGTP->new(Host=>'gossip.example.com')
or die "Cannot connect to RGTP server!";
$rgtp->login('spqr1@cam.ac.uk', 'DEADBEEFCAFEF00D');
for my $itemid (keys %{$rgtp->items}) {
my $item = $rgtp->item($itemid);
print $itemid, ' ', $item->{'subject'}, ' has ',
scalar(@{$item->{'posts'}}),
" posts.\n";
}
DESCRIPTION
Net::RGTP
is a class implementing the RGTP bulletin board protocol, as used in the Cambridge University GROGGS system. At present it provides read-only access only.
OVERVIEW
RGTP stands for Reverse Gossip Transfer Protocol. An RGTP board, such as GROGGS, consists essentially of a set of "items", each denoted by an eight-character itemid such as "A1240111". An item consists of a sequence of posts on a given subject, identified by a subject string attached to the item. When an item has reached a certain size, attempting to post to it will instead generate a new item, known as a "continuation" or "child" item, with a new itemid and subject string. RGTP keeps track of which items are children of which parent items, thus allowing long chains of discussion to be built.
The first character of itemids was "A" in 1986, the first year of GROGGS's existence, and has been incremented through the alphabet every year since. (The letter for 2005 is "U".)
Every user is identified to RGTP by their email address. They are usually identified to the other users by a string known as their "grogname". (These are usually fanciful, and regular contests are held as to the best ones.)
Every action which causes a state change on an RGTP server is given a monotonically increasing sequence number. Most actions are also given timestamps. These are in seconds since midnight UTC, 1 January 1970.
CONSTRUCTOR
- new ([ OPTIONS ])
-
This is the constructor for a new Net::RGTP object.
OPTIONS
are passed in a hash-like fashion, using key and value pairs. Possible options are:Host - the name of the RGTP server to connect to. If this is omitted, it will default to
rgtp-serv.groggs.group.cam.ac.uk
.Port - the port number to connect to. If this is omitted, it will default to 1471, the IANA standard number for RGTP.
Debug - set this to 1 if you want the traffic between the server and client to be printed to stderr. This does not print the contents of files (e.g. the index, or items) as they transfer.
METHODS
- login ([USERID, [SECRET]])
-
Logs in to the RGTP server.
USERID is the user identity to use on the RGTP server, typically an email address. If left blank it will default to "guest".
SECRET is the shared-secret which is sent out by mail. It must either be a hex string with an even number of digits, or undef. It should be undef only if you are expecting not to have to go through authentication (for example, on many RGTP servers the account called "guest" needs no authentication step).
This method returns the current access level, in the format returned by the
access_level
method. The method returnsundef
on failure, and sets$@
to an appropriate message. - access_level
-
Returns the current access level. 0 means only the message of the day may be read. 1 means the index and any item may be read, but nothing may be written. 2 means that items may be posted to. 3 means that the contents of the items, including posts made by other users, may be edited.
- latest_seq
-
Returns the highest sequence number which has been seen in the index of this server. This may be
undef
if we have not downloaded the index (or if the server is entirely empty). - motd
-
Returns a hashref containing only the key posts, which maps to an arrayref containing only one element, a hashref which contains three keys:
seq: the sequence number of the message of the day;
text: the text of the message of the day; and
timestamp: the time the message of the day was last set.
The reason for the baroque formatting is that it matches the format of the response of the
item
method.Returns
undef
if there is no message of the day. - item(ITEMID)
-
Returns a hashref which may if applicable contain the keys:
parent, which is the itemid of the given item's parent;
child, which is the itemid of the given item's child;
subject, which is the subject line of the given item;
reply, which is the sequence number of the most recent reply to the given item; and
edit, which is the sequence number of the most recent edit. (That is, an edit by an editor, not an ordinary reply.)
The hashref will always contain a key posts. This maps to an arrayref of hashrefs, each representing a post to this item. Each hashref may if applicable contain the keys:
seq, which is the sequence number of this post;
timestamp, which is the timestamp of this post;
grogname, which is the grogname of the poster; and
poster, which is the user ID of the poster (that is, their email address).
There will also always be a key text, which contains the text of the post.
item
returnsundef
if the item does not exist.As a special case,
item("motd")
is equivalent to calling themotd
method. - quick_item(ITEMID)
-
Similar to the
item
method, but the hashref returned does not contain the key posts. Use this method if you only need to know, for example, the item's most recent sequence number or its subject line. It executes many times faster than theitem
method, because the content of the item does not need to be transferred.This implements the RGTP function "STAT". The method is not called
stat
because that is a perl builtin. - items
-
Returns a hashref describing every item on the current server.
The keys of the hashref are the itemids of the several items, except for the key "motd", which describes the message of the day. Each key maps to a hash describing the item. The keys of this hash are:
subject: the subject line of the item. This may be truncated by the RGTP server; you may find the exact subject line using the
item
method.posts: a count of posts.
timestamp: the timestamp of the most recent change to this item.
seq: the sequence number of the most recent change to this item.
parent: the itemid of the parent of this item. Exists only for items which have a parent.
child: the itemid of the child of this item. Exists only for items which have a child.
This method may take a long time to execute on popular RGTP servers the first time it is called. This is because it has to download the entire index. Subsequent calls will use cached data and will be faster. See also the
state
method. - state([STATE])
-
This method exists because the
items
method is slow on first use. (Over my home connection, for the main GROGGS server, it takes about forty seconds). When called with no arguments,state
returns a scalar describing the state ofitems
's cache. When called with this scalar as argument, it re-fills the cache with this data. This scalar can be seralised so that the advantages of caching can be gained between sessions. - post(WHERE, WHAT, [OPTS])
-
Adds some content to the RGTP board.
WHAT is a block of text wrapped at 80 columns. I recommend the use of Text::ASCIITable::Wrap to format arbitrary text in this way.
WHERE can be one of three things:
The string
new
. In this case WHAT is posted as a new item on the server.A valid and existing itemid. In this case WHAT is appended as a reply to the given item.
The string
continue
. This only works when the continuation flag is set (see CONTINUATIONS below). WHAT is posted as the first entry in a continuation item.OPTIONS
are passed in a hash-like fashion, using key and value pairs. Possible options are:Seq. The sequence number of the last known reply to this item. Ignored when WHAT is
new
orcontinue
. If this is undefined, the sequence number will not be checked. See COLLISIONS below.Grogname. The grogname to use when posting. If this is undefined, no grogname will be used. Grognames which are too long may cause the method to return an error.
Title. The title to use for the new item. Required when WHAT is
new
orcontinue
, and ignored at all other times.On success, in list context, this method returns a list consisting of the itemid followed by the sequence number of the post. In scalar context, it returns only the itemid.
The method returns
undef
on failure, and sets$@
to an appropriate message. It also causes the functionsitem_is_full
anditem_is_grown
to return values which represent the reason it failed. - item_is_full
-
Returns true iff the most recent call to
post
failed because the target item had gone into a continuation. This is known as the "continuation flag": see CONTINUATIONS below. - item_has_grown
-
Returns true iff the most recent call to
post
failed because of a collision in the target item. See COLLISIONS below.
CONTINUATIONS
Items have a maximum size. Thus after a certain amount of posting to any given item, it will cease to be possible to post any more content. When this happens, post
will return undef
, and set the "continuation flag", which may be inspected using the item_is_full
method.
When the flag is set, and only when the flag is set, it is possible to call post
again with the WHERE parameter set to "continue"
. This creates a continuation item following on from the item you were originally trying to post to.
COLLISIONS
Because RGTP is not threaded, most users want to check, when they reply to an item, that it has not been replied to already while they were composing their reply. The lack of a built-in way to do this is a fundamental flaw in RGTP; most clients get around the problem by doing a STAT (equivalent to our quick_item
method) immediately before posting, and comparing the sequence number given to one taken before the reply was composed. Net::RGTP provides an easy way to accomplish this: setting the Seq option to the post
command. If this check fails, post
will return undef
and item_has_grown
will subsequently return true.
However, any such mechanism introduces a race condition into the protocol. The chance of a race occurring is slight, and the problems caused thereby are small, but programmers should be aware of it.
The only case when RGTP does tell us when an item has been updated is when an item has gone into a continuation. In this case post
and item_has_grown
will behave as if Seq had been specified, even if it was not.
UNIMPLEMENTED
The following aspects of RGTP have not been implemented. This will be addressed in a later revision:
- Edit log
-
Viewing the log of editors' changes to the board.
- Registration
-
Creating new user accounts.
- Editing
-
Using superuser powers to modify other people's comments.
AUTHOR
Thomas Thurman <marnanel@marnanel.org>
CREDITS
Firinel Thurman - for being there to bounce ideas off, and, well, everything.
John Stark - for inventing GROGGS.
Ian Jackson - for inventing RGTP.
Tony Finch - whose RGTP to Atom converter made the idea of this module click for me.
SEE ALSO
The RGTP protocol, at http://www.groggs.group.cam.ac.uk/protocol.txt .
The GROGGS home page, at http://www.groggs.group.cam.ac.uk/ .
Yarrow, a CGI RGTP client, at http://rgtp.thurman.org.uk/gossip/groggs/browse .
GREED, an Emacs RGTP client, at http://www.chiark.greenend.org.uk/~owend/free/GREED.html .
COPYRIGHT
Copyright (c) 2005 Thomas Thurman. All rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
2 POD Errors
The following errors were encountered while parsing the POD:
- Around line 813:
You forgot a '=back' before '=head1'
- Around line 867:
You forgot a '=back' before '=head1'