Sha256: bb43d66b1db3985fb5cf3e66a979d0a3cafd63434cd639f7d689db9ec0b5dd93

Contents?: true

Size: 1.17 KB

Versions: 2

Compression:

Stored size: 1.17 KB

Contents

# Builtin Contracts
class  Any;     end
module Boolean; end
TrueClass.send(:include, Boolean)
FalseClass.send(:include, Boolean)

class Module
  private
  def __haskell__
    prepend (@__haskell__ = Module.new) unless @__haskell__
    @__haskell__
  end

  def typesig(hash)
    meth = hash.keys.first
    *arg_types, type_pair = hash.values.first

    __haskell__.send(:define_method, meth) do |*args, &block|
      ::Haskell.assert_arg_type(meth, args, arg_types << type_pair.keys.first)
      rtn = super(*args, &block)
      ::Haskell.assert_trn_type(meth, rtn, type_pair.values.first)
      rtn
    end
    self
  end
end

module Haskell
  class << self
    def assert_arg_type(meth, args, klasses)
      args.each_with_index do |arg, i|
        if wrong_type?(arg, klasses[i])
          raise ArgumentError, "Wrong type of argument, type of #{arg.inspect} should be #{klasses[i]}"
        end
      end
    end

    def assert_trn_type(meth, rtn, klass)
      if wrong_type?(rtn, klass)
        raise TypeError, "Expected #{meth} to return #{klass} but got #{rtn.inspect} instead"
      end
    end

    def wrong_type?(obj, klass)
      !(obj.is_a?(klass) || klass == Any)
    end
  end
end

Version data entries

2 entries across 2 versions & 1 rubygems

Version Path
haskell-0.2.3 lib/haskell.rb
haskell-0.2.2 lib/haskell.rb