require_relative '../serializer' require_relative '../xdm/value' module Saxon module XSLT # Represents the invocation of a compiled and loaded transformation, # providing an idiomatic way to send the result a transformation to a # particular Destination, or to serialize it directly to a file, string, or # IO. class Invocation attr_reader :s9_transformer, :invocation_lambda private :s9_transformer, :invocation_lambda # @api private def initialize(s9_transformer, invocation_lambda, raw) @s9_transformer, @invocation_lambda, @raw = s9_transformer, invocation_lambda, raw end # Serialize the result to a string using the options specified in # ++ in the XSLT # # @return [String] the serialized result of the transformation def to_s serializer_destination.serialize end # Serialize the result of the transformation. When called with a # block, the block will be executed via instance-exec so that output # properties can be set, e.g. # # result.serialize { # output_property[:indent] = 'yes' # } # # @overload serialize(io) # Serialize the transformation to an IO # @param [File, IO] io The IO to serialize to # @return [nil] # @yield the passed block bound via instance-exec to the new serializer # @overload serialize(path) # Serialize the transformation to file path # @param [String, Pathname] path The path of the file to serialize to # @return [nil] # @yield the passed block bound via instance-exec to the new serializer # @overload serialize # Serialize the transformation to a String # @return [String] The serialized XdmValue def serialize(path_or_io = nil, &block) serializer_destination(&block).serialize(path_or_io) end # Return the result of the transformation as an XDM Value, either as it is # returned (if the XSLT::Executable was created with :raw => true), or # wrapped in a Document node and sequence normalized. # # @return [Saxon::XDM::Value] the XDM value returned by the transformation def xdm_value if raw? XDM.Value(invocation_lambda.call(nil)) else invocation_lambda.call(s9_xdm_destination) XDM.Value(s9_xdm_destination.getXdmNode) end end # Send the result of the transformation to the supplied Destination # # @param s9_destination [net.sf.saxon.s9api.Destination] the Saxon # destination to use # @return [nil] def to_destination(s9_destination) invocation_lambda.call(s9_destination) nil end # Whether the transformation was invoked as a 'raw' transformation, where # an XDM Value result will be returned without being wrapped in a Document # node, and without sequence normalization. # # @return [Boolean] whether the transformation was invoked 'raw' or not def raw? @raw end private def serializer_destination(&block) Saxon::Serializer::Destination.new(s9_transformer.newSerializer, invocation_lambda, &block) end def s9_xdm_destination @s9_xdm_destination ||= Saxon::S9API::XdmDestination.new end end end end