module SPARQL; module Algebra class Operator ## # The SPARQL GraphPattern `order` operator. # # @example # (prefix ((foaf: )) # (project (?name) # (order ((asc ?name)) # (bgp (triple ?x foaf:name ?name))))) # # @see http://www.w3.org/TR/rdf-sparql-query/#modOrderBy class Order < Operator::Binary include Query NAME = [:order] ## # Executes this query on the given `queryable` graph or repository. # Orders a solution set returned by executing operand(1) using # an array of expressions and/or variables specified in operand(0) # # @param [RDF::Queryable] queryable # the graph or repository to query # @param [Hash{Symbol => Object}] options # any additional keyword options # @return [RDF::Query::Solutions] # the resulting solution sequence # @see http://www.w3.org/TR/rdf-sparql-query/#sparqlAlgebra def execute(queryable, options = {}) debug(options) {"Order"} @solutions = operands.last.execute(queryable, options.merge(:depth => options[:depth].to_i + 1)).order do |a, b| operand(0).inject(false) do |memo, op| debug(options) {"=> #{op.inspect}"} memo ||= begin a_eval = op.evaluate(a, options.merge(:queryable => queryable, :depth => options[:depth].to_i + 1)) rescue nil b_eval = op.evaluate(b, options.merge(:queryable => queryable, :depth => options[:depth].to_i + 1)) rescue nil comp = if a_eval.nil? RDF::Literal(-1) elsif b_eval.nil? RDF::Literal(1) elsif op.is_a?(RDF::Query::Variable) a_eval <=> b_eval else Operator::Compare.evaluate(a_eval, b_eval) end comp = -comp if op.is_a?(Operator::Desc) comp == 0 ? false : comp end end || 0 # They compare equivalently if there are no matches end end ## # Returns an optimized version of this query. # # Return optimized query # # @return [Union, RDF::Query] `self` def optimize operands = operands.map(&:optimize) end end # Order end # Operator end; end # SPARQL::Algebra