Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: opal/opal
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: f43a481e5af0
Choose a base ref
...
head repository: opal/opal
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: 86bdd4e4effb
Choose a head ref
  • 2 commits
  • 4 files changed
  • 2 contributors

Commits on Apr 7, 2014

  1. Hash fixes: low-hanging fruit.

      * Hash#delete accepts a block for default value
      * Hash#default= and Hash#default_proc= unset their counterpart.
      * Hash#merge/Hash#update try to coerce with :to_hash
      * Hash#== Checks for value existence in rhs before comparing
      * Hash[] throws proper errors for more malformed argument list "geometries".
    
    Also closes #522 (value existence on rhs)
    mieko committed Apr 7, 2014

    Verified

    This commit was created on GitHub.com and signed with GitHub’s verified signature. The key has expired.
    Copy the full SHA
    331a412 View commit details
  2. Merge pull request #524 from mieko/hash_default

    Hash fixes: low-hanging fruit.
    adambeynon committed Apr 7, 2014
    Copy the full SHA
    86bdd4e View commit details
Showing with 66 additions and 37 deletions.
  1. +38 −14 opal/corelib/hash.rb
  2. +14 −0 opal/corelib/module.rb
  3. +14 −2 opal/corelib/runtime.js
  4. +0 −21 spec/filters/bugs/hash.rb
52 changes: 38 additions & 14 deletions opal/corelib/hash.rb
Original file line number Diff line number Diff line change
@@ -13,21 +13,17 @@ def self.allocate
hash.map = {};
hash.keys = [];
hash.none = nil;
hash.proc = nil;
return hash;
}
end

def initialize(defaults = undefined, &block)
%x{
if (defaults != null) {
self.none = defaults;
}
else if (block !== nil) {
self.proc = block;
}
return self;
self.none = (defaults === undefined ? nil : defaults);
self.proc = block;
}
end

@@ -50,8 +46,7 @@ def ==(other)
for (var i = 0, length = self.keys.length; i < length; i++) {
var key = self.keys[i], obj = map[key], obj2 = map2[key];
if (#{`obj` != `obj2`}) {
if (obj2 === undefined || #{`obj` != `obj2`}) {
return false;
}
}
@@ -141,22 +136,40 @@ def clone
end

def default(val = undefined)
@none
%x{
if (val !== undefined && self.proc !== nil) {
return #{@proc.call(self, val)};
}
return self.none;
}
end

def default=(object)
@none = object
%x{
self.proc = nil;
return (self.none = object);
}
end

def default_proc
@proc
end

def default_proc=(proc)
@proc = proc
%x{
if (proc !== nil) {
proc = #{Opal.coerce_to!(proc, Proc, :to_proc)};
if (#{proc.lambda?} && #{proc.arity.abs} != 2) {
#{raise TypeError, "default_proc takes two arguments"};
}
}
self.none = nil;
return (self.proc = proc);
}
end

def delete(key)
def delete(key, &block)
%x{
var map = self.map, result = map[key];
@@ -167,6 +180,9 @@ def delete(key)
return result;
}
if (block !== nil) {
return #{block.call(key)};
}
return nil;
}
end
@@ -448,6 +464,10 @@ def length

