Revision 3.05
Since 3.04
Since 3.03
Since 3.02
This document describes the proposed design of the TAO Implementation Repository, which was originally known as the reactivator/activation service. If you have any questions or comments on our design, please post them to the ACE mailing list <ace-users@cs.wustl.edu> or send email to Darrell Brunsch <brunsch@cs.wustl.edu>.
CORBA defines two types of object references: persistent and transient. The difference between the two stems from the lifetime of the reference in relation to the lifetime of the server process that created it. The lifetime of a transient object reference is limited to the lifetime of its server process. Once the server process exits the transient object reference no longer exists. All references to this object should now be invalid, even if the server is restarted. In contrast, persistent object references can outlive their originating server process. Therefore, the server can exit and be restarted without invalidating its persistent object references. This enables the implementation of features like automatic server activation and object migration.
Note that both persistent and transient object references can refer to objects that reside in manually activated servers, i.e., the so-called ``persistent servers.'' A persistent server is a server that is launched manually, i.e., it is always running. A persistent server can generate transient references and/or persistent references.
Developers should be aware that persistence of the object reference does not imply any persistence on the object implementation state. It is certainly possible to provide persistent object references for objects whose state is not persistent. Therefore, servant implementors are responsible for preserving the state of their servants, e.g., using a database or file.
According to the CORBA specification, "The Implementation
Repository contains information that allows the ORB to locate and
activate implementations of objects" [CORBA Spec Rev. 2.2:
2.1.14] In earlier revisions of the specification, there was a
method get_implementation in the CORBA Object
interface. This has been deprecated as of the CORBA 2.2
specification, leaving both the interface and implementation of
the Implementation Repository to the ORB vendor.
A good paper describing the functionality of the CORBA Implementation Repository is "Binding, Migration, and Scalability in CORBA" [Michi Henning]. This paper describes the following three functions of the Implementation Repository:
The TAO Implementation Repository is based on the design in this paper. He also wrote an earlier email that may be useful to see the issues involved. The next section details our goals and plans for the implementation.
The following is an brief outline of TAO'S Implementation Repository.
LOCATION_FORWARD IIOP
        messages and multiple profiles in IORs, even if the
        client is not implemented using TAO.TAO's Implementation Repository must keep track of whether an object's implementation is currently running or is stopped. To have a record for every object would require too much overhead, but having a record for every executable server would be inflexible and prevent the migration of objects. In the Henning paper, he mentions the use of a server name as the index for the table maintained by the Implementation Repository.
A virtual server does not refer to the executable but instead to a group of objects. An executable may have one or more virtual servers within it. This allows one virtual server to be moved off the executable to another executable (for instance, onto another machine) without affecting the objects in other virtual servers on the original executable.
Each virtual server will be indexed in the Implementation Repository by a name that is given to it by the user. It is the users responsibility to make sure that each virtual server name is unique. By default, this name is the name of the executable (since by default there is only one virtual server per executable). However, this default behavior can be overridden.
Ping objects are simple objects that reside in the server, one for every virtual server. It is contacted by the Implementation Repository to determine if the virtual server is still running and responding. At certain intervals the Implementation Repository will invoke a one-way method on the ping object, and then will expect a "pong" message to be sent back. Different strategies for pinging (which will depend on a TAO option) will be used by the implementation repository. If a server is expected to be responsive, the Implementation Repository will not wait long for a response before considering the server to be gone. Other servers may be computationally-intensive and need to be held under less stringent expectations.
We chose the ping method to be a one-way (instead of two-way) because if the server became unresponsive, it would not return from the method invocation. The Implementation Repository needs some form of a timeout with the ping to be able to determine if the server is unresponsive or not.
Standard CORBA IORs contain the following two sections:
| Type ID | Sequence of Tagged Profiles | 
The Type ID is an indicator for the most derived type known at the time of the reference creation. It is used as a hint for the client in determining what interfaces the object can support. The Sequence of Tagged Profiles consist of one or more profiles that encapsulate information used by the associated protocol in order to communicate with the object (host, port, object id, etc.).
Currently, TAO uses only one IIOP 1.0 Tagged Profile, which is defined as follows:
| Version | Host | Port | Object Key | 
| Object Key: | 
 | 
To accomodate the Implementation Repository and IIOP 1.1, the Profile was changed according to the CORBA specification as follows:
| Version | Host | Port | Object Key | Components | 
| Object Key: | 
 | 
