Difference between revisions of "Northwind Tutorial"

From RifidiWiki

Jump to: navigation, search
(Sending out events with JMS)
 
(94 intermediate revisions by 2 users not shown)
Line 1: Line 1:
 +
= '''WARNING: This document is for SDK 1.1.  For 1.2, please refer to the Northwind instructions documented in the Developer's  PDF in the SDK''' =
 +
 +
[[Developer's Guide]]
 +
 
This document provides step-by-step instructions on how to get started on developing your first application that runs on the Rifidi Edge Server.  The application we will develop will use Esper to collect tag reads from a reader and put them on a JMS queue to be consumed by a client application.  Many of the steps are applicable to many kinds of plugins for the edge server, including [[How to create a sensor plugin| creating a sensor plugin]].
 
This document provides step-by-step instructions on how to get started on developing your first application that runs on the Rifidi Edge Server.  The application we will develop will use Esper to collect tag reads from a reader and put them on a JMS queue to be consumed by a client application.  Many of the steps are applicable to many kinds of plugins for the edge server, including [[How to create a sensor plugin| creating a sensor plugin]].
 
=The Scenario: Northwind Shipping=
 
=The Scenario: Northwind Shipping=
Line 20: Line 24:
  
 
To get started see [[Edge Server Development Environment|Setting up a Development Environment]].
 
To get started see [[Edge Server Development Environment|Setting up a Development Environment]].
 +
=Source=
 +
You can download the source for this project here: [[Image:Northwind_1.0.0.zip]]. Just unzip the project, open up eclipse and select import->existing projects into workspace.
  
=Writing the RFID Application=
+
=Outline=
 
+
#[[Northwind Creating the Application | Step 1: Create the Application]]
 
+
#[[Northwind Using Emulator | Step 2: Using Emulator]]
==Creating the new project==
+
#[[Northwind Esper: Track Packages| Step 3: Esper: Track Packages]]
First thing you need to do is to create a new OSGi bundle (aka plug-in) project using the wizard provided by eclipse.
+
#[[Northwind Esper: Alerts| Step 4: Esper: Alerts]]
# Go to File-> New -> Project.  Select "Plug-in Project" from the file chooser. Click "Next."<br> [[Image:Tutorial-newwizard1.png|thumb|400px|none]]
+
#[[Northwind JMS Send | Step 5: Send Notifications Over JMS]]
# Assign the project a name (for the purposes of this tutorial, I went with "com.northwind.rfid.shipping").  Make sure the plugin is targeted to run with the standard OSGi framework.  This makes sure that your plugin is general enough to be deployed in any OSGi runtime (felix, knoplerfish, etc). Click next.<br>[[Image:tutorial-newwizard2.png|thumb|400px|none]].
+
#[[Northwind Creating the Web Application | Step 6: Create the Web Application]]
#Assign the plugin an ID, which is used to identify the bundle in the OSGi environment.  Give it a descriptive name as well as the the name of the entity (company, person, project) that will maintain the bundle.  For this bundle, we will keep the Activator, although most of the time when you use spring DM (as we are), it is not necessary to have the activator.  Click Finish.<br>[[Image:tutorial-newwizard3.png|thumb|400px|none]]
+
#[[Northwind Hello World Servlet | Step 7: Write a Hello World Servlet]]
 
+
#[[Northwind TagLocationService | Step 8: Write a Tag Location Service and JMS Listener]]
==Injecting Esper==
+
#[[Northwind Display Events using JSP: Step 9: Display events in a JSP]]
What you will learn:
+
* Declaring OSGi bundle dependencies in the manifest
+
* Dependency injection via spring
+
* Running an OSGi application in eclipse
+
* Redeploying bundles without restarting the server
+
==Stating Dependencies in the Manifest==
+
 
+
{| cellspacing="5" align="right" style="margin: 1em auto 1em auto;background-color:rgb(240, 247, 255); width=100px;border:1px dashed; margin-left:10px; margin-top:10px; margin-bottom:10px"
+
| align="center" width="350px"| '''ClassNotFoundException'''
+
|-
+
| width="350px"|Sometimes when running a bundle, you will get a ClassNotFoundException even though there were no compile-time errors in your source code.  Many times, this is easily solved by adding the package of the problematic class to the import-package statement in the Manifest of your bundle.
+
|}
+
 
+
An OSGi bundle is simply a jar with some extra information in the manifest.  Part of that information is the dependencies of the bundle.  There are two ways to state dependency information: bundle-dependencies, and package-dependencies.  A bundle dependency means that the bundle you are creating can see all of the exported package of the bundle that is depended on.  A package dependency means that some bundle in the OSGi runtime must export that package, but it doesn't matter which bundle.  Bundle dependencies are often simpler to state if you will use a large number of packages from the same bundle.  However, package dependencies are more flexible since the dependency can be met from any bundle.  They make alot of sense for things like javax or apache commons packages since you generally don't know or care exactly which bundle you will use.
+
 
+
In order to get Esper running, we will need to make two bundle-dependencies in the Manifest.
+
#Open up the Manifest.MF file in the META-INF folder
+
#Click on the dependencies tab at the bottom of the editor.
+
#Click "Add" in the Required Plug-ins section.
+
#Add <tt>org.rifidi.edge.core.services</tt> and <tt>org.rifidi.com.espertech.esper</tt>.
+
#Click "Add" in the Imported Packages section.
+
#Add <tt>net.sf.cglib.reflect</tt>
+
#Save the changes.
+
[[Image:tutorial-manifest.png|thumb|none|400px]]
+
 
+
===Creating the Application===
+
Now it's time to actually create the source code for the application. 
+
#Right-click on the 'com.northwind.rfid.shipping' package and select New->Class
+
#Give the class a name.  I chose ShippingApp. Click finish.
+
#Add a private member of type <tt>EsperManagementService</tt>.  Also make it volatile since the Spring thread will inject the reference (actually a dynamic-proxy) to the object.  The volatile keyword will prevent us from experiencing the [http://jeremymanson.blogspot.com/2008/11/what-volatile-means-in-java.html visibility problem] in java.
+
#Add a public setter method. to set the service.  This is the method that spring will call when it creates our object.
+
#Add a constructor. Add a printline in the constructor so we know when the object was started.
+
<pre>
+
package com.northwind.rfid.shipping;
+
 
+
import org.rifidi.edge.core.services.esper.EsperManagementService;
+
 
+
/**
+
* @author Kyle Neumeier - kyle@pramari.com
+
*
+
*/
+
public class ShippingApp {
+
+
/**Esper service*/
+
private volatile EsperManagementService esperService;
+
 
+
/**
+
* Constructor
+
*/
+
public ShippingApp() {
+
System.out.println("HELLO RFID WORLD!");
+
}
+
 
+
/**
+
* Called by spring
+
* @param esperService
+
*/
+
public void setEsperService(EsperManagementService esperService) {
+
this.esperService = esperService;
+
}
+
}
+
 
+
</pre>
+
 
+
===Creating the Spring Context XML===
+
Next we need to create the spring context XML file.  Spring is an application development framework for java; it relieves the amount of boiler-plate code that is normally required when writing applications.  In addition, through spring dynamic modules, spring is integrated with OSGi to help start up bundles and access the OSGi service registry among other things.
+
 
+
Rifidi Edge Server applications should use spring to create objectes, register them as services if need be, and look up services from the OSGi service registry and inject them into their objects.  To create a spring context xml:
+
#Create a folder called 'spring' in the META-INF folder.  Spring will read any xml files in this folder in and use it when starting up the bundle.
+
#Create an xml file. By convention, I call mine spring.xml
+
#Copy and paste the following xml into the file.
+
<pre>
+
<beans xmlns="http://www.springframework.org/schema/beans"
+
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+
xmlns:osgi="http://www.springframework.org/schema/osgi"
+
xmlns:amq="http://activemq.apache.org/schema/core"
+
xsi:schemaLocation="http://www.springframework.org/schema/beans
+
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
+
    http://www.springframework.org/schema/osgi
+
    http://www.springframework.org/schema/osgi/spring-osgi.xsd">
+
 
+
<!-- Create the  Application bean and inject dependencies-->
+
<bean id="rfidapp" class="com.northwind.rfid.shipping.ShippingApp">
+
<property name="esperService" ref="esperManagementService" />
+
</bean>
+
 
+
<!-- Get a reference to the Esper Management Service from the OSGi Service Registry -->
+
<osgi:reference id="esperManagementService"
+
interface="org.rifidi.edge.core.services.esper.EsperManagementService" />
+
</beans>
+
</pre>
+
 
+
The first <tt>beans</tt> tag gives the xml processor the location of the xsds that define the namespace information.  This allows us to use things like <tt>bean</tt> and <tt>osgi</tt> in the xml tags.  This makes for cleaner, easier to read files.
+
 
+
The <tt>osgi:reference</tt> tag tells spring to look up a service in the OSGi service registry which implements the <tt>org.rifidi.edge.core.services.esper.EsperManagementService</tt> interface.  Assign the object to id 'esperManagementService'. 
+
 
+
The second <tt>bean</tt> tag tells spring to instantiate an object of type <tt>com.northwind.rfid.shipping.ShippingApp</tt>.  In addition, we need to inject a bean that is referenced by the ID 'esperManagementService' (which we looked up in the previous tag). To set the service, use a setter method called "setEsperService" (set+<nameProperty>) which exists in the bean.
+
 
+
===Running the Application===
+
Now we are ready to run the application.  To do this, we need to modify the run configuration slightly in eclipse.
+
# Go to Run-> Run Configurations...
+
# Select the "Edge Server" run configuration under 'OSGi Framework'
+
# Select the Bundles tab.  You will see a list of bundles.  Each bundle that has a check mark by it will be installed and started when you run the application.  The bundles that serve as the 'core' Rifidi Edge Server have already been selected.
+
# Under 'Workspace', check the 'com.northwind.rfid.shipping' bundle.<br>[[Image:Tutorial-RunConfiguration.png|thumb|400px|none]]
+
# Click "Apply". Click "Run".
+
 
+
Now you should see "HELLO RFID WORLD!" appear in the console, along with other debug information.
+
 
+
===Modifying and Redeploying the Application===
+
Now suppose you want to modify your application to print a message when the EsperMangemnetService is injected.  You can do this by modifying your application as follows.
+
<pre>
+
/**
+
* Called by spring
+
* @param esperService
+
*/
+
public void setEsperService(EsperManagementService esperService) {
+
this.esperService = esperService;
+
System.out.println("ESPER SERVICE INJECTED!");
+
}
+
</pre>
+
 
+
Now its time to test out your changes.  One thing you could do is shut down the whole edge server and restart it. All the time spent starting and stopping the edge server can add up.  One of the benefits of OSGi is being able to start,stop,and update individual bundles without restarting the rest of them.  You can use this feature when developing to help cut down on wasted development time.
+
 
+
# Make sure the edge server is started inside eclipse
+
# Type 'ss'.  This is an equinox command to list every bundle that is currently installed in the OSGi system.
+
# Locate 'com.northwind.rfid.shipping' in the list.  Remember the number next to it.  For me, the number is '3'.<br>[[Image:Tutorial-ss.png|thumb|none|765px]].
+
# Now type 'update <bundle_number>'. This is the equinox command to update and restart the given bundle.
+
# You should see both printlines appear.
+
 
+
==Event Stream Processing==
+
What you will learn:
+
* Using Rifidi Emulator to speed up RFID development.
+
* Controlling the edge server from the command line.
+
* Writing Esper queries and listeners.
+
===Running Rifidi Emulator===
+
Rifidi Emulator is a developer tool that emulates the interfaces of many popular RFID readers.  It can be used to speed up development of RFID applications, since you can write applications that talk to RFID readers without the need for access for a physical reader.  To get Rifidi Emulator up and running follow the steps in the [[Emulator User's Guide]].
+
 
+
For the purposes of this tutorial, create two Alien ALR readers.  The first reader will represent the Dock Door.  Assign it IP address <tt>127.0.0.1:20000</tt>  The second reader will represent the weigh station.  Asssing it <tt>127.0.0.1:20001</tt>.  Create a few tags (does not matter which kind).  Drag and drop the tags onto the antenna.  Start the readers.
+
 
+
Instead of creating the readers yourself, you can load the configuration from [[Image:Northwind-emulator.rfts.zip| this file]].  Simply unizp it and choose File->Open IDE Configuration and choose the rtfs file.
+
 
+
[[Image:Tutorial-Emulator.png|thumb|none|600px]]
+
 
+
===Connecting to Emulator from the Edge Server===
+
There are two ways (currently) to create a connection to a reader using the edge server.  One is to use [[Workbench User's Guide|Workbench]]. 
+
 
+
However, when developing applications on the edge server, it is often quicker to use the [[Edge Server Console]].
+
 
+
Once the edge server is up and going, type the following commands into the console:
+
#<tt>readertypes</tt>. This lists the kind of reader adapters available.  You should see one called <tt>Alien9800</tt>
+
#<tt>createreader Alien9800</tt>. This makes a new instance of a reader configuration with the default properties (including the IP & port, which by default is <tt>127.0.0.1:20000</tt> for the Alien).  This will connect to the Dock Door, for the purposes of this demo.  You should see output indicating the sensor was created along with it's ID (probably <tt>Alien9800_1</tt>).
+
#<tt>readers</tt>.  This lists the created readers.  You should see your reader listed here.
+
#<tt>createsession Alien9800_1</tt>.  This creates a session that you will use to connect to Dock Door Reader.
+
#<tt>commandtypes</tt> This lists the available kind of commands you can execute.
+
#<tt>createcommand Alien9800-GetTagList</tt>.  Like <tt>createreader</tt> this creates a new configuration with default properties, except this time its a command configuration.
+
#<tt>executecommand Alien9800_1 1 Alien9800-GetTagList_1 1000</tt>.  This tells the edge server to schedule a Alien9800-GetTagList command on session 1 of Alien9800_1 once every second.
+
#<tt>startsession Alien9800_1 1</tt> This tells the session to make the TCP/IP connection to the Dock Door Reader.  At this point you should see activity on the console of the emulator.
+
#<tt>createreader Alien9800 Port 20001</tt>. This makes a new instance of a reader configuration that connects to <tt>127.0.0.1:20001</tt> (which is the weigh station, for the purposes of this demo).
+
#<tt>createsession Alien9800_2</tt>.  This creates a session that you will use to connect to Weigh Station Reader.
+
#<tt>executecommand Alien9800_2 1 Alien9800-GetTagList_1 1000</tt>. You can execute the command that you've previously created on the second session.
+
#<tt>startsession Alien9800_2 1</tt> This tells the session to make the TCP/IP connection to the Weigh Station Reader. 
+
#<tt>save</tt> - This saves the configurations so that you don't have to type all these commands again.
+
 
+
Once you make sure this configuration is working, you can stop the sessions for now.  Just type
+
# <tt>stopsession Alien9800_1 1</tt> - Stop the Dock Door Reader
+
# <tt>stopsession Alien9800_2 1</tt> - Stop the Weigh Station Reader
+
 
+
===Preparing for Esper Queries===
+
 
+
Esper is an event processing language and run time.  It allows you to define queries in an SQL-like syntax that operate on events rather than the table in a database.  It is useful to think of esper queries as SQL queries turned upside down: rather than on-demand queries that operate on static data, you define a relatively static query and the data flows through the query as it is produced.  For more information see [[How to write esper queries]].
+
 
+
For this tutorial, we will define a simple query that returns all data produced by a certain reader.  Modify <tt>MyApplication.java</tt> to add the following changes:
+
* A private Set to keep up with all esper queries that we will define
+
* Give a reference of this to the Activator (more on this later)
+
* A start method where the esper queries will be submitted to the esper runtime
+
* A stop method that will clean things up.  This will be called by the Activator.
+
<pre>
+
package com.northwind.rfid.shipping;
+
 
+
import java.util.Set;
+
import java.util.concurrent.CopyOnWriteArraySet;
+
 
+
import org.rifidi.edge.core.services.esper.EsperManagementService;
+
 
+
/**
+
* @author Kyle Neumeier - kyle@pramari.com
+
*
+
*/
+
public class ShippingApp {
+
 
+
/** Esper service */
+
private volatile EsperManagementService esperService;
+
/**All statements that have been defined so far*/
+
private final Set<EPStatement> statements = new CopyOnWriteArraySet<EPStatement>();
+
 
+
/**
+
* Constructor
+
*/
+
public ShippingApp() {
+
Activator.myApp=this;
+
}
+
 
+
/**
+
* Called by spring
+
*
+
* @param esperService
+
*/
+
public void setEsperService(EsperManagementService esperService) {
+
this.esperService = esperService;
+
start();
+
setUpAlerts();
+
setupListeners();
+
}
+
+
/**
+
* A method that starts the application
+
*/
+
private void start(){
+
//TODO: Esper Statements will go here!
+
 
+
}
+
 
+
/**
+
* A method that sets up listeners to handle esper events
+
*/
+
private void setupListeners(){
+
//TODO: Esper Listeners will go here!
+
 
+
}
+
 
+
/**
+
* A method that sets up business alerts
+
*/
+
private void setUpAlerts(){
+
//TODO: Alerts will go here!
+
}
+
+
/**
+
* Iterate through all statements and stop them.
+
*/
+
protected void stop(){
+
for(EPStatement statement : statements){
+
statement.destroy();
+
}
+
}
+
}
+
</pre>
+
 
+
In addition, you will need to edit the Activator file to look like this:
+
<pre>
+
/**
+
* The Activator for the bundle. The start() method is called when the bundle is
+
* starting up. The stop method is called when it is shutting down
+
*
+
* @author Kyle Neumeier - kyle@pramari.com
+
*
+
*/
+
public class Activator implements BundleActivator {
+
 
+
/** A reference to the instance of MyApplication */
+
protected volatile static MyApplication myApp;
+
 
+
/*
+
* (non-Javadoc)
+
*
+
* @see
+
* org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext
+
* )
+
*/
+
public void start(BundleContext context) throws Exception {
+
}
+
 
+
/*
+
* (non-Javadoc)
+
*
+
* @see
+
* org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)
+
*/
+
public void stop(BundleContext context) throws Exception {
+
// stop MyApplication when the bundle shuts down
+
myApp.stop();
+
}
+
 
+
}
+
</pre>
+
The key changes:
+
# The start() method is called after the EsperService is injected by spring.
+
# A stop method (which cleans up the statements that you will write from the Esper runtime) is called when the bundle stops.  This happens, because you gave the reference of the instance of the application to Activator.  The stop() method is called on the Activator by the OSGi runtime when that bundle is shutting down.  Cleaning up the statements allows us to update the bundle without restarting the whole edge server.
+
 
+
===Tracking the packages===
+
{| cellspacing="5" align="right" style="margin: 1em auto 1em auto;background-color:rgb(240, 247, 255); width=100px;border:1px dashed; margin-left:10px; margin-top:10px; margin-bottom:10px"
+
| align="center" width="350px"| '''Class Hierarchy of Events'''
+
|-
+
| width="350px"|Esper is similar to SQL in syntax.  However, with Esper, you query streams of objects rather than tables in a database. You can then [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_representation.html query the objects] using the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_representation.html#eventrep-javabeangetter and setter methods] in the objects The Rifidi Edge Server, by default, will put objects of type <tt>ReadCycle</tt> into the Esper engine. Therefore, your queries will be focused on querying <tt>ReadCycle</tt> objects, and it is important to understand the class hierarchy of the events so that you will know what you can query for in esper. Please see [[ReadCycle Class Hierarchy]] for more information.
+
|}
+
 
+
The first and foremost goal of the application is to track the packages from the dock door to the weigh station.  We will do this by creating two [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#named_overview named windows] in esper: one that keeps track of the tags that can currently be seen at the dock door.  The second keeps track of tags currently seen at the weigh station.  Think of a named window as an empty bucket for events.  When a package is seen at the dock door, we need to insert the ID of the package into the dock door bucket.
+
 
+
The tougher question is to figure out when the package has departed from the dock door.  Because of the physics of RFID, sometimes a tag is not read by the RFID reader every single time it polls its antennas, even though the item is still within the read zone.  Depending on how we configure the RFID reader, it might be giving us several tag reports a second. We don't want to flood our system with false arrival and departure events, so let's assume that if a tag has not been seen for 10 seconds, the chances are good that it has moved from the dock door read zone (hopefully on its way to the weigh station!), and we need to remove it from the bucket.
+
 
+
The final part of this requirement is that we need to be notified in some way as packages arrive and depart.  We do this through creating listeners that are hooked up to queries.  The listeners allow us to "handle" the event in some way.  For now, we will just print out a message (Later we can put a message on a JMS queue in order to inform some higher power).  The key thing to note about the listeners is that we configured them to be notified when things are inserted into the "buckets" (i.e. named windows) and when things are deleted from the "buckets". 
+
 
+
====The Code====
+
Here is a closer look at the esper statements required for this step:
+
<pre>
+
/**
+
* A method that starts the application
+
*/
+
private void start(){
+
+
// 1 create a named window that handles events at the Dock Door
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"create window dockdoor.std:firstunique(tag_ID) (tag_ID String)"));
+
+
// 2 create a named window that handles events at the Weight Station
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"create window weighstation.std:firstunique(tag_ID) (tag_ID String)"));
+
+
// 3 Insert information into the named windows
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"on ReadCycle[select * from tags]" +
+
"insert into dockdoor select cast(tag.epc?, String) as tag_ID where readerID = 'Alien9800_1'" +
+
"insert into weighstation select cast(tag.epc?, String) as tag_ID where readerID = 'Alien9800_2'"));
+
+
// 4 Remove events from the Dock Door Named Window
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"on pattern [every tag=dockdoor ->" +
+
"(timer:interval(10 sec) and not dockdoor(tag_ID = tag.tag_ID))]" +
+
"delete from dockdoor where tag_ID = tag.tag_ID"));
+
+
// 5 Remove events from the Dock Door Named Window
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"on pattern [every tag=weighstation ->" +
+
"(timer:interval(10 sec) and not weighstation(tag_ID = tag.tag_ID))]" +
+
"delete from weighstation where tag_ID = tag.tag_ID"));
+
}
+
+
/**
+
* A method that sets up listeners to handle esper events
+
*/
+
private void setupListeners(){
+
+
// 6 create a listener to handle Dock Door events
+
StatementAwareUpdateListener dockDoorListener = new StatementAwareUpdateListener() {
+
@Override
+
public void update(EventBean[] arrivals, EventBean[] departures, EPStatement arg2,
+
EPServiceProvider arg3) {
+
if(arrivals!=null){
+
for(EventBean bean : arrivals){
+
System.out.println("Dock Door Arrival: " + bean.get("tag_ID"));
+
}
+
}
+
+
if(departures!=null){
+
for(EventBean bean : departures){
+
System.out.println("Dock Door Departure: " + bean.get("tag_ID"));
+
}
+
}
+
+
}
+
};
+
+
// 7 Create a query that is triggered on insert and remove events from Dock Door Window
+
EPStatement queryDockDoor = esperService.getProvider().getEPAdministrator().createEPL(
+
"select irstream * from dockdoor");
+
queryDockDoor.addListener(dockDoorListener);
+
statements.add(queryDockDoor);
+
+
// 8 Create a listener to handle Weigh Station Events
+
StatementAwareUpdateListener weighStationListener = new StatementAwareUpdateListener() {
+
@Override
+
public void update(EventBean[] arrivals, EventBean[] departures, EPStatement arg2,
+
EPServiceProvider arg3) {
+
if(arrivals!=null){
+
for(EventBean bean : arrivals){
+
System.out.println("Weigh Station Arrival: " + bean.get("tag_ID"));
+
}
+
}
+
+
if(departures!=null){
+
for(EventBean bean : departures){
+
System.out.println("Weigh Station Departure: " + bean.get("tag_ID"));
+
}
+
}
+
+
}
+
};
+
+
// 9 Create a query that is triggered on insert and remove events from Weigh Station Window
+
EPStatement queryWeighStation= esperService.getProvider().getEPAdministrator().createEPL(
+
"select irstream * from weighstation");
+
queryWeighStation.addListener(weighStationListener);
+
statements.add(queryWeighStation);
+
}
+
</pre>
+
====Explanation of the Code====
+
# Create a named window "bucket" for the dock door.  Notice two important things:
+
## We used a built in view for this window called [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl-views.html#view-std-firstunique <tt>firstunique</tt>], which (as the name implies, only retains the first unique entry for the filter value.  This is important because the RFID reader is constantly sending us new tag reads every second, many of which could be the same tag over and over.  If we did not specify firstunique, we would have a bucket full of duplicate tag reads!
+
## We [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#named_create_explicit specified the kind of information] that can go into the bucket - A string called <tt>tag_ID</tt>.  Another way to think about a named window is a database table.  When we created the table, we created it with one column called tag_ID of type String.
+
#Create a named window for the weigh station.  We create it the exact same way we create the previous one.
+
#Here is where we are filling the buckets up with tag data. There are a few important concepts here:
+
## Because the <tt>ReadCycle</tt> may have many individual events inside of it, we use the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#epl-containedeventselect Constrained-Event Selection] mechanism to only return the tags inside of each <tt>ReadCycle</tt>
+
## We use the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#named_querying <tt>on select</tt>] clause to trigger the inserts whenever a new <tt>ReadCycle</tt> event arrives.
+
## We use the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#epl-insert-into <tt>insert into</tt>] clause to fill the named windows
+
## Inside the select clause, you will notice this: <tt>cast(tag.epc?, String)</tt>. Without getting into too much detail about the object-oriented structure of <tt>ReadCycle</tt>, it is not known at compile time whether or not the events inside of a <tt>ReadCycle</tt> will have a property called <tt>epc</tt> or not (this generalization allows us to process tags that are not EPC Gen2 Tags).  Because of this uncertainty, we have to use the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_representation.html#eventrep-dyncproperties duck-typing] feature (which is signified by the '?').  In addition, we use the  [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/functionreference.html#epl-single-row-function-cast cast function] of esper.
+
# Here we delete the tags from the dock door window
+
## Here we use the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#named_delete <tt>on delete</tt>] clause to remove tags.
+
## The pattern uses the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_patterns.html#pattern-logical-every <tt>every</tt>] operator to indicate that the pattern should be repeated and not just fire once and stop.
+
## We use the [[http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_patterns.html#pattern-temporal-followed-by <tt>-></tt> (followed by)] operator to indicate that once we have seen a new tag arrive at the Dock Door, we are interested in an event that will follow it.
+
## The [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_patterns.html#pattern-timer-interval <tt>timer:interval</tt>] causes the patter to wait for the specified length of time before continuing.
+
## The [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_patterns.html#pattern-logical-and <tt>and</tt>] and [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_patterns.html#pattern-logical-not <tt>not</tt>] operators perform logical truth operations on the pattern.
+
# This statement removes tags from the Weigh Station, using the same principle as how we remove tags from the Dock Door
+
# This is a listener that will handle arrival and departure events from the dock door.
+
## The first argument contains all the events in the <tt>insert stream</tt>.  The second argument argument contains all events in the <tt>remove stream</tt> for the update.
+
## The <tt>tag_ID</tt> property is there because we declared it when we defined the named window in the first step.
+
# This statement creates the esper statement that selects the events that will feed the listener.
+
## The only special part of this query is the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#epl-select-using-istream-rstream <tt>irstream</tt>] keyword.  This stands for "insert and remove stream".  It tells esper that we are interested in events that are entering and leaving the named window.  By default, esper would only notify us of events that arrive in the named window.
+
# This is the handler for for arrival and departure events from the weigh station.
+
# This is the statement that feeds the weigh station handler.
+
 
