I released version 0.1.2 of SPQR this morning; it is available from gemcutter (as an installable gem package) or from fedorahosted.org (as source). This version contains many enhancements and fixes when compared to the versions described my previous SPQR posts. I’ve noticed that people are still installing older versions of SPQR and would like to encourage everyone who’s interested in using SPQR to adopt the most recent version.
The 0.1.x series and the 0.0.x series are not API-compatible, but the incompatibility is for a good reason: it enables SPQR to expose normal Ruby methods (that is, those that don’t use keyword arguments). To see what I mean, compare the “Hello, world” example from v0.1.2 with its counterpart in v0.0.4. (The former is also embedded below, but you might have to click through to the post to see it if you’re reading this in a syndication feed.)
require 'spqr/spqr'
require 'spqr/app'
require 'logger'
class Hello
include SPQR::Manageable
# Input (in and inout) parameters are passed to exposed methods in
# the order they are declared in the expose block (see below).
def hello(name)
# We will keep track of the number of people we have greeted over
# the lifetime of this agent (see below for the people_greeted
# statistic declaration).
@people_greeted = @people_greeted + 1
# In methods that return only one value --- that is, those which
# have only one parameter that is either out or inout --- the
# return value is provided normally. Methods that have multiple
# out and inout parameters should return a list that includes the
# returned values in the order they are specified in the expose block.
"Hello, #{name}!"
end
# This block indicates that we intend to expose the "hello" method
# over QMF, that "name" is the first (and only) formal input
# parameter, and "result" is the first (and only) formal output
# parameter. Argument declarations include a name, a type, a
# direction, and optional keyword arguments.
:hello do |args|
expose .declare :name, :lstr, :in
args.declare :result, :lstr, :out
argsend
# This is the method that will be called to get the value of the
# service_name property
def service_name
@service_name = "HelloAgent"
end
# The following two declarations provide the QMF package and class
# names for the published class. These can be provided either as
# symbols or as strings. If the class name is omitted, it will be
# generated from the class name.
:hello
qmf_package_name :Hello
qmf_class_name
# The following two declarations create named QMF statistics and
# properties with given types. The value for a statistic or
# property is taken from the return values of the instance method
# with the same name as that statistic or property. If a method
# with this name does not exist, one is created with attr_reader
# and attr_writer.
:people_greeted, :int
qmf_statistic :service_name, :lstr
qmf_property
# These should return the same object for the lifetime of the agent
# app, since this example has no persistent objects.
def Hello.find_all
@@hellos ||= [Hello.new]
end
def Hello.find_by_id(id)
@@hellos ||= [Hello.new]
@@hellos[0]
end
def initialize
@people_greeted = 0
end
end
= SPQR::App.new(:loglevel => :debug)
app .register Hello
app
.main app
Here are some other enhancements to the code since the last time I mentioned SPQR:
- Many fixes and enhancements to SPQR/Rhubarb (simple object-graph persistence) integration; most glue methods are now automatically generated at runtime.
- Improved code generation from XML QMF schema files, including support for generating classes that are both exposed over QMF (with SPQR) and persistent (with Rhubarb).
- A stable and mostly-repeatable test suite. (I hope to write up the process of developing unit tests for QMF agents at some point in the near future.)
- Enhancements to the app skeleton, which can now specify the broker that manages the QMF bus.
- Many fixes and stability enhancements, most of which have come out of my experience with a substantial SPQR/Rhubarb application I’m developing.
The full list is here. Enjoy, and please don’t hesitate to write with questions.