NAME
RPC::ExtDirect::Client::Async - Asynchronous Ext.Direct client in Perl
SYNOPSIS
use RPC::ExtDirect::Client::Async;
my $client = RPC::ExtDirect::Client::Async->new(host => 'localhost');
$client->call_async(
action => 'Action',
method => 'Method',
arg => [ 'foo', 'bar' ],
cookies => { foo => 'bar' },
cb => sub {
my ($result, $success, $error) = @_;
if ( $success ) {
# Do something with $result
...
}
else {
# Handle $error
...
}
},
);
DESCRIPTION
This module implements a fully asynchronous Ext.Direct client based on AnyEvent::HTTP. It can be used with any event loop supported by AnyEvent.
If you are not familiar with Ext.Direct, start with RPC::ExtDirect::Intro before going further.
CAVEATS
API initialization
This client is fully asynchronous, and does not block on any operation. However before a client object can run Ext.Direct requests, it will need an instance of RPC::ExtDirect::API that holds the available API declaration. This object can be initialized in two ways: either remotely from a server, or locally by passing the api
argument to the Client constructor:
my $api = RPC::ExtDirect::API->new_from_hashref({
api_href => { ... },
});
my $client = RPC::ExtDirect::Client::Async->new(
host => 'localhost',
port => 8080,
api => $api,
);
If no local API instance is provided, the client object will attempt to retrieve it from the server. This operation is non-blocking; constructor will return the client object immediately and it can be used to make calls without waiting for the API to be initialized. However any requests made to the client before its API object is available will be queued internally by the client object, and dispatched only when API is retrieved. This is done transparently to the caller.
You can pass an option callback to be fired when the API is available via "api_cb" option:
my $client = RPC::ExtDirect::Client::Async->new(
...
api_cb => sub {
my ($client, $success, $error) = @_;
if ( !$success ) {
# $error is defined
...
}
},
);
This callback will be fired when a local API instance is provided, too.
If the remote API cannot be retrieved for some reason, the client object will set the error internally as fatal exception. Every request that was queued before the API retrieval failed will have its callback fired as if the request itself has failed, with the $error
argument set to the API retrieval exception. Every subsequent request will have its callback sub fired immediately with an error.
Error handling
In a cooperative multi-tasking environment such as AnyEvent it is very hard, if at all possible, to guarantee that an exception thrown in a module code will be caught properly in the calling code. Because of that, Async client will not die
upon encountering errors, except when the error is fatal. Instead, the error will be passed to the callback:
$client->call_async(
action => 'foo',
method => 'bar',
...
cb => sub {
my ($result, $success, $error) = @_;
if ( $success ) {
# $result is defined, $error is undefined
}
else {
# vice versa: $result is undefined, $error is set
}
},
);
There are two exceptions to this rule: fatal errors encountered in Client constructor, and missing callback argument in "call_async", "submit_async", or "poll_async". This kind of errors usually means that there is a typo somewhere in the calling code, and should be caught early on.
Condition variables
Async client supports AnyEvent's condition variables to signal beginning and finishing a request, if such variable is provided. This can be used to block until several requests are finished:
my $cv = AnyEvent->condvar;
$client->call_async( ..., cv => $cv );
$client->upload_async( ..., cv => $cv );
# This will return only when both requests are done
$cv->recv;
Condition variable can also be used instead of a callback sub:
my $cv = AnyEvent->condvar;
$client->call_async( ..., cb => $cv );
# This will block until the request has finished
my $result = $cv->recv;
# Handle the error, if any
if ( not defined $result ) {
my $error = ($cv->recv)[2];
...
}
See "CONDITION VARIABLES" in AnyEvent for more information.
CLIENT OBJECT INTERFACE
RPC::ExtDirect::Client::Async provides several public methods:
new
-
Constructor. Returns a new Client instance and initializes an Ext.Direct API instance from the server, unless "api" parameter is provided. Accepts named arguments in a hash.
Parameters:
api
-
Optional RPC::ExtDirect::API instance to use. If not given, a new instance will be created from the remote API provided by the server.
config
-
Optional RPC::ExtDirect::Config instance to use. If not provided, a new Config instance will be created unless "api" parameter is provided, in which case the Config instance in the API object will be used instead.
host
-
Server's host name or IP address. This parameter is mandatory.
port
-
Optional port number to use when connecting to the server. Defaults to
80
. api_cb
-
Optional callback to be fired when API retrieval operation is finished. This function will receive three positional arguments: the Client reference, success flag, and the error string if it occured. An example:
my $client = RPC::ExtDirect::Client::Async->new( ... api_cb => sub { my ($client, $success, $error) = @_; # $error is only defined when API retrieval failed if ( !$success ) { ... } }, );
cv
-
Optional condition variable to be signaled when API retrieval finishes. See "Condition variables" for more detail.
-
Cookies to set when calling server side; can be either HTTP::Cookies object or a hashref containing key-value pairs. Setting this in constructor will pass the same cookies to all subsequent client calls.
...
-
All other arguments are stored as options and are applied to HTTP requests. See "http_request" in AnyEvent::HTTP for more detail.
Instance method. Returns an "api_class_client" object with the Ext.Direct API declaration published by the server.
Accepts one mandatory positional argument, API type, that can be either
'remoting'
or'polling'
to retrieve the corresponding API object. call_async
-
Instance method. Calls the specified Ext.Direct Method on the server side and passes the execution Result or Exception to the callback function specified in arguments. Accepts named arguments in a hash.
Parameters:
action
-
Ext.Direct Action (class) name. This parameter is mandatory.
method
-
Ext.Direct Method name to call. This parameter is mandatory.
arg
-
Ext.Direct Method arguments; use arrayref for methods that accept ordered parameters or hashref for named parameters. This parameter is mandatory even for Methods that accept no arguments; in such case, pass an empty arrayref.
cb
-
Mandatory callback function that will be fired when request finishes. This function will receive three positional arguments: the execution Result (or Exception), success flag, and error string if an error has occured:
$client->call_async( ... cb => sub { my ($result, $success, $error) = @_; if ( $success ) { # Do something with $result ... } else { # Handle the $error ... } }, );
cv
-
Optional condition variable to be signaled when the request is finished. See "Condition variables" for more detail.
-
Optional set of cookies for this particular call only. Cookies should be in the same format as for constructor, see "new".
...
-
Any other arguments are passed on to
http_request
function. See "http_request" in AnyEvent::HTTP.
submit_async
-
Instance method. Submits an HTML form request to a Form Handler method and passes the execution Result or Exception to the callback function specified in arguments. Accepts named arguments in a hash.
Parameters:
action
-
Ext.Direct Action (class) name. This parameter is mandatory.
method
-
Ext.Direct Method name to call. This parameter is mandatory.
arg
-
A hashref of the Method arguments. This parameter is mandatory, unless "upload" is specified.
upload
-
An optional arrayref of file names to upload. Files should be readable by the current process, or "submit" will die with an error.
cb
-
Mandatory callback function that will be fired when request finishes. See "call_async".
cv
-
Optional condition variable to be signaled when the request is finished. See "Condition variables" for more detail.
-
Optional set of cookies for this particular call only. Cookies should be in the same format as for constructor, see "new".
upload_async
-
A shortcut for "submit_async", for better readability when uploading files.
poll_async
-
Instance method. Polls server side for Ext.Direct Events, and passes the result to the callback function specified in arguments. Accepts named arguments in a hash.
In case of successful poll, i.e. when no errors have occured, the callback will receive an arrayref of event hashrefs:
$client->poll_async( ... cb => sub { my ($events, $success, $error) = @_; if ( $success ) { # Do something with event data for my $event ( @$events ) { ... } } } );
A poll may return an empty event list, in which case an empty arrayref will be passed to the callback. This is not an error.
Parameters:
cb
-
Mandatory callback function that will be fired when the poll finishes. See above.
cv
-
Optional condition variable to be signaled when the request is finished. See "Condition variables" for more detail.
-
Optional set of cookies for this particular call only. Cookies should be in the same format as for constructor, see "new".
CONFIGURATION OPTIONS
RPC::ExtDirect::Client::Async adds the following option specific to the Client, and sets it in the RPC::ExtDirect::Config instance it uses:
api_class_client
-
Class name to use when instantiating API objects from remote server JavaScript response. Default is
RPC::ExtDirect::Client::API
; use subclass name if you need to augment the stock API class behavior.
ACCESSOR METHODS
For RPC::ExtDirect::Client::Async, the following accessor methods are provided:
config
-
Return the current RPC::ExtDirect::Config instance held in the client object, or set a new one.
host
-
Return the current host name or IP address of the server, or set a new one.
port
-
Return the current port used to connect to the server, or set a new one.
-
Return the set of cookies to use with every Ext.Direct request, or set a new one. See "new" for more information.
http_params
-
Return a hashref with HTTP parameters to be used with every Ext.Direct request, or set a new one. These parameters will be passed to the HTTP request, see "http_request" in AnyEvent::HTTP.
By default this hashref will be populated with all "extra" arguments passed to "new" and does not need to be manipulated directly.
api_cb
-
Return the optional API retrieval callback function, or set a new one. See "API initialization" for more information.
api_ready
-
Will return truthy value if API has been successfully initialized, or falsy if it has not been initialized yet, or an error has occured while retrieving remote API declaration. See "exception" below.
exception
-
Will return the error that API retrieval resulted in, if any. The same error will be passed to the callback functions of all subsequent requests made through the Client object.
SEE ALSO
For more information on using Ext.Direct with Perl, see RPC::ExtDirect. For a blocking synchronous client, see RPC::ExtDirect::Client.
KNOWN ISSUES
AnyEvent::HTTP tries to reuse existing connection for idempotent requests, but for some obscure reason this does not always work as expected. As a consequence, first GET request (Ext.Direct API declaration retrieval) succeeds, while subsequent GET requests, e.g. event polling, will fail with 'Connection reset by peer' error. Which is not very helpful, to say the least.
To work around this issue, persistent connections has been disabled by default for all types of requests, including idempotent ones. If you are sure that this issue will not be a problem for you and want to use persistent connections, pass persistent => 1
parameter either to Client constructor, or to the individual "poll_async" calls.
Note that this issue does not apply to "call_async" or "submit_async" calls, since both of these are translated into HTTP POST requests which are not affected.
ACKNOWLEDGEMENTS
I would like to thank IntelliSurvey, Inc for sponsoring my work on this module.
BUGS AND LIMITATIONS
At this time there are no known bugs in this module. Please report problems to the author, patches are always welcome.
Use Github tracker to open bug reports, this is the easiest and quickest way to get your issue fixed.
COPYRIGHT AND LICENSE
Copyright (c) 2013-2014 Alex Tokarev <tokarev@cpan.org>.
This module is free software; you can redistribute it and/or modify it under the same terms as Perl itself. See perlartistic.