+
====Run the Code!====
+
Now, make sure the sessions in the edge server are started and collecting tag data (you can use the <tt>startsession</tt>) command on the osgi console if you need to.  In the console, type <tt>update <bundle_id_of_application></tt> to update your Northwind Application bundle to reflect your latest changes.  You can drag and drop tags on antennas in the emulator, and you should be able to see Arrival and Departure events happen on the console.
+
[[Image:Tutorial-Deploy-Application.png|thumb|800px|none|Bringing the Northwind Application to life!]]
+
 
+
===Alert: Package Moved Backwards!===
+
At this point, we can track the items that can be seen at the various read zones in our scenario.  Now we need to implement the functionality that will fire an alert when an item moves backwards (that is from the weigh station to the dock door).
+
====The Code====
+
<pre>
+
/**
+
* A method that sets up business alerts
+
*/
+
private void setUpAlerts(){
+
// 1 Create  a window for alert messages
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"create window alerts.win:length(20) (alert_type int, tag_ID String)"));
+
 
+
// 2 create a window for item leaving the weighstation
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"create window weighstation_recent.std:unique(tag_ID) (tag_ID String)"));
+
+
// 3 Insert items into weightstation_recent once they leave weighstation
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"insert rstream into weighstation_recent select tag_ID from weighstation"));
+
+
// 4 whenever an item is seen at the weighstation and then seen at the dockdoor, insert a new item into the alerts window
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"on pattern[every-distinct(tag.tag_ID) tag=weighstation_recent -> dockdoor(tag_ID = tag.tag_ID)] " +
+
"insert into alerts " +
+
"select 1 as alert_type, tag_ID as tag_ID " +
+
"from weighstation_recent where tag_ID = tag.tag_ID"));
+
}
+
</pre>
+
 
