Summary
So here's my pitch. I've been working on a grid system using Jini and mobile agents, for about 12-18 months, and now I'm nearly ready to release it as a Jini.org project.
Here's a bit of an overview to Neon's features and capabilities.
Advertisement
Neon is a framework for hosting mobile, collaborative objects. These objects
are modeled as agents; not agents in the JMX way, but mobile autonomous
agents.
Here are the main features of Neon, and I'll go through each of them in
turn, to give you a quick overview
Deploy your business objects into Neon
Multiple agent types
Partitioning and Domains
Deployment Suitability and runtime configurations
Manages life cycle of agents.
Provides a grid 'fabric' to your network
Provides automatic recovery, migration and agent transfer
facilities
Simplified Jini Distributed transaction support
Secure calling on context-provided agents
Run your Tomcat instance internally, with access to the grid.
Deploy your business objects into Neon
By modeling core parts of your logic as agents, you can deploy your code
into the system, and have it managed by Neon. You can allow the various
different parts of your application to work with each other through
their collaborative interfaces, even though they may exist on different
machines, or even change which machine they are on between one call and the
next.
Multiple Agent types
Neon provides a number of core agent interfaces, that you can implement, or
that mark out an agent in a particular way. By 'tagging' your agents with
different interfaces, and if required implementing the methods, your agent
can react to Neon in different ways.
The main agent types that we'll discuss here are:
Agent and AbstractAgent
LocalAgent
NonContinuousAgent
SensorAgent
TransactionalAgent
The Agent interface is the interface that makes your class an Agent in the
eyes of Neon, all agents must implement this interface, and unfortunately
there are a substantial amount of methods that need to be implemented; many
of these deal with lifecycle, identity and configuration. Most of the time,
however, you can probably make do with extending AbstractAgent. If you
extend Abstract Agent, you will have to only implement two methods, init()
and start().
LocalAgent marks an agent as being locally bound and therefore not
persistable at shutdown time, man of the utility agents such as ServiceAgent
are Local, and new instances are created through the normal bootstrapping
procedure.
NonContinuousAgent is there to mark your agents as departing from the normal
lifecycle of agents. In general, an Agent calls init(), then start(), and
even if start() returns straightaway, the agent remains registered and
maintained by the system, even though it's own execution thread has
completed, i.e. The agent is waiting to be told to do something or to be
notified of something. So in effect all agents are temporally continuous,
they will exist indefinitely, and if checkpointed regularly will exist even
beyond complete system failure (such as somebody pulling the power lead),
except in two cases i) Local agent, these agents are designed, as noted
above, to be run for the duration of the container, and no longer, and ii)
NonContinuousAgents are deregistered and removed from the system once
start() has completed.
SensorAgents allow other agents to have listeners registered on them and
will fire EventMessages to the registered parties, for instance, if a File
arrives in a directory that is being polled by an agent, making this Agent a
sensor will allow it to tell other agents about it, thus decoupling the
processing of the file, from the recognition that the file has arrived.
TransactionalAgents add two-phase commit semantics to your agent, thus you
need to implement commit(), abort() and prepare(). Transactions within Neon
are explained more fully below.
Partitioning and Domains
Neon provides logical partitioning of the system (but not a more rigid
partitioning via different classloaders, for example; Partitions do not set
their own classloaders). A partition is defined at startup, but as Neon
makes its way towards a 1.0 release, partition management will be required,
and even now is a fairly high priority. Partitions are logical groupings of
agents under a single category, how you define these categories is up to
you, they could be based on different applications you run (Orde
Processing), your departments (Accounts, HR), or even different needs of
your application (Persistence, Legacy Integration,etc). Partitions can talk
to each other if they are configured to do so. Each partition in a Neon
instance (or host) must have a unique name, although the same name may be
used in different instances, all the partitions across the network that
share the same name define a Domain.
Deployment Suitability
Before sending an agent into the system at runtime, you may want to specify
certain criteria about the instance that need to be met in order to deploy
the agent into it. This is not just about maintaining a level of service for
the agent, but also about limiting the effect of runtime errors that may
occur such as ClassNotFoundException, and in the case of deployment time
considerations such as COM integration or native libraries, ensuring that
the application you want is actually available on that machine etc. Agents
can specify some of their classes that need to be available to the instance,
what operating system the agent may need to run on, and free memory
requirements etc. Agents that don't meet their constraints don't get
deployed.
Each agent is also capable of having a Jini configuration attached to it
that it can then access at runtime, this can be anything from a URL, to a
configuration file stored with in a download jar.
Manages lifecycle
Neon can put agents into different states based upon what is happening to
them or user defined policies, or whether the agent is in a transaction,
etc. There are a number of Agent states, from Busy or Locked right up to
Hibernated and Death. Different states will effect your ability to
collaborate with an agent or even effect the physical location of an agent
at that point in time.
Provides a grid fabric
All Neon instances in a group can work together. The secret is in how the
methods on other agents are called from your agent. Neon forgoes the normal
view of using a conventional callstack for calling from one object to
another, and instead replaces it with a set of dynamic proxies, and a
messaging system called Zenith. Each message bus is dynamic allowing
channels to be added and removed at runtime, as agents join and leave the
partition. On the network is a message broker that is informed of each
addition and removal in the channel set by each message bus. The broker, in
turn, tells each and every other Neon instance in the group about the
channel, and which message bus service holds that channel. Thus when Neon
needs to call out of it's process, it simply looks up in its own table and
looks for the channel name it wants, and picks a corresponding service id to
route the message through.
Recovery, Migration and Transfer
As alluded to above, Neon is able to transfer objects from one instance to
another, however, there is another time when agents need to move - when
the instance is being shutdown. When you Ctrl-C an instance of Neon,
shutdown hooks are called, and all Non-local agents are stopped and
persisted to a JavaSpace, when the service is deregistered from the Jini
group, an agent is called on every other Neon instance, this agent is the
RecoveryAgent. The RecoveryAgent is woken up whenever an AgentService is
removed, when this happens, it starts to try to deploy and restart any
agents that were persisted during the shutdown phase. In the case, where
upon restarting the shutdown instance, agents that were stored into the
space are still there, these will be redeployed (subject to constraints, of
course)
Simplified Distributed Transaction support
I'll explain this with a bit of an example.
Normal Jini transaction
The first thing you'll notice is that we ask the TransactionAgent to create our transaction, however, we don't actually reference it, and then we don't use it to pass to the JavaSpaceOps agent .... but things are still written to the JavaSpace under a distributed transaction. Neon hides a lot of transaction processing, and assigns it's own txnID to the Jini Distributed Transaction Uuid, and it is possible to build agents that have 2PC methods that are called when the transaction is actioned. Agents that require access to the real distributed transaction, like JavaSpaceOps, can do so and then use that transaction to pass on to another service. All agents are capable of running under a Transaction (there is an exception - have your class implement NonTransactionalResource), but may not react to the transaction semantics (commit, abort, etc). Any agent running under a transaction will have it's state changed to LOCKED. Transactions are automatically propagated between agents and instances of Neon.
Secure calling on Agents
When you call context.getAgent(), you don't actually receive a reference to
the agent, you actually receive a dynamic proxy that dispatches method calls
as messages, this dynamic proxy only implements any interfaces that you mark
as being Collaborative, but importantly, it does not choose to implement the
Agent interface, thus you cannot do the following
Neon is capable of taking your existing Tomcat installation and running it
internally with the Neon process. Once this is done you can access the Neon
system and obtain agents from servlets and JSP's. Also Neon provides a
RenderAgent as one of it's Stock agents. Render Agent allows agents to
register a presentation (currently XSL stylesheets) for themselves, and the
Render Agent can then call back on the agent when required to get any data
that it wishes to present. The render agent then runs the data against XSL
stylesheet and can then return that to the caller-Neon comes with a very
small JSP that calls render on a provided agent ID and shows the results
back to the browser. An advantage of this approach is that this kind of
rendering can utilise the grid and potentially render pages on different
hosts, alleviating pressure on the Tomcat instance.
Well that's the introduction out of the way, once I've got the code and
binaries available, and out there for people to have a look at, I hope to
blog more on what Neon can do. Of course Neon is not going to do everything
for everyone - being a framework, certain compromises have to be made and
therefore, like every other framework, it's not a perfect fit for all
scenarios. Still, I hope some people like the sound of it and at least
have a look, and as with all projects - 'all help is gratefully received'. Most updates will appear on the Neon project page @ neon.jini.org.
<p>Hi Colin, <br>I am much interested to see where you're heading with this! I do like the concepts that you described. <p>One thing I am not clear about is the way the Grid Fabric would work. Have you put in some Master/Worker pattern on top of Neon which would make it easy to integrate in? <p> <p>Also, usually, objects need more than just jars to be deployed shared libs etc.. and require dependency handling so that they don't get repeated endlessly. have you looked into that?. <p> <p>I do like what you are mentioning Re: tomcat. I have been looking at Rio and I am not too keen on their way of handling the GUIs as it assumes that the object and its dependencies being viewed will eventually need to be streamed to the client which might not be possible and even if it is, is not efficient. <p> <p>Rather, I think the interface should be a thin client sitting on an embedded servlet container on the node where it is is more efficient and less intrusive in people's development. <p> <p>Finally, are you also putting provisionning features in Neon or would you rely on other frameworks (like Rio). <p> <p>Best regards, <p>Jean Safar