Data AccessorsData Accessors
Building Data Accessors
Home > Books > NetKernel Extensions > Accessor Development > Data Accessors

Rate this page:
Really useful
Satisfactory
Not helpful
Confusing
Incorrect
Unsure
Extra comments:


Data Accessor Development

Data accessors provide gateway access to a set of resources. Typically a data accessor will implement a full URI scheme and service more than one type of request type.

According to the URI specification, a scheme designates a set of rules that defines the syntax and semantics of valid addresses. When a data accessor implements an entire scheme, it is responsible for parsing and interpreting the URI addresses. {What processing is provided by NetKernel when handling the address portion of the URI address?}

Because it implements an entire scheme, a suitable set of requests types must be supported. Typically, an accessor applies these semantics to the standard NetKernel request types:

  • SOURCE: retrieve a representation of the resource at the given address.
  • SINK: update the resource to reflect the information contained in the passed representation.
  • NEW: create a new (null valued) resource at the location specified by the address. Note that in some cases creating a new resource can be an implicit part of a SINK operation.
  • DELETE: remove the underlying resource at the location specified by the address.
  • EXISTS: test to see if a resource is located at the specified address, return a boolean indicator.
However, these semantics are only a guideline. The effective semantics are left entirely to the accessor developer.

This guide illustrates how to build a data accessor for a new scheme called transient:. First read the guide to building active accessors to learn how to write an accessor in Java. This guide expands on that material and discusses the enhancements required to create a data accessor.

transient: scheme

The transient: URI scheme provides in-memory storage for representations of resources. The URI address is used as a key to an in-memory cache. The supported logical operations place a representation in memory (SINK), retrieve a representation from memory (SOURCE), delete a representation from memory (DELETE), and determine if a representation is stored in memory (EXISTS). Examples of URI addresses that are supported include:

  • transient:myResult.txt
  • transient:/sessions/1/data.gif
An example of DPML code places a representation of a file into transient: is:
<instr>
  <type>copy</type>
  <operand>ffcpl:/myFile.txt</operand>
  <target>transient:tempFile.txt</target>
</instr>
To send the information from the transient storage as the representation for a response:
<instr>
  <type>copy</type>
  <operand>transient:tempFile.txt</operand>
  <target>this:response</target>
</instr>

Implementing transient:

Step 1 - declare class

As with active accessor development, create a subclass of org.ten60.netkernel.layer1.nkf.impl.NKFAccessorImpl located in the module urn:org:ten60:netkernel:ext:layer1, and provide implementations for the lifecycle methods. Because this example accessor implements transient storage, there is no work to be completed in the initialise() and destroy() methods.

package org.myDomain.myApp;
      
import java.util.HashMap;
import java.util.Map;
  
import org.ten60.netkernel.layer1.nkf.INKFConvenienceHelper;
import org.ten60.netkernel.layer1.nkf.INKFRequestReadOnly;
import org.ten60.netkernel.layer1.nkf.INKFResponse;
import org.ten60.netkernel.layer1.nkf.impl.NKFAccessorImpl;
import org.ten60.netkernel.layer1.representation.BooleanAspect;
  
import com.ten60.netkernel.urii.IURAspect;
import com.ten60.netkernel.urii.IURRepresentation;
  
public class TransientDataAccessor extends NKFAccessorImpl
{ private final static Map mResources = new HashMap();
  
  public TransientDataAccessor()
  { super(NOT_SAFE_FOR_CONCURRENT_USE, 
          INKFRequestReadOnly.RQT_SOURCE   | 
          INKFRequestReadOnly.RQT_SINK     | 
          INKFRequestReadOnly.RQT_DELETE   | 
          INKFRequestReadOnly.RQT_EXISTS
         );
  }
  ...
}
This partial implementation declares and instantiates a class static Map variable that holds the representations. The constructor indicates that the class supports SOURCE, SINK, EXISTS and DELETE request types. (If a request type is sent to an accessor and the accessor does not declare support for that type, then NetKernel automatically raises an Unsupported Request Type exception. {Is this really the type of the exception?}

Step 2 - implement functionality

The processRequest(...) method is called when a request is received by the accessor. This method test the request type and calls an appropriate method depending on the request verb.

public void processRequest(INKFConvenienceHelper context) throws Exception
{ switch (context.getThisRequest().getRequestType())
  {
  case INKFRequestReadOnly.RQT_SINK:
    sink(context);
    break;
  case INKFRequestReadOnly.RQT_SOURCE:
    source(context);
    break;
  case INKFRequestReadOnly.RQT_DELETE:
    delete(context);
    break;
  case INKFRequestReadOnly.RQT_EXISTS:
    exists(context);
    break;
  default:
    throw new Exception("Unsupported Request Type");
  } 
}
  
protected void sink(INKFConvenienceHelper context) throws Exception
{ IURRepresentation rep = context.source(INKFRequestReadOnly.URI_SYSTEM);
  IURAspect aspect = (IURAspect) rep.getAspects().iterator().next();
  String path = context.getThisRequest().getURIWithoutFragment();
  mResources.put(path, aspect);
}
  
protected void source(INKFConvenienceHelper context) throws Exception
{ String path = context.getThisRequest().getURIWithoutFragment();
  IURAspect aspect = (IURAspect) mResources.get(path);
  if (null == aspect)
  { throw new Exception("Resource " + path + " not found.");
  }   
  else
  { INKFResponse resp = context.createResponseFrom(aspect);
    context.setResponse(resp);
  }
}
  
protected void delete(INKFConvenienceHelper context) throws Exception
{ String path = context.getThisRequest().getURIWithoutFragment();
  mResources.remove(path);
  IURAspect aspect = new BooleanAspect(true);
  INKFResponse resp = context.createResponseFrom(aspect);
  context.setResponse(resp);
}
  
protected void exists(INKFConvenienceHelper context) throws Exception
{ String path = context.getThisRequest().getURIWithoutFragment();
  boolean result = (null != mResources.get(path));
  BooleanAspect ba = new BooleanAspect(result);
  INKFResponse resp = context.createResponseFrom(ba);
  context.setResponse(resp);
}
The sink() method saves the passed representation in the static map. When a SINK request is received, the representation to be stored is always located in the request context by INKFRequestReadOnly.URI_SYSTEM. The representation is placed in the map using the request URI as the key. Note that this implementation of SINK performs an implicit NEW operation.

The source() method uses the request URI as a key to locate a representation in the static map. If it find a representation, it is returned, otherwise an exception is thrown.

The delete() method uses the request URI as a key and asks the static map to delete that entry.

The exists() method uses the request URI as a key and determines if there is an entry in the static map at that key value.

Step 3 - register the accessor

The data accessor must be registered with a module in a manner that directs all requests for the scheme to this new accessor. This is accomplished with the following entry in the mapping section of module.xml:

<ura>
  <match>transient:.*</match>
  <class>org.myDomain.myApp.TestNKFDataAccessor</class>
</ura>

© 2003-2007, 1060 Research Limited. 1060 registered trademark, NetKernel trademark of 1060 Research Limited.