+
The following code is an addition to the setupListeners() method that was previously defined:
+
<pre>
+
// Create a listener to handle Alerts
+
StatementAwareUpdateListener alertListener = new StatementAwareUpdateListener() {
+
@Override
+
public void update(EventBean[] arrivals, EventBean[] departures,
+
EPStatement arg2, EPServiceProvider arg3) {
+
if (arrivals != null) {
+
for (EventBean bean : arrivals) {
+
int alertType = (Integer) bean.get("alert_type");
+
String id = (String) bean.get("tag_ID");
+
switch (alertType) {
+
case 1:
+
System.out.println("Package moved backwards: " + id );
+
break;
+
 
+
}
+
}
+
}
+
}
+
};
+
+
//Create a query that is triggered on insert and remove events from Weigh Station Window
+
EPStatement queryAlert= esperService.getProvider().getEPAdministrator().createEPL(
+
"select * from alerts");
+
queryAlert.addListener(alertListener);
+
statements.add(queryAlert);
+
</pre>
+
 
+
====Explanation of the Code====
+
# First we need to create a window that will hold alerts.
+
## Using the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl-views.html#view-win-length <tt>win:length()</tt>] view, the window will only keep track of the last 20 alerts
+
## This window is defined to hold two pieces of information per entry: an int that is the alert type, and a string that is the tag ID
+
# Next we create a window that will hold tags that have departed from the weigh station
+
# This statement inserts items that have left the <tt>weighstation</tt> window into the <tt>weighstation_recent</tt> window
+
# Finally we get to define the rule!  This rule simply looks for the pattern of an event entering the <tt>weighstation_recent</tt> window followed by an event entering the <tt>dockdoor</tt> window which have the same ID.  Once this occurs, it makes a new entry in the <tt>alerts</tt> window.
+
## We use the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/event_patterns.html#pattern-logical-everydistinct <tt>every-distinct</tt>] operator to ensure that we will only match the same tag once in the pattern.
+
## We also insert '1' as the alert type.  We will assume that that event type corresponds to a 'tag moved backwards' alert.
+
 
