NAME
RMI - Remote Method Invocation with transparent proxies
SYNOPSIS
#process 1: an example server on host "myserver"
use RMI::Server::Tcp;
my $s = RMI::Server::Tcp->new(port => 1234);
$s->run;
#process 2: an example client
use RMI::Client::Tcp;
my $c = RMI::Client::Tcp->new(
host => 'myserver',
port => 1234,
);
$c->call_use('IO::File');
$r = $c->call_class_method('IO::File','new','/etc/passwd');
$line1 = $r->getline; # works as an object
$line2 = <$r>; # works as a file handle
@rest = <$r>; # detects scalar/list context correctly
$r->isa('IO::File'); # transparent in standard ways
$r->can('getline');
ref($r) eq 'RMI::ProxyObject'; # the only sign this isn't a real IO::File...
# (see RMI::Client's use_remote() to fix this)
DESCRIPTION
RMI stands for Remote Method Invocation. The RMI modules allow one process to have virtual object "stubs" which are proxies for real objects in another process. When methods are invoked on the proxy, the method actually runs in the other process. When results are returned, those values may also be proxies for the real items in the other process. Parameters from the client are also automatically proxied on the server side during method execution.
In addition to invoking methods on proxy objects trasparently, an RMI::Client can invoke class methods, regular function calls, and other Perl functionaity on the remote server. Calls like these are typically the first step to obtain a remote object in the first place. This is different than implementations in other languages, which typically require that a server have limited and specific objects it returns, with all further proxying happening through them.
The procedure typically goes as follows:
1. a server is started which has access to some objects or data which is of value
2. a client connects to that server, and asks that it execute code on its behalf
3. the results returned may contain objects or other references, which the client recieves as proxies which "seem like" the real thing
4. further interaction with the returned proxy objects/refs automatically make calls through the client to the server internally
METHODS
The RMI module has no public methods of its own.
See RMI::Client and RMI::Server for detailed API information and examples.
See RMI::ProxyObject and RMI::ProxyReference for details on behavior of proxies.
See RMI::Node for internals and details on the wire protocol.
PROXY OBJECTS AND REFERENCES
A proxy object is an object on one "side" of an RMI connection which represents an object which really exists on the other side. When an RMI::Client calls a method on its associated RMI::Server, and that method returns a reference of any kind, a proxy is made on the client side, rather than a copy. The proxy object appears to be a reference to the real object, but internally it engages in messaging across the client to the server for all method calls, dereferencing, etc. It contains no actual data, and implements no actual methods calls.
By the same token, when a client passes objects or other references to the server as parameters to a method call, the server generates a proxy for those objects, so that the remote method call may "call back" the client for detailed access to the objects it passed.
The choice to proxy by default rather than generate a copy on the remote side by default is distinct from some remoting systems. It is, of course, possible to explicitly ask the server to serialize a given object, but because a serialized object may not behave the same way when it has lost its environment, this is not the default behavior.
Proxied objects are only revealed as such by a call to ref(), which reveals the object is actually an RMI::ProxyObject. Calls to isa() and can() are proxied across the connection to the remote side, and will maintain the correct API. Remote objects which implement AUTOLOAD for their API will still work correctly.
Plain proxied references, and as well as objects, are "tied" so as to operate as the correct type of Perl primitive. SCALAR, ARRAY, HASH, CODE and GLOB/IO references, blessed or otherwise, will be proxied as the same type of reference on the other side. The RMI system uses Perl's "tie" functionality to do this.
GARBAGE COLLECTION
Until a proxy is destroyed, the side which sent the item will keep an additional reference to the real item, both to facilitate proxying, and to prevent garbage collection. Upon destruction on the reciever side, a message is sent to the sender to expire its link to the item in question, and allow garbage collection if no other references exist.
DEBUGGING RMI CODE
When the RMI_DEBUG environment variable set to 1, the RMI modules will emit detailed information to STDERR during all "conversations" between itself and the remote side. This works for RMI::Client, RMI::Server, and anything else which inherits from RMI::Node.
This value is available inside the application as $RMI::DEBUG.
The package variable $RMI::DEBUG_MSG_PREFIX will be printed at the beginning of each message. Changing this value allows the viewer to separate both halves of a conversation. (The test suite for RMI sets this value to ' ' for the server side, causing server activity to be indented relative to client activity in the debug output.)
RMI_DEBUG=1 perl -I lib t/01_*.t
SECURITY
no inherent security is built-in
If you require restrctions on what the server provides, a custom sub-class should be written around the server to restrict the types of calls it will receive.
This is wise whenever the server is exposed to an untrusted network.
LIMITS TO TRANSPARENCY
calls to ref($my_proxy) reveal the true class RMI::ProxyObject
Proxied objects/references reveal that they are proxies when ref($o) is called on them, unless the entire package is proxied with ->use_remote.
Calls to ->isa() still operate as though the proxy were the object it represents. Code which goes around the isa() override to UNIVERSAL::isa() will circumvent the illusion as well.
remote objects do not stringify to matche the original object
Like ref(), this reveals the actual reference (and possibly class) of the proxy, not the object which is proxied.
calls to use_remote() does not auto-proxy all package variables
Calls to "use_remote" will proxy subroutine calls, but not package variable access automatically, besides @ISA. If necessary, it must be done explicitly with a call to bind().
the client may not be able to "tie" variables which are proxies
The RMI modules use "tie" on every proxy reference to channel access to the other side. The effect of attempting to tie a proxy reference may destroy its ability to proxy. (This is untested.)
In most cases, applications do not tie a variable created elsewhere because it destroys its prior value. As such this is unlikely to be a problem, but is still technically a hole in transparency.
change to $_[N] values will not affect the original variable
Remote calls to subroutines/methods which modify aliases in @_ directly to tamper with the caller's variables will not work as it would with a local method call.
This is supportable, but adds considerable overhead to support modules which create a side effect which is avoided because it is, mostly, a bad idea.
Perl technically passes an alias to even non-reference values, though the common "my ($v1,$v2) = @_;" makes a copy which safely allows the subroutine to behave as though the values were pass-by-copy.
sub foo {
$_[0]++; # BAD!
}
my $v = 1;
foo($v);
$v == 2; # SURPRISE!
If foo() were called via RMI, in the current implementation, $v would still have its original value.
Packages which implement this surprise behavior include Compress::Zlib! If this feature were added the overhead to Compress::Zlib would still make you want to wrap the call...
code which relies on caller() will probably fail
This means that some modules which perform magic during import() may not work as intended.
This problem is prevented in one place automatically by the current RMI implementation: there is custom code to handle exporting of methods into the caller's namespace inside "use_remote".
IMPLEMENTATIONS IN OTHER LANGUAGES
The use of transparent proxy objects goes by the term "RMI" in Java, "Drb" in Ruby, "PYRO" in Python, "Remoting" in .NET.
It is similar in functionality to architectures such as CORBA, SOAP, RPC and DCOM.
None of the above use the same protocols (except Java's RMI has an optional CORBA-related implementation). This module is no exception, sadly. Patches are welcome.
SEE ALSO
RMI::Server, RMI::Client, RMI::Node, RMI::ProxyObject, RMI::ProxyReference, SOAP, RPC
AUTHORS
Scott Smith <sakoht@cpan.org>
COPYRIGHT
Copyright (c) 2008 - 2009 Scott Smith <sakoht@cpan.org> All rights reserved.
LICENSE
This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
The full text of the license can be found in the LICENSE file included with this module.