The two main changes is the addition of the Components field (which is a IIOP 1.1 thing) and the addition of TAO with its version. Transient object references will still have a TimeStamp to ensure uniqueness, but persistent object references will have a server name to identify themselves to the Implementation Repository.
For servers that move around or need to be restarted often, the IOR will contain a reference to the Implementation Repository with the object key of the server and the server name imbedded. Once the client contacts the Implementation Repository, it will be forwarded to the correct object. This IOR will look like the following:
| Version | Host | Port | Object Key | Components | 
| Object Key: | 
 | 
For servers that expect to remain in the same host/port for a long time, the above IOR can be optimized by placing the server profile in the IOR before the Implementation Repository profile. TAO clients will first try the server, and if that fails, then try the Implementation Repository. Clients from other ORBs may behave the same way, but this isn't guaranteed since the handling of multiple profiles is not yet in the CORBA spec. There will be an option to only generate the IR-only IORs for ORBs that do not support multiple profiles in the needed manner.
There will be a POA policy to determine which type of Persistent IOR to use. By default, the Implementation Repository alone version will be used.
We need a place to put a TAO marker in the IOR so TAO servers can differentiate TAO IORs from IORs of other vendors. In the original scheme used in TAO, Persistent IORs had a null timestamp. To support virtual servers, we will use that slot to store the server name so the Implementation Repository knows which server to launch.
It needs to know what the object key of the object when forwarding is used. A server may contain more than one object, so the object key is needed to forward to the correct object on the server.
TAO's POA will contain a new TAO-specific method called create_reference_with_virtual_server[_and_id]
(...). This method takes additional arguments for a
virtual server name and a sequence of Implementation Repository
IORs. The POA will register the virtual server name with each of
the Implementation Repositories in the sequence passed in.
Several Implementation Repositories can be specified to enhance
availability through redundancy. 
TAO's POA will also contain a policy for the type of IOR
created with create_reference.   It can either
produce the standard type, with just a reference to the
Implementation Repository, or it can produce one also containing
a reference to the current server.
The following are features that may be added to support TAO's Implementation Repository:
Most often servers that have Persistent IORs will save their state to secondary storage. Databases are a good example of this, where the server can be stopped and restarted with all the information remaining on disk.
The server must also make sure it creates the POA and Object in a way that does not change the POA ID and Object ID. The Implementation Repository forwards requests based on the information in the IOR; if the POA ID or Object ID changes, then the Implementation Repository will be unable to sucessfully forward requests. If the server implements dynamic servants and dynamic POA activations, then this is not an issue since the necessary POAs and servants will be created on demand.
The following is a proposed IDL interface for the TAO Implementation Repository:
module TAO
{
  // ....
  exception Already_Registered {};
  // Object already bound in the Implementation Repository
  exception Cannot_Activate
  {
    string reason_;
  };
  exception Not_Found {};
  // Object not found in the Implementation Repository  
  
  struct Environment_Variable
  {
    string name_;
    string value_;
  };
  // One environment variable
  