+
===Alert: Package Skipped the Dock Door!===
+
The next requirement for the Northwind application is to detect when a package appears at the weigh station but was never seen at the dock door.  This is a bit different from detecting events going from the weigh station back to the dock door, because we cannot use the followed-by operator on events coming out of the dock door.  It is precisely the events that were never seen at the dock door we are looking for!
+
====The code====
+
Add the following statements to the <tt>setUpAlerts</tt> method:
+
<pre>
+
// 1 create a window for item leaving the dock door
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"create window dockdoor_recent.std:unique(tag_ID) (tag_ID String)"));
+
+
// 2 Insert items into dockdoor_recent once they leave dockdoor
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"insert rstream into dockdoor_recent select tag_ID from dockdoor"));
+
+
// 3 Any time we see a new weighstation event, check to see if it is not already in dockdoor_recent.  If not, make a new alert.
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"insert into alerts " +
+
"select 2 as alert_type, tag_ID as tag_ID " +
+
"from weighstation as w " +
+
"where not exists (select * from dockdoor_recent as d where d.tag_ID = w.tag_ID)"));
+
</pre>
+
 
+
In addition, you need to modify the switch statement of the alerts handler:
+
<pre>
+
switch (alertType) {
+
case 1:
+
System.out.println("Package moved backwards: " + id );
+
break;
+
case 2: System.out.println("Package skipped the dock door: " + id);
+
break;
+
 
+
}
+
</pre>
+
 
