# #-- # Copyright (c) 2006-2007, John Mettraux, OpenWFE.org # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # . Redistributions of source code must retain the above copyright notice, this # list of conditions and the following disclaimer. # # . Redistributions in binary form must reproduce the above copyright notice, # this list of conditions and the following disclaimer in the documentation # and/or other materials provided with the distribution. # # . Neither the name of the "OpenWFE" nor the names of its contributors may be # used to endorse or promote products derived from this software without # specific prior written permission. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. #++ # # $Id: definitions.rb 2725 2006-06-02 13:26:32Z jmettraux $ # # # "made in Japan" # # John Mettraux at openwfe.org # #require 'rexml/document' require 'openwfe/rudefinitions' require 'openwfe/expressions/flowexpression' module OpenWFE # # An 'abstract' class storing bits (trees) of process definitions just # parsed. Upon application (apply()) these raw expressions get turned # into real expressions. # The first and classical extension of this class is XmlRawExpression. # class RawExpression < FlowExpression def initialize \ (fei, parent_id, env_id, application_context, raw_representation) super(fei, parent_id, env_id, application_context, nil) @raw_representation = raw_representation #new_environment() if not @environment_id # # now done in the launch methods of the expression pool end def instantiate_real_expression \ (workitem, exp_class=nil, attributes=nil) exp_class = expression_class() unless exp_class raise "unknown expression '#{expression_name}'" \ unless exp_class #ldebug do # "instantiate_real_expression() exp_class is #{exp_class}" #end attributes = extract_attributes() unless attributes expression = exp_class.new( @fei, @parent_id, @environment_id, @application_context, attributes) handle_descriptions() expression.children = extract_children() expression.store_itself() expression end def apply (workitem) exp_name = expression_name() exp_class = expression_class() attributes = nil template = lookup_variable(exp_name) # # is it a subprocess ? if (not template) and (not exp_class) template = get_participant_map.lookup_participant(exp_name) unless template exp_name = OpenWFE::to_underscore(exp_name) template = get_participant_map.lookup_participant(exp_name) end end # # is it a directly a participant ? if template if template.kind_of? OpenWFE::FlowExpressionId launch_template(template, workitem) return end if template.kind_of? OpenWFE::Participant exp_class = OpenWFE::ParticipantExpression attributes = extract_attributes() attributes["ref"] = exp_name end end # # the classical case... expression = instantiate_real_expression( workitem, exp_class, attributes) expression.apply_time = OpenWFE::now() expression.apply(workitem) end # # This method is called by the expression pool when it is about # to launch a process, it will interpret the 'parameter' statements # in the process definition and raise an exception if the requirements # are not met. # def check_parameters (workitem) extract_parameters.each do |param| param.check(workitem) end end #def reply (workitem) # no implementation necessary #end def is_definition? () return get_expression_map.is_definition?(expression_name()) end def expression_class () return get_expression_map.get_class(expression_name()) end def definition_name () return raw_representation.attributes['name'].to_s end def expression_name () return raw_representation.name end protected # # Takes care of extracting the process definition descriptions # if any and to set the description variables accordingly. # def handle_descriptions default = false ds = extract_descriptions ds.each do |k, description| vname = if k == "default" default = true "description" else "description__#{k}" end set_variable vname, description end return if ds.length < 1 set_variable "description", ds[0][1] \ unless default end def launch_template (template, workitem) @attributes = extract_attributes() #require 'pp' #pp attributes #pp lookup_attributes(workitem, attributes) params = lookup_attributes(workitem) extract_text_children.each_with_index do |value, index| params[index.to_s] = value end get_expression_pool().launch_template( self, 0, template, workitem, params) end def extract_attributes () raise NotImplementedError.new("'abstract method' sorry") end def extract_children () raise NotImplementedError.new("'abstract method' sorry") end def extract_descriptions () raise NotImplementedError.new("'abstract method' sorry") end def extract_parameters () raise NotImplementedError.new("'abstract method' sorry") end def extract_text_children () raise NotImplementedError.new("'abstract method' sorry") end # # Encapsulating # # class Parameter def initialize (field, match, default, type) @field = field @match = match @default = default @type = type end # # Will raise an exception if this param requirement is not # met by the workitem. # def check (workitem) unless @field raise \ OpenWFE::ParameterException, "'parameter'/'param' without a 'field' attribute" end field_value = workitem.attributes[@field] field_value = @default unless field_value unless field_value raise \ OpenWFE::ParameterException, "field '#{@field}' is missing" \ end check_match(field_value) enforce_type(workitem, field_value) end protected # # Will raise an exception if it cannot coerce the type # of the value to the one desired. # def enforce_type (workitem, value) value = if not @type value elsif @type == "string" value.to_s elsif @type == "int" or @type == "integer" Integer(value) elsif @type == "float" Float(value) else raise "unknown type '#{@type}' for field '#{@field}'" end workitem.attributes[@field] = value end def check_match (value) return unless @match unless value.to_s.match(@match) raise \ OpenWFE::ParameterException, "value of field '#{@field}' doesn't match" end end end end end