module Ruby module Signature class MethodType class Block attr_reader :type attr_reader :required def initialize(type:, required:) @type = type @required = required end def ==(other) other.is_a?(Block) && other.type == type && other.required == required end def to_json(*a) { type: type, required: required }.to_json(*a) end def sub(s) self.class.new( type: type.sub(s), required: required ) end end attr_reader :type_params attr_reader :type attr_reader :block attr_reader :location def initialize(type_params:, type:, block:, location:) @type_params = type_params @type = type @block = block @location = location end def ==(other) other.is_a?(MethodType) && other.type_params == type_params && other.type == type && other.block == block end def to_json(*a) { type_params: type_params, type: type, block: block, location: location }.to_json(*a) end def sub(s) s.without(*type_params).yield_self do |sub| map_type do |ty| ty.sub(sub) end end end def update(type_params: self.type_params, type: self.type, block: self.block, location: self.location) self.class.new( type_params: type_params, type: type, block: block, location: location ) end def free_variables(set = Set.new) type.free_variables(set) block&.type&.free_variables(set) set.subtract(type_params) end def map_type(&block) self.class.new( type_params: type_params, type: type.map_type(&block), block: self.block&.yield_self do |b| Block.new(type: b.type.map_type(&block), required: b.required) end, location: location ) end def each_type(&block) if block_given? type.each_type(&block) self.block&.yield_self do |b| b.type.each_type(&block) end else enum_for :each_type end end def to_s s = case when block && block.required "(#{type.param_to_s}) { (#{block.type.param_to_s}) -> #{block.type.return_to_s} } -> #{type.return_to_s}" when block "(#{type.param_to_s}) ?{ (#{block.type.param_to_s}) -> #{block.type.return_to_s} } -> #{type.return_to_s}" else "(#{type.param_to_s}) -> #{type.return_to_s}" end if type_params.empty? s else "[#{type_params.join(", ")}] #{s}" end end end end end