+
==== Explanation of the Code ====
+
# Create a named window that will hold tags that have recently departed from the dock_door
+
# Next we insert the tags that have left the <tt>dockdoor</tt> window into the <tt>dockdoor_recent</tt> window.
+
# This statement inserts a new alert into the alerts window whenever a tag arrives at the <tt>weighstation</tt> window which has not been seen at the dock door.
+
## We will assume that 2 is the alert type of the 'package skipped dock door' alert.
+
## The statement <tt>select * from dockdoor_recent as d where d.tag_ID = w.tag_ID</tt> is a [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#epl-subqueries <tt>subquery</tt>] that will return results (just like a normal query).  By using it in conjunction with the [http://esper.codehaus.org/esper-3.2.0/doc/reference/en/html/epl_clauses.html#epl-subqueries-exists <tt>exists</tt>] condition, we can test whether or not the subquery returned any results.  If it does not return results, we need to fire an alert.
+
 
+
===Alert: Package is Lost!===
+
====The Code====
+
Add the following to the setUpAlerts method
+
<pre>
+
// 1 Create a new alert whenever a package departs from the dock door and is not seen at the weighstation with a given time period
+
statements.add(esperService.getProvider().getEPAdministrator().createEPL(
+
"on pattern[every-distinct(tag.tag_ID) tag=dockdoor_recent -> " +
+
"(timer:interval(5 min) and not weighstation(tag_ID = tag.tag_ID))] " +
+
"insert into alerts " +
+
"select 3 as alert_type, tag_ID as tag_ID " +
+
"from dockdoor_recent where (tag_ID = tag.tag_ID)"));
+
</pre>
+
In addition, you need to modify the switch statement of the alerts handler:
+
<pre>
+
switch (alertType) {
+
case 1: System.out.println("Package moved backwards: " + id );
+
break;
+
case 2: System.out.println("Package skipped the dock door: " + id);
+
break;
+
case 3: System.out.println("Package is lost: " + id);
+
break;
+
}
+
</pre>
+
 
