module SPARQL; module Algebra class Operator ## # The SPARQL numeric `divide` operator. # # [117] MultiplicativeExpression::= UnaryExpression ( '*' UnaryExpression | '/' UnaryExpression )* # # @example SPARQL Grammar # PREFIX : # SELECT ?s WHERE { # ?s :p ?o . # ?s2 :p ?o2 . # FILTER(?o / ?o2 = 4) . # } # # @example SSE # (prefix ((: )) # (project (?s) # (filter (= (/ ?o ?o2) 4) # (bgp # (triple ?s :p ?o) # (triple ?s2 :p ?o2))))) # # @see https://www.w3.org/TR/xpath-functions/#func-numeric-divide class Divide < Operator::Binary include Evaluatable NAME = [:'/', :divide] ## # Returns the arithmetic quotient of the operands. # # @param [RDF::Literal::Numeric] left # a numeric literal # @param [RDF::Literal::Numeric] right # a numeric literal # @return [RDF::Literal::Numeric] # @raise [TypeError] if either operand is not a numeric literal def apply(left, right, **options) case when left.is_a?(RDF::Literal::Numeric) && right.is_a?(RDF::Literal::Numeric) # For xsd:decimal and xsd:integer operands, if the divisor is # (positive or negative) zero, an error is raised. raise ZeroDivisionError, "divided by #{right}" if left.is_a?(RDF::Literal::Decimal) && right.zero? # As a special case, if the types of both operands are # xsd:integer, then the return type is xsd:decimal. if left.is_a?(RDF::Literal::Integer) && right.is_a?(RDF::Literal::Integer) RDF::Literal(left.to_d / right.to_d) else left / right end else raise TypeError, "expected two RDF::Literal::Numeric operands, but got #{left.inspect} and #{right.inspect}" end end ## # # Returns a partial SPARQL grammar for this operator. # # @return [String] def to_sparql(**options) "(#{operands.first.to_sparql(**options)} / #{operands.last.to_sparql(**options)})" end end # Divide end # Operator end; end # SPARQL::Algebra