class Steep::Interface::Substitution
def self.build(vars, types = nil, instance_type: AST::Types::Instance.new, module_type: AST::Types::Class.new, self_type: AST::Types::Self.new)
def self.build(vars, types = nil, instance_type: AST::Types::Instance.new, module_type: AST::Types::Class.new, self_type: AST::Types::Self.new) types ||= vars.map {|var| AST::Types::Var.fresh(var) } raise InvalidSubstitutionError.new(vars_size: vars.size, types_size: types.size) unless vars.size == types.size dic = vars.zip(types).each.with_object({}) do |(var, type), d| d[var] = type end new(dictionary: dic, instance_type: instance_type, module_type: module_type, self_type: self_type) end
def self.empty
def self.empty new(dictionary: {}, instance_type: AST::Types::Instance.new, module_type: AST::Types::Class.new, self_type: AST::Types::Self.new) end
def [](key)
def [](key) dictionary[key] or raise "Unknown variable: #{key}" end
def add!(v, ty)
def add!(v, ty) merge!(Substitution.new(dictionary: { v => ty }, instance_type: nil, module_type: nil, self_type: nil)) end
def except(vars)
def except(vars) self.class.new( dictionary: dictionary.reject {|k, _| vars.include?(k) }, instance_type: instance_type, module_type: module_type, self_type: self_type ) end
def initialize(dictionary:, instance_type:, module_type:, self_type:)
def initialize(dictionary:, instance_type:, module_type:, self_type:) @dictionary = dictionary @instance_type = instance_type @module_type = module_type @self_type = self_type end
def key?(var)
def key?(var) dictionary.key?(var) end
def merge(s)
def merge(s) Substitution.new(dictionary: dictionary.dup, instance_type: instance_type, module_type: module_type, self_type: self_type).merge!(s) end
def merge!(s)
def merge!(s) dictionary.transform_values! {|ty| ty.subst(s) } dictionary.merge!(s.dictionary) do |key, a, b| if a == b a else raise "Duplicated key on merge!: #{key}, #{a}, #{b}" end end self end
def to_s
def to_s a = [] dictionary.each do |x, ty| a << "#{x} -> #{ty}" end a << "[instance_type] -> #{instance_type}" a << "[module_type] -> #{module_type}" a << "[self_type] -> #{self_type}" "{ #{a.join(", ")} }" end