+
====Explanation of the Code====
+
# The only statement we have to add to implement the required functionality is, like the 'Package cannot move backwards' alert, implemented in two parts. The first part (on pattern) looks for a sequence to happen. The second part (insert into) does something whenever the first part happens.
+
## We can use the followed-by operator to specify that that we should look for an event leaving a dockdoor followed by a specific time period (15 seconds in our case) and no package with a corresponding id at the weigh station.
+
## When this pattern triggers, we need to insert a new event into the alerts window.
+
## For this kind of alert, we will assume the type will be 3.
+
 
+
==Sending out events with JMS==
+
What you will learn:
+
* Importing the JMS services provided by Rifidi Edge Server
+
* Exporting packages from a bundle
+
* Sending out events over JMS
+
 
+
Once you have the esper queries defined, debugged, and working, it's time to figure out exactly how you want to react to the various events.  You might want to log the events to a database.  You may want to expose the events to a .Net application via web services.  Or you might want to integrate with an ERP system such as SAP.  For the Northwind application, we want to send out notifications over Java Messaging Service (JMS) for a web application (which we will build soon) to display.
+
 
+
===Create Notification Classes===
+
===Use JMS to Send out Notifications in Application===
+
 
+
=Writing A Web Application=
+
=Exporting the application=
+
What you will learn:
+
* Exporting your application out of eclipse
+
* Running the application in a deployed instance of Rifidi Edge Server
+

