class Option def or_nil end def ==(that) or_nil == that.or_nil end end class Some < Option def initialize(value) @value = value end def to_a [get] end def get @value end def get_or_else(&blk) get end def foreach(&blk) blk.call(get) nil end def or_nil get end def empty? false end def map(&blk) Option(blk.call(get)) end def flat_map(&blk) result = blk.call(get) case result when Option then return result else raise TypeError, "Must be Option" end end def fold(if_empty, &blk) blk.call(get) end def exists?(&blk) !! blk.call(get) end end class NoneClass < Option def to_a [] end def get raise IndexError, "None.get" end def get_or_else(&blk) blk.call end def foreach(&blk) nil end def or_nil nil end def empty? true end def map(&blk) flat_map(&blk) end def flat_map(&blk) self end def fold(if_empty, &blk) if_empty.call end def exists?(&blk) false end end None = NoneClass.new def Some(value) Some.new(value) end def Option(value) value.nil? ? None : Some(value) end