module SPARQL; module Algebra
class Operator
##
# The SPARQL `min` set function.
#
# [127] Aggregate::= ... | 'MIN' '(' 'DISTINCT'? Expression ')'
#
# @example SPARQL Grammar
# PREFIX :
# SELECT (MIN(?o) AS ?min)
# WHERE { ?s :dec ?o }
#
# @example SSE
# (prefix ((: ))
# (project (?min)
# (extend ((?min ??.0))
# (group () ((??.0 (min ?o)))
# (bgp (triple ?s :dec ?o))))))
#
# @see https://www.w3.org/TR/sparql11-query/#defn_aggMin
class Min < Operator
include Aggregate
NAME = :min
def initialize(*operands, **options)
raise ArgumentError,
"min operator accepts at most one argument with an optional :distinct" if
(operands - %i{distinct}).length != 1
super
end
##
# Min is a SPARQL set function that return the minimum value from a group respectively.
#
# It makes use of the SPARQL ORDER BY ordering definition, to allow ordering over arbitrarily typed expressions.
#
# @param [Enumerable>] enum
# enum of evaluated operand
# @return [RDF::Literal] The maximum value of the terms
def apply(enum, **options)
# FIXME: we don't actually do anything with distinct
operands.shift if distinct = (operands.first == :distinct)
if enum.empty?
raise TypeError, "Minumuim of an empty multiset"
elsif enum.flatten.all? {|n| n.literal?}
RDF::Literal(enum.flatten.min)
else
raise TypeError, "Minumuim of non-literals: #{enum.flatten}"
end
end
##
#
# Returns a partial SPARQL grammar for this operator.
#
# @return [String]
def to_sparql(**options)
distinct = operands.first == :distinct
args = distinct ? operands[1..-1] : operands
"MIN(#{'DISTINCT ' if distinct}#{args.to_sparql(**options)})"
end
end # Min
end # Operator
end; end # SPARQL::Algebra