  struct INET_Addr
  {
    unsigned short port_;
    unsigned long host_;
  };      
  // The location of a server
  typedef sequence<Environment_Variable> Environment;
  // Complete environment
  typedef sequence<string> Command_Line_Options;
  // Command line options
  struct Process_Options
  {
    string executable_name_;
    // Executable name
    Command_Line_Options command_line_options_; 
    // Command line options
    Environment environment_; 
    // Environment
    string working_directory_;  
    // Working directory
    unsigned long creation_flags_;
    // Creation flags
  };
  interface Ping_Object
  {
    oneway void ping ();
    // Used for checking for liveness of a server.  When the server receives
    // this, it should send back a response indicating it is sill alive.  
    // Depending on the policy specified, a timeout can be reached where the
    // Implementation Repository will restart the server.
  };
  interface Implementation_Repository
  {
    Object activate_object (in Object obj)
      raises (Not_Found,
              Cannot_Activate);
      // Restart server that will contain this persistent object and return the
      // new Object reference.
      //
      // The <Not_Found> exception is raised when <obj> is not found
      // in the Implementation Repository.  The <Cannot_Activate> exception
      // is raised when <obj> is found in the Repository but could not be
      // activated.
    INET_Addr activate_server (in string server)
      raises (Not_Found,
              Cannot_Activate);
      // Restart server that is named <server> and return the host/port
      // 
      //
      // The <Not_Found> exception is raised when <server> is not found
      // in the Implementation Repository.  The <Cannot_Activate> exception
      // is raised when <server> is found in the Repository but could not be
      // activated.
    void register_server (in string server,
                          in Process_Options options)
      raises (Already_Registered);
      // Restart server process when client is looking for <server>.
      //
      // The <Already_Registered> exception is raised when <server> has
      // already been registered with the Implementation Repository.
      // 
      // The <Object_Not_Persistent> exception is raised when <server> is
      // not a Persistent Object Reference.
    void reregister_server (in string server,
                            in Process_Options options)
      raises (Already_Registered);
      // Restart server process when client is looking for <server>.
      //
      // The <Already_Registered> exception is raised when <server> has
      // already been registered with the Implementation Repository.
      // 
      // The <Object_Not_Persistent> exception is raised when <server> is
      // not a Persistent Object Reference.
    void remove_server (in string server)
      raises (Not_Found);
      // Remove <server> from the Implementation Repository.
      //
      // The <Not_Found> exception is raised when <server> is not found
      // in the Implementation Repository.
    Profile server_is_running (in string server,
                               in INET_Addr addr, 
                               in Ping_Object ping);
      // Used to notify the Implementation Repository that <server> is alive and
      // well at <addr>.
    void server_is_shutting_down (in string server);
      // Used to tell the Implementation Repository that <server> is shutting
      // down.
    };
};
Other ORB vendors use alternative techniques for their Implementation Repositories. These techniques usually require new naming techniques to access persistent object references and new client-side APIs to bind to persistent object references. TAO's Implementation Repository will not require such extensions.
A possible design alternative for the IR might use an Object Reference that points to the Implementation Repository instead of pointing directly to the persistent object. This extra level of indirection would be used by the Implementation Repository to start the server, if needed. The Location Forwarding mechanism would then be used to forward the client request to the server. The difference between this design and TAO's design is that the persistent IOR in TAO will contain a profile pointing to a location of the server (where it still might be running) to try first, and then only if that fails does the client contact the Implementation Repository. This is an optimization for case where the server does not shut down often, and most requests do not need to be forwarded to a new address.
In cases where most requests will require a forward, TAO can support a policy that is just like this alternative, where the Implmentation Repository will be contacted first.
The Implementation Repository will be transparent to the clients and the servers. Clients will only deal with IIOP 1.1 IORs, and in the default case all the Implementation Repository logic on the server side will be handled internally by the ORB and the POA.
A helper application will be included with the Implementation Repository. It will be a command-line utility that will assist users with adding and removing server records (containing virtual server names and executable name/options) from the Implementation Repository.
In the default case, the Implementation Repository will be
found via the command-line, environment variables, and multicast
(in that order). This location strategy is consistent with that
used by TAO to local its default Naming Service instance. Using
the POA extensions, other Implementation Repositories can be
specified in the call to POA::create_reference_with_virtual_server.
The default port of the Implementation Repository can be
overridden through command-line options or environment variables.
One or more Implementation Repositories will be stored in additional profiles in the IOR. Other Implementation Repositories can also be located by multicasting (on a default multicast group) the server name of the Persistent Object the client is interested in. The default multicast group and default port of the Implementation Repository can be overridden through command line options or environment variables.
In most cases, one Implementation Repository will be enough. For redundancy, several Implementation Repositories can be specified.
Before a server starts, it must be registered (via a command-line utility) with an implementation repository. On platforms that don't support multicast, the Implementation Repository must be specified on the command line or in an environment variable.
ORB_init.
        ORB_init, if not passed a server name, will
        take argv[0] and use that as a default server name (TAO
        expects this to be the executable name). ORB_init will create a ping object.ORB_init will look for Implementation
        Repositories on the command-line, environmental
        variables, and then through multicast (in that order).
        Once it finds one it registers itself and passes the ping
        object to the implementation repository with server_is_running
        operation. POA::create_reference
        operation.create_reference operation will create
        the local profile.As with the default case, the server must be registered with an Implementation Repository, although it does not need to be multicast aware since the IORs will be passed to the POA by the program.
ORB_init is called and does the default work
        (if it has Implementation Repositories to contact).POA::create_reference_with_virtual_server[_and_id]
        will be called with a server name and list of
        Implementation Repositories. ORB_init and the
        object key is passed to the Implementation Repositories,
        and their profiles are returned.For all Clients:
If everything fails, then most clients will return failure for the request. TAO clients will attempt to contact other Implementation Repositories that are specified on the command-line, in environment variables, or found through multicast.
TAO clients will have an optimization where if there are several IORs that have the same server name, and one of them gets forwarded, then the client will be able to change its other IORs without going through the overhead of contacting Implementation Repository.
Last update to this document: $Date$
Back to TAO Release Notes
Back to Implementation Repository