Class: Solis::Query

Inherits:
Object
  • Object
show all
Includes:
Enumerable, QueryFilter
Defined in:
lib/solis/query.rb,
lib/solis/query/construct.rb

Defined Under Namespace

Classes: Construct, Runner

Constant Summary collapse

XSD_NUMERIC_DATATYPES =

XSD numeric datatypes must be ordered by value; wrapping them in STR() would sort them lexically (“100” < “9”). Every other datatype is STR()-wrapped in the outer ORDER BY (see #sort_key_expression) to dodge a Virtuoso collation bug.

%w[
  integer decimal float double long int short byte
  nonNegativeInteger nonPositiveInteger negativeInteger positiveInteger
  unsignedLong unsignedInt unsignedShort unsignedByte
].map { |t| "http://www.w3.org/2001/XMLSchema##{t}" }.freeze

Class Method Summary collapse

Instance Method Summary collapse

Methods included from QueryFilter

#filter

Constructor Details

#initialize(model) ⇒ Query

Returns a new instance of Query.



94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
# File 'lib/solis/query.rb', line 94

def initialize(model)
  @construct_cache = File.absolute_path(Solis::Options.instance.get[:cache])
  @model = model
  @shapes = @model.class.shapes
  @metadata = @model.class.
  @sparql_endpoint = @model.class.sparql_endpoint
  if Solis::Options.instance.get.key?(:graphs) && Solis::Options.instance.get[:graphs].size > 0
    @sparql_client = Solis::Store::Sparql::Client.new(@sparql_endpoint)
  else
    @sparql_client = Solis::Store::Sparql::Client.new(@sparql_endpoint, graph: @model.class.graph_name)
  end
  @filter = {values: ["VALUES ?type {#{target_class}}"], concepts: ['?concept a ?type .'] }
  @sort = 'ORDER BY ?concept'
  @sort_select = ''
  # @sort_project carries the sort key(s) out of the inner (paginated) subquery so the
  # outer query can re-sort on them; @sort_outer is that outer ORDER BY. A subquery's
  # ORDER BY only decides which rows survive LIMIT/OFFSET — it does NOT propagate
  # through the enclosing join — so the outer query must sort too. See #sort for the
  # datatype-aware STR() handling that the outer ORDER BY needs.
  @sort_project = ''
  @sort_outer = ''
  @language = Graphiti.context[:object]&.language || Solis::Options.instance.get[:language] || 'en'
  @query_cache = self.class.shared_query_cache
end

Class Method Details

.graph_nameObject



66
67
68
# File 'lib/solis/query.rb', line 66

def self.graph_name
  Solis::Options.instance.get.key?(:graphs) ? Solis::Options.instance.get[:graphs].select{|s| s['type'].eql?(:main)}&.first['name'] : ''
end

.invalidate_cache_for(model_class_name, cache_dir = nil) ⇒ Object

Invalidate all cached query results for a given model type.



82
83
84
85
86
87
88
89
90
91
92
# File 'lib/solis/query.rb', line 82

def self.invalidate_cache_for(model_class_name, cache_dir = nil)
  cache = shared_query_cache
  tag_key = "TAG:#{model_class_name}"
  if cache.key?(tag_key)
    cache[tag_key].each { |key| cache.delete(key) }
    cache.delete(tag_key)
    Solis::LOGGER.info("CACHE: invalidated entries for #{model_class_name}") if ConfigFile[:debug]
  end
rescue StandardError => e
  Solis::LOGGER.warn("CACHE: invalidation failed for #{model_class_name}: #{e.message}")
end

.reset_shared_query_cache!Object

Reset the shared cache (useful when config changes, e.g., in tests)



77
78
79
# File 'lib/solis/query.rb', line 77

def self.reset_shared_query_cache!
  @shared_query_cache = nil
end

.run(entity, query, options = {}) ⇒ Object



23
24
25
# File 'lib/solis/query.rb', line 23

def self.run(entity, query, options = {})
  Solis::Query::Runner.run(entity, query, options)
end

.run_construct(query, id_name, entity, ids, from_cache = '1') ⇒ Object



36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
# File 'lib/solis/query.rb', line 36

def self.run_construct(query, id_name, entity, ids, from_cache = '1')
  raise 'Please supply one or more uuid\'s' if ids.nil? || ids.empty?

  result = {}

  key = uuid("#{entity}-#{ids}")

  if result.nil? || result.empty? || (from_cache.eql?('0'))
    ids = ids.split(',') if ids.is_a?(String)
    ids = [ids] unless ids.is_a?(Array)
    ids = ids.map do |m|
      if URI(m).class.eql?(URI::Generic)
        "<#{graph_name}#{entity.tableize}/#{m}>"
      else
        "<#{m}>"
      end
    end
    ids = ids.join(" ")

    language = Graphiti.context[:object]&.language || Solis::Options.instance.get[:language] || 'en'
    q = query.gsub(/{ ?{ ?VALUES ?} ?}/, "VALUES ?#{id_name} { #{ids} }").gsub(/{ ?{ ?LANGUAGE ?} ?}/, "bind(\"#{language}\" as ?filter_language).").gsub(/{ ?{ ?ENTITY ?} ?}/, "<#{graph_name}#{entity.classify}>")

    result = Solis::Query.run(entity, q)
  end
  result
rescue StandardError => e
  puts e.message
  raise e
end

.run_construct_with_file(filename, id_name, entity, ids, from_cache = '1') ⇒ Object



27
28
29
30
# File 'lib/solis/query.rb', line 27

def self.run_construct_with_file(filename, id_name, entity, ids, from_cache = '1')
  f = File.read(filename)
  run_construct(f, id_name, entity, ids, from_cache)
end

.shared_query_cacheObject

Shared class-level query cache to ensure consistent reads/writes/invalidations



71
72
73
74
# File 'lib/solis/query.rb', line 71

def self.shared_query_cache
  cache_dir = File.absolute_path(Solis::Options.instance.get[:cache])
  @shared_query_cache ||= Moneta.new(:HashFile, dir: cache_dir)
end

.uuid(key) ⇒ Object



32
33
34
# File 'lib/solis/query.rb', line 32

def self.uuid(key)
  UUIDTools::UUID.sha1_create(UUIDTools::UUID_URL_NAMESPACE, key).to_s
end

Instance Method Details

#countObject



172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
# File 'lib/solis/query.rb', line 172

def count
  sparql_client = @sparql_client
  if model_construct?
    sparql_client = Solis::Query::Construct.new(@model).run
  end

  relationship = ''
  core_query = core_query(relationship)
  count_query = core_query.gsub(/SELECT .* WHERE/, 'SELECT (COUNT(distinct ?concept) as ?count) WHERE')

  # count_query = count_query.split('a ?type')[0]+'a ?type }'
  result = sparql_client.query(count_query)
  solution = result.first
  solution.nil? ? 0 : solution[:count].object || 0
end

#each(&block) ⇒ Object



119
120
121
122
123
124
125
126
127
# File 'lib/solis/query.rb', line 119

def each(&block)
  data = query
  return unless data.methods.include?(:each)
  data.each(&block)
rescue StandardError => e
  message = "Unable to get next record: #{e.message}"
  LOGGER.error(message)
  raise Error::CursorError, message
end

#paging(params = {}) ⇒ Object



162
163
164
165
166
167
168
169
170
# File 'lib/solis/query.rb', line 162

def paging(params = {})
  current_page = params[:current_page] || 1
  per_page = params[:per_page] || 10

  @offset = 0
  @offset = (current_page - 1) * per_page if current_page > 1
  @limit = per_page
  self
end

#sort(params) ⇒ Object



129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
# File 'lib/solis/query.rb', line 129

def sort(params)
  @sort = ''
  @sort_select = ''
  @sort_project = ''
  @sort_outer = ''
  if params.key?(:sort)
    i = 0
    outer = ''
    params[:sort].each do |attribute, direction|
      meta = @model.class.[:attributes][attribute.to_s]
      path = meta[:path]
      @sort_select += "optional {\n" if meta[:mincount] == 0
      @sort_select += "?concept <#{path}> ?__#{attribute} . "
      @sort_select += "}\n" if meta[:mincount] == 0
      @sort += ',' if i.positive?
      @sort += "#{direction.to_s.upcase}(?__#{attribute})"
      # Carry the sort key out of the subquery so the outer query can re-sort on it.
      @sort_project += " ?__#{attribute}"
      outer += ',' if i.positive?
      outer += "#{direction.to_s.upcase}(#{sort_key_expression(attribute, meta)})"
      i += 1
    end

    if i.positive?
      @sort = "ORDER BY #{@sort}"
      # ?s tiebreaker keeps each subject's triples contiguous and ties deterministic.
      @sort_outer = "ORDER BY #{outer} ?s"
    end
  end

  self
end