def merge(other, &block)
%x{
if (! #{Hash === other}) {
other = #{Opal.coerce_to!(other, Hash, :to_hash)};
}
var keys = self.keys, map = self.map,
result = $opal.hash(), keys2 = result.keys, map2 = result.map;
@@ -491,6 +511,10 @@ def merge(other, &block)

def merge!(other, &block)
%x{
if (! #{Hash === other}) {
other = #{Opal.coerce_to!(other, Hash, :to_hash)};
}
var keys = self.keys, map = self.map,
keys2 = other.keys, map2 = other.map;
14 changes: 14 additions & 0 deletions opal/corelib/module.rb
Original file line number Diff line number Diff line change
@@ -304,6 +304,20 @@ def include(*mods)
self
end

def include?(mod)
%x{
for (var cls = self; cls; cls = cls.parent) {
for (var i = 0; i != cls.__inc__.length; i++) {
var mod2 = cls.__inc__[i];
if (mod === mod2) {
return true;
}
}
}
return false;
}
end

def instance_method(name)
%x{
var meth = self._proto['$' + name];
16 changes: 14 additions & 2 deletions opal/corelib/runtime.js
Original file line number Diff line number Diff line change
@@ -762,7 +762,14 @@
var args = arguments[0];

for (var i = 0, length = args.length; i < length; i++) {
var key = args[i][0], obj = args[i][1];
var pair = args[i];

if (pair.length !== 2) {
throw Opal.ArgumentError.$new("value not of length 2: " + pair.$inspect());
}

var key = pair[0],
obj = pair[1];

if (assocs[key] == null) {
keys.push(key);
@@ -780,7 +787,12 @@
}
}
else {
for (var i = 0, length = arguments.length; i < length; i++) {
var length = arguments.length;
if (length % 2 !== 0) {
throw Opal.ArgumentError.$new("odd number of arguments for Hash");
}

for (var i = 0; i < length; i++) {
var key = arguments[i],
obj = arguments[++i];

21 changes: 0 additions & 21 deletions spec/filters/bugs/hash.rb
Original file line number Diff line number Diff line change
@@ -1,32 +1,17 @@
opal_filter "Hash" do
fails "Hash includes Enumerable"

fails "Hash#assoc only returns the first matching key-value pair for identity hashes"

fails "Hash.[] creates a Hash; values can be provided as a list of value-pairs in an array"
fails "Hash.[] coerces a single argument which responds to #to_ary"
fails "Hash.[] ignores elements that are not arrays"
fails "Hash.[] raises an ArgumentError for arrays of more than 2 elements"
fails "Hash.[] raises an ArgumentError when passed a list of value-invalid-pairs in an array"
fails "Hash.[] raises an ArgumentError when passed an odd number of arguments"
fails "Hash.[] calls to_hash"
fails "Hash.[] returns an instance of a subclass when passed an Array"
fails "Hash.[] returns instances of subclasses"
fails "Hash.[] returns an instance of the class it's called on"
fails "Hash.[] does not call #initialize on the subclass instance"
fails "Hash.[] passed an array treats elements that are 2 element arrays as key and value"
fails "Hash.[] passed an array treats elements that are 1 element arrays as keys with value nil"
fails "Hash.[] passed a single argument which responds to #to_hash coerces it and returns a copy"

fails "Hash#default_proc= uses :to_proc on its argument"
fails "Hash#default_proc= overrides the static default"
fails "Hash#default_proc= raises an error if passed stuff not convertible to procs"
fails "Hash#default_proc= raises a TypeError if passed a lambda with an arity other than 2"

fails "Hash#default uses the default proc to compute a default value, passing given key"
fails "Hash#default= unsets the default proc"

fails "Hash#delete calls supplied block if the key is not found"

fails "Hash#each properly expands (or not) child class's 'each'-yielded args"
fails "Hash#each yields the key only to a block expecting |key,|"
@@ -42,7 +27,6 @@
fails "Hash#== computes equality for complex recursive hashes"
fails "Hash#== does not compare keys with different hash codes via eql?"
fails "Hash#== first compares keys via hash"
fails "Hash#== does not compare values when keys don't match"

fails "Hash#eql? compares the values in self to values in other hash"
fails "Hash#eql? returns true iff other Hash has the same number of keys and each key-value pair matches"
@@ -96,11 +80,8 @@
fails "Hash#member? compares keys with the same #hash value via #eql?"
fails "Hash#member? returns true if argument is a key"

fails "Hash#merge tries to convert the passed argument to a hash using #to_hash"
fails "Hash#merge returns subclass instance for subclasses"

fails "Hash#merge! tries to convert the passed argument to a hash using #to_hash"

fails "Hash.new raises an ArgumentError if more than one argument is passed"
fails "Hash.new raises an ArgumentError if passed both default argument and default block"

@@ -141,6 +122,4 @@
fails "Hash.try_convert returns nil when the argument does not respond to #to_hash"
fails "Hash.try_convert returns the argument if it's a kind of Hash"
fails "Hash.try_convert returns the argument if it's a Hash"

fails "Hash#update tries to convert the passed argument to a hash using #to_hash"
end