Latest revision as of 20:49, 8 December 2010

WARNING: This document is for SDK 1.1. For 1.2, please refer to the Northwind instructions documented in the Developer's PDF in the SDK

Developer's Guide

This document provides step-by-step instructions on how to get started on developing your first application that runs on the Rifidi Edge Server. The application we will develop will use Esper to collect tag reads from a reader and put them on a JMS queue to be consumed by a client application. Many of the steps are applicable to many kinds of plugins for the edge server, including creating a sensor plugin.

The Scenario: Northwind Shipping

Congratulations! You are the proud new founder of Northwind Shipping Inc. -- "delivering packages faster than a caffeinated lightning bug"™. One of your core business strategies is to out perform your competitor -- Pony Express Shipping Inc. -- by capitalizing on increased efficiencies gained by your innovative use of technology. You have heard all the hype about RFID and want to employ in it your new, state-of-the-art distribution center. You have decided to use the Rifidi Edge Server to run the RFID applications you will need in your distribution center. This tutorial is a step-by-step guide on how to develop an RFID application and web UI using the Rifidi Edge Server API and deploy the application on the Rifidi Edge Server.

The Northwind distribution center has many complex processes, which should be automated as much as possible. Because you are new to RFID (and because too many processes would overwhelm this tutorial ;-) ) you have decided to start small and only implement a basic process with a few business rules. The first process you will automate using RFID is the receiving process (DC-Inbound). You have a dock door which will be receiving incoming shipments from trucks. The packages must be checked in at the dock door so that your ERP system knows the packages have arrived. Once this happens, the packages must be moved to a separate staging area where it needs to be weighed and prepared for further routing.

