The Services Framework is new to OFBiz in
Version 2. Services are defined as independent pieces of logic which
when placed together can be used to process many different types of
business requirements. Services can be of many different types:
Workflow, Rules, Java, SOAP, BeanShell, etc. A service with the type
Java is much like an event where it is a static method, however with
the Services Framework we do not limit to web based applications.
Services require input parameters to be in a Map and the results are
returned in a Map as well. This is nice since a Map can be serialized
and stored or passed via HTTP (SOAP).
Services are defined through the Service Definition and are assigned to a specific Service Engine. Each Service Engine is responsible for invoking the defined service in an appropriate way. Since services are not tied to web based applications this allows services to run when there is no response object available. This allows services to be scheduled to run at specific times to run in the background via the Job Scheduler.
Services have the ability to call other services. So, chaining small services together to accomplish a larger task makes reusing existing services much easier. Services which are used in different applications can be defined only once by creating Global Service Definition files or services specific to an application can be restricted and available only to that application.
When used in a web application services are available to web events, which allow events to stay small and reuse existing logic in the Services Framework. Also, services can be defined as 'exportable' which means they are allowed to be accessed by outside parties. Currently there is a SOAP EventHandler which allows services to be made available via SOAP. Other forms of remote invocation may be added to the framework in the future.
The Service Dispatcher handles dispatching services to the appropriate Service Engine where it is then invoked. There is exactly one ServiceDispatcher for each Entity Delegator. If there are multiple delegators in an application there will also be multiple dispatchers. The ServiceDispatcher is accessed via a LocalDispatcher. There can be many LocalDispatchers associated with a ServiceDispatcher. Each LocalDispatcher is uniquely named and contains its own list of service definitions. When creating an instance of a LocalDispatcher, a DispatchContext is also created and passed to the ServiceEngine.
A LocalDispatcher is associated with an application. Applications never talk directly to the ServiceDispatcher. The LocalDispatcher contains an API for invoking services, which are routed through the ServiceDispather. However, applications may be running in different threads than the actual ServiceDispatcher, so it is left to the LocalDispatcher to keep a DispatchContext which among other things keeps a reference to the applications classloader.
The DispatchContext is created by the
LocalDispatcher upon instantiation. This is the runtime dispatcher
context. It contains necessary information to process services for each
dispatcher. This context contains the reference to each of the service
definition files, the classloader which should be used for invocation,
a reference to the delegator and its dispatcher along with a 'bag' of
user defined attributes. This context is passed on to each service when
invoked and is used by the dispatcher to determine the service's model.
This is where the service is actually invoked. Each service has an engine name assigned in its definition. This engine name is mapped via the servicesengine.xml file and is instantiated by the GenericEngineFactory when called upon. Third-party engines are supported and must follow the GenericEngine interface when implemented. See the Service Engine Configuration Guide for details on defining engines.
It is the job of the engine to handle invocation
of both synchronous and asynchronous services. Engines which use the Job Scheduler for asynchronous services can
extend the GenericAsyncEngine.
The overhauled job scheduler is now integrated with the services framework. This is the most appropriate place for a scheduler. Since it cannot be guarantee that an HttpServletRequest and HttpServletResponse object will be available when a job is ready to run, it does not make sense to integrate with the web controller. Plus, this feature is most useful when not limited to web environments.
The scheduler is a multithreaded component with a single thread used for job managing/scheduling and separate threads used for invocation of each service. When a job is scheduled to run, the scheduler will call the service dispatcher associated with the job to invoke the service in its own thread. This will prevent long or time consuming jobs from slowing down other jobs in the queue.
The scheduler now supports the iCalendar rule structure for recurrence. The jobs are no longer stored in an XML file and each is part of a ServiceDispatcher. There is one Job Scheduler for each ServiceDispatcher (which means there is only one per GenericDelegator as well).
How it works:
The best usage example of the scheduler is an asynchronous service call. When an asynchronous service is invoked, it is passed to the Job Scheduler to be queued to run. A recurrence entry is created (RecurrenceInfo and RecurrenceRule entities are created), the job is stored, (JobSandbox entity is created) and the context (Map) is serialized and stored (RuntimeData entity is created). The scheduler then adds the job to the top of the list of scheduled jobs (asynchronous services do not have any delay time) and invoked.
Jobs are no longer defined in an XML file. This has been moved to the JobSandbox entity. There is a web based client in planning for adding predefined jobs to the queue, but currently the entities will have to be created by hand.
Services are defined in Service Definition Files. There are global definition files used for all service dispatchers as well as individual files associated only with a single dispatcher. When a LocalDispatcher is created it is passed a Collection of Arils which point to these definition files. These files are composed using XML and define the necessary information needed to invoke a service. The DTD of this file can be found here.
Services are defined with a unique name, associated to a specific service engine and the input and output parameters are defined explicitly. Below is an example of a service definition:
<service name="userLogin" engine="java"
location="org.ofbiz.commonapp.security.login.LoginServices" invoke="userLogin">
<description>Authenticate a username/password; create a UserLogin object</description>
<attribute name="login.username" type="String" mode="IN"/>
<attribute name="login.password" type="String" mode="IN"/>
<attribute name="userLogin" type="org.ofbiz.entity.GenericValue" mode="OUT" optional="true"/>
</service>
SERVICE ELEMENT:
IMPLEMENTS ELEMENT:
ATTRIBUTE ELEMENT:
*underlined values are defaults
Above you can see the name of this service is userLogin and it uses the java engine. This service expects two required IN parameters: login.username and login.password. Required parameters are tested before the service is invoked. If the parameters does not match by name and object type the service is not invoked. Parameters which may or may not be sent to the service should be defined as optional. After the service is invoked, the OUT parameters are tested. Only required parameters are tested, however if a parameter is passed which is not defined as optional or required it will cause the service to fail. This service has not required OUT parameters, so the result is simply returned.
Internal Usage of the Services Framework is quite simple. In a Web Application the LocalDispatcher is stored in the ServletContext which can be accessed via the Session object in an Event. For non-web based applications you simply create a GenericDispatcher:
GenericDelegator delegator = GenericDelegator.getGenericDelegator("default");
LocalDispatcher dispatcher = new GenericDispatcher("UniqueName",delegator);
Now we have a dispatcher which we can use to invoke services. To invoke the test service create a Map for the context which contains the IN parameter message then invoke the service:
Map context = UtilMisc.toMap("message","This is a test.");
Map result = null;
try {
result = dispatcher.runSync("testScv",context);
}
catch (GenericServiceException e) {
e.printStackTrace();
}
if (result != null)
System.out.println("Result from service: " + (String) result.get("resp"));
Now look at the console and see what the test service has echoed.
*** The test service is located in core/docs/examples/ServiceTest.java You must compile this and place it in the classpath.
To schedule a service to run at a later time or to repeat use this:
// This example will schedule a job to run now.
Map context = UtilMisc.toMap("message","This is a test.");
try {
long startTime = (new Date()).getTime();
dispatcher.schedule("testScv", context, startTime);
}
catch (GenericServiceException e) {
e.printStackTrace();
}
// This example will schedule a service to run now and repeat once every 5 seconds a total of 10 times.
Map context = UtilMisc.toMap("message","This is a test.");
try {
long startTime = (new Date()).getTime();
int frequency = RecurrenceRule.SECONDLY;
int interval = 5;
int count = 10;
dispatcher.schedule("testScv", context, startTime, frequency, interval, count);
}
catch (GenericServiceException e) {
e.printStackTrace();
}
There have been a number of 'Advanced' features added to the service engine. You will find examples, definitions and information on each below.
The interface
service engine has
been implemented to help with defining services which share
a number of the same parameters. An interface
service
cannot be invoked, but rather is a defined service which other services
inherit from. Each interface
service will be defined
using the interface
engine, for example:
<service name="testInterface" engine="interface" location="" invoke="">
<description>A test interface service</description>
<attribute name="partyId" type="String" mode="IN"/>
<attribute name="partyTypeId" type="String" mode="IN"/>
<attribute name="userLoginId" type="org.ofbiz.entity.GenericValue" mode="OUT" optional="true"/>
</service>
**Note that the location
and the invoke
fields are required in the DTD, so we
simply leave these as empty strings when used as an interface
.
Now that we have an interface
we
need to define a service which implements
this interface
<service name="testExample1" engine="simple"
location="org/ofbiz/commonapp/common/SomeTestFile.xml" invoke="testExample1">
<description>A test service which implements testInterface</description>
<implements service="testInterface"/>
</service>
The testExample1
service will have
the exact same required and optional attributes as the
testInterface
service. Any service which implements testInterface
will also inherit
the parameters/attributes. If needed, additional attributes can be
added to a specific service by including
the attribute
tag along with the implements
tag. You may also override an attribute
by re-defining it after the implements
tag.
ECA (Event Condition Action) is much like a
trigger. When a service is called, a lookup is performed to see
if any ECAs are defined for this event
. Events include
before authentication, before IN parameter validation, before actual
service invocation, before OUT parameter validation, before transaction
commit, or before the service returns. Next each condition
in the ECA definition is evaluated and if all come back as true, each action
is performed. An action
is just a service which must be
defined to work with the parameters already in the service's context.
There are no limit to the number
of conditions or actions each ECA may define.
<service-eca>
<eca service="testScv" event="commit">
<condition field-name="message" operator="equals" value="12345"/>
<action service="testBsh" mode="sync"/>
</eca>
</service-eca>
eca
tag:
Attribute Name | Required? | Description |
service | Y | The name of the service this ECA is attached to. |
event | Y | The event on which this ECA will run can be (before): auth,
in-validate, out-validate, invoke, commit, or return . |
run-on-error | N | Should this ECA run if there is an error in the service (default false) |
The eca
element should also have 0
or more condition
or condition-field
elements and 1 or more action
elements.
condition
tag
Attribute Name | Required? | Description |
map-name | N | The name of the service context field that contains the map that the field to be validated will come from. If not specified the field-name will be treated as a service context field name (an env-name). |
field-name | Y | The name of the map field that will be compared. |
operator | Y | Specified the comparison operator must be one of the following: less, greater, less-equals, greater-equals, equals, not-equals, or contains. |
value | Y | The value that the field will compared to. Must be a String, but can be converted to other types. |
type | N | The data type to use for the comparison. Must be one of the following: String, Double, Float, Long, Integer, Date, Time, or Timestamp. If no type is specified the default will be String. |
format | N | A format specifier to use when converting String objects to other data types, mainly Date, Time and Timestamp. |
condition-field
tag
Attribute Name | Required? | Description |
map-name | N | The name of the service context field that contains the map that the field to be validated will come from. If not specified the field-name will be treated as a method environment field name (an env-name). |
field-name | Y | The name of the map field that will be compared. |
operator | Y | Specified the comparison operator must be one of the following: less, greater, less-equals, greater-equals, equals, not-equals, or contains. |
to-map-name | N | The name of the service context field that contains the map that the field to be compared will come from. If left empty will default to the map-name, or the method environment if map-name is also unspecified. |
to-field-name | N | The name of the to-map field that the main field will be compared to. If left empty will default to the field-name. |
type | N | The data type to use for the comparison. Must be one of the following: String, Double, Float, Long, Integer, Date, Time, or Timestamp. If no type is specified the default will be String. |
format | N | A format specifier to use when converting String objects to other data types, mainly Date, Time and Timestamp. |
action
tag
Attribute Name | Required? | Description |
service | N | The name of the service this action will invoke. |
mode | Y | The mode in which this service should be invoked. Can be sync or async. Note that async actions will not update the context even when set to true. |
result-to-context | N | Should the results of the action service update
the main service's context. Default true . |
ignore-error | N | Ignore any errors caused by the action service. If true
the error will cause the original service to fail. Default true . |
persist | N | The action service store / run. Can be true or false. Only
effective when mode is async. Default false . |
Service groups are a set of services which
should run when calling the initial service. You define
a service using the group
service engine, and include all
the parameters/attributes needed
for all the services in the group. The location
attribute
is not needed for group
services, the invoke
attribute defines the name of the group
to run. When this service is invoked the group is called and the
services defined in the group are called as defined.
The group definition is very simple, it contains
a group elements along with 1 or more service elements.
The group element contains a name
attribute and a mode
attribute used to define how
the group will function. The service element is much like the action
element in an ECA,
the difference being the default value for result-to-context.
<service-group>
<group name="testGroup" send-mode="all">
<service name="testScv" mode="sync"/>
<service name="testBsh" mode="sync"/>
</group>
</service-group>
group
tag
Attribute Name | Required? | Description |
name | Y | The name of the service this action will invoke. |
send-mode | N | The mode in which the service(s) should be invoked. The
options are: none, all, first-available, random, or round-robin. The
default is all . |
service
tag
Attribute Name | Required? | Description |
service | N | The name of the service this action will invoke. |
mode | Y | The mode in which this service should be invoked. Can be sync or async. Note that async actions will not update the context even when set to true. |
result-to-context | N | Should the results of the action service update
the main service's context. Default false . |
Route services are defined using the route
service engine. When a route
service is called, there is
no invoke performed, but all ECAs defined will run during the proper
events.
This type of service is not used very often, but can be used to 'route'
to other services by utilizing the ECA options available to services.
Using HTTP services is a way of invoking remote
services defined on other systems. The local definition should match
that of the remote definition, but the engine
used should
be http
, the location
should be a fully
qualified URL to the httpService event running on the remote
system, and the invoke
should be the name of the service
on the remote system you are requesting to be run. The remote system
must have the httpService event mounted for HTTP services to be
accepted. By default the commonapp web application has this event
mounted to receive requests for services. Services on the remote system
must have export
set to true
in order to
allow the service to be run remotely. HTTP services by nature are
synchronous.
JMS services are much like HTTP services, except
the service request is sent to a JMS topic/queue. The engine
should be set to jms
, the location
should
be the name of the jms-service defined in the serviceengine.xml file
(see Service
Config document), and the invoke
should be the name
of the service on the remote system
you are requesting to run. JMS services by nature are asynchronous.