sine_axiom_selection.rb revision 905b6df701bc9f0b8a921c1758fd18052306fd09
# This class implements the SInE axiom selection algorithm proposed in:
#
# K. Hoder and A. Voronkov. Sine qua non for large theory reasoning.
# In Automated Deduction - CADE-23 - 23rd International Conference on Automated
# Deduction, Wroclaw, Poland, July 31 - August 5, 2011. Proceedings,
# pages 299–314, 2011.
class SineAxiomSelection < ActiveRecord::Base
acts_as :axiom_selection
attr_accessible :commonness_threshold, :depth_limit, :tolerance
validates_numericality_of :commonness_threshold,
greater_than_or_equal_to: 0,
only_integer: true
# Special case: The depth limit of -1 is considered to be infinite.
validates_numericality_of :depth_limit,
greater_than_or_equal_to: -1,
only_integer: true
validates_numericality_of :tolerance, greater_than_or_equal_to: 1
delegate :goal, :ontology, :lock_key, :mark_as_finished!, to: :axiom_selection
def call
Semaphore.exclusively(lock_key) do
unless finished
cleanup
preprocess unless other_finished_sine_axiom_selections.any?
select_axioms
mark_as_finished!
end
end
end
# This can't be made a 'has one through' association because the
# `axiom_selection` association is polymorphic.
def sine_symbol_commonnesses
SineSymbolCommonness.where(axiom_selection_id: axiom_selection)
end
# This can't be made a 'has one through' association because the
# `axiom_selection` association is polymorphic.
def sine_symbol_axiom_triggers
SineSymbolAxiomTrigger.where(axiom_selection_id: axiom_selection)
end
def destroy
cleanup
super
end
protected
def cleanup
sine_symbol_commonnesses.each(&:destroy)
sine_symbol_axiom_triggers.each(&:destroy)
end
def preprocess
calculate_commonness_table
calculate_symbol_axiom_trigger_table
end
def calculate_commonness_table
ontology.symbols.each { |symbol| calculate_commonness(symbol) }
end
# Number of axioms in which the symbol occurs
def calculate_commonness(symbol)
ssc = sine_symbol_commonnesses.
where(symbol_id: symbol.id).first_or_initialize
unless ssc.commonness
ssc.commonness = query_commonness(symbol)
ssc.save!
end
end
def query_commonness(symbol)
ontology.all_axioms.joins(:symbols).where(:'symbols.id' => symbol.id).count
end
def calculate_symbol_axiom_trigger_table
ontology.all_axioms.each do |axiom|
axiom.symbols.each do |symbol|
calculate_symbol_axiom_trigger(symbol, axiom)
end
end
end
def calculate_symbol_axiom_trigger(symbol, axiom)
ssat = sine_symbol_axiom_triggers.
where(symbol_id: symbol.id,
axiom_id: axiom.id).first_or_initialize
unless ssat.tolerance
ssat.tolerance = needed_tolerance(symbol, axiom)
ssat.save!
end
end
def needed_tolerance(symbol, axiom)
lcs_commonness = commonness_of_least_common_symbol(axiom)
sym_commonness = symbol.sine_symbol_commonness.commonness
sym_commonness.to_f / lcs_commonness.to_f
end
def least_common_symbol(axiom)
axiom.symbols.includes(:sine_symbol_commonness).
order('sine_symbol_commonnesses.commonness ASC').first
end
def commonness_of_least_common_symbol(axiom)
least_common_symbol(axiom).sine_symbol_commonness.commonness
end
def other_finished_sine_axiom_selections
goal.proof_attempts.includes(:axiom_selection).map(&:axiom_selection).
select do |as|
as.specific.class == self.class && as.finished &&
as.id != axiom_selection.id
end.map(&:specific)
end
def select_axioms
@selected_axioms = triggered_axioms_by_commonness_threshold.to_a
select_new_axioms(goal, 0)
self.axioms = @selected_axioms.uniq
end
def select_new_axioms(sentence, current_depth)
return if depth_limit_reached?(current_depth)
new_axioms = select_axioms_by_sentence(sentence) - @selected_axioms
@selected_axioms += new_axioms
new_axioms.each { |axiom| select_new_axioms(axiom, current_depth + 2) }
end
def select_axioms_by_sentence(sentence)
sentence.symbols.map { |symbol| triggered_axioms(symbol) }.flatten
end
def triggered_axioms(symbol)
Axiom.unscoped.joins(:sine_symbol_axiom_triggers).
where('sine_symbol_axiom_triggers.symbol_id = ?', symbol.id).
where('sine_symbol_axiom_triggers.tolerance <= ?', tolerance)
end
def triggered_axioms_by_commonness_threshold
Axiom.unscoped.joins(:symbols).
where('symbols.id' => OntologyMember::Symbol.
where(ontology_id: ontology.id).joins(:sine_symbol_commonness).
where('sine_symbol_commonnesses.commonness < ?', commonness_threshold).
map(&:id))
end
def depth_limit_reached?(current_depth)
depth_limited? && current_depth >= depth_limit
end
def depth_limited?
depth_limit != -1
end
end