There are several goals for the application

  • Track the packages from the dock door (stage 1) to the weigh station (stage 2).
  • Send an alert if a package moves backwards (from stage 2 to stage 1).
  • Send an alert if a package arrives at the weigh station but was not seen at the dock door
  • Send an alert if a package departs the dock door and is not seen at the weigh station within five minutes

The Architecture

The solution that we will build will consist of two parts: the application bundle which will implement all the business rules, and the web application which will display the items. The two pieces will communicate using JMS.

Northwind.jpeg

Prerequisites

For this tutorial, we will use Eclipse to develop the application. While it would be possible to develop the application in any IDE that you are familiar with, Eclipse provides great tooling around OSGi application development and deployment, and thus makes this process much easier. If this is your first time developing with eclipse, there will be a learning curve. However, the payoff is worth it.

To get started see Setting up a Development Environment.

Source

You can download the source for this project here: File:Northwind 1.0.0.zip. Just unzip the project, open up eclipse and select import->existing projects into workspace.

Outline

  1. Step 1: Create the Application
  2. Step 2: Using Emulator
  3. Step 3: Esper: Track Packages
  4. Step 4: Esper: Alerts
  5. Step 5: Send Notifications Over JMS
  6. Step 6: Create the Web Application
  7. Step 7: Write a Hello World Servlet
  8. Step 8: Write a Tag Location Service and JMS Listener
  9. Northwind Display Events using JSP: Step 9: Display events in a JSP
Personal tools