Commit c116e504 authored by Kevin Modzelewski's avatar Kevin Modzelewski

Merge pull request #207 from lameiro/dict_methods

Dict methods
parents 1d0a3da4 88b14539
...@@ -46,6 +46,12 @@ Box* dictRepr(BoxedDict* self) { ...@@ -46,6 +46,12 @@ Box* dictRepr(BoxedDict* self) {
return boxString(std::string(chars.begin(), chars.end())); return boxString(std::string(chars.begin(), chars.end()));
} }
Box* dictClear(BoxedDict* self) {
RELEASE_ASSERT(self->cls == dict_cls, "");
self->d.clear();
return None;
}
Box* dictCopy(BoxedDict* self) { Box* dictCopy(BoxedDict* self) {
RELEASE_ASSERT(self->cls == dict_cls, ""); RELEASE_ASSERT(self->cls == dict_cls, "");
...@@ -165,6 +171,22 @@ Box* dictPop(BoxedDict* self, Box* k, Box* d) { ...@@ -165,6 +171,22 @@ Box* dictPop(BoxedDict* self, Box* k, Box* d) {
return rtn; return rtn;
} }
Box* dictPopitem(BoxedDict* self) {
RELEASE_ASSERT(self->cls == dict_cls, "");
auto it = self->d.begin();
if (it == self->d.end()) {
raiseExcHelper(KeyError, "popitem(): dictionary is empty");
}
Box* key = it->first;
Box* value = it->second;
self->d.erase(it);
auto rtn = new BoxedTuple({ key, value });
return rtn;
}
Box* dictGet(BoxedDict* self, Box* k, Box* d) { Box* dictGet(BoxedDict* self, Box* k, Box* d) {
assert(self->cls == dict_cls); assert(self->cls == dict_cls);
...@@ -195,6 +217,19 @@ Box* dictNonzero(BoxedDict* self) { ...@@ -195,6 +217,19 @@ Box* dictNonzero(BoxedDict* self) {
return boxBool(self->d.size()); return boxBool(self->d.size());
} }
Box* dictFromkeys(BoxedDict* self, Box* iterable, Box* default_value) {
RELEASE_ASSERT(self->cls == dict_cls, "");
auto rtn = new BoxedDict();
for (Box* e : iterable->pyElements()) {
dictSetitem(rtn, e, default_value);
}
return rtn;
}
extern "C" Box* dictNew(Box* _cls, BoxedTuple* args, BoxedDict* kwargs) { extern "C" Box* dictNew(Box* _cls, BoxedTuple* args, BoxedDict* kwargs) {
if (!isSubclass(_cls->cls, type_cls)) if (!isSubclass(_cls->cls, type_cls))
raiseExcHelper(TypeError, "dict.__new__(X): X is not a type object (%s)", getTypeName(_cls)->c_str()); raiseExcHelper(TypeError, "dict.__new__(X): X is not a type object (%s)", getTypeName(_cls)->c_str());
...@@ -300,8 +335,12 @@ void setupDict() { ...@@ -300,8 +335,12 @@ void setupDict() {
dict_cls->giveAttr("__iter__", dict_cls->giveAttr("__iter__",
new BoxedFunction(boxRTFunction((void*)dictIterKeys, typeFromClass(dict_iterator_cls), 1))); new BoxedFunction(boxRTFunction((void*)dictIterKeys, typeFromClass(dict_iterator_cls), 1)));
dict_cls->giveAttr("clear", new BoxedFunction(boxRTFunction((void*)dictClear, NONE, 1)));
dict_cls->giveAttr("copy", new BoxedFunction(boxRTFunction((void*)dictCopy, DICT, 1))); dict_cls->giveAttr("copy", new BoxedFunction(boxRTFunction((void*)dictCopy, DICT, 1)));
dict_cls->giveAttr("has_key", new BoxedFunction(boxRTFunction((void*)dictContains, BOXED_BOOL, 2)));
dict_cls->giveAttr("fromkeys",
new BoxedFunction(boxRTFunction((void*)dictFromkeys, DICT, 3, 1, false, false), { None }));
dict_cls->giveAttr("items", new BoxedFunction(boxRTFunction((void*)dictItems, LIST, 1))); dict_cls->giveAttr("items", new BoxedFunction(boxRTFunction((void*)dictItems, LIST, 1)));
dict_cls->giveAttr("iteritems", dict_cls->giveAttr("iteritems",
new BoxedFunction(boxRTFunction((void*)dictIterItems, typeFromClass(dict_iterator_cls), 1))); new BoxedFunction(boxRTFunction((void*)dictIterItems, typeFromClass(dict_iterator_cls), 1)));
...@@ -314,6 +353,8 @@ void setupDict() { ...@@ -314,6 +353,8 @@ void setupDict() {
dict_cls->giveAttr("iterkeys", dict_cls->getattr("__iter__")); dict_cls->giveAttr("iterkeys", dict_cls->getattr("__iter__"));
dict_cls->giveAttr("pop", new BoxedFunction(boxRTFunction((void*)dictPop, UNKNOWN, 3, 1, false, false), { NULL })); dict_cls->giveAttr("pop", new BoxedFunction(boxRTFunction((void*)dictPop, UNKNOWN, 3, 1, false, false), { NULL }));
dict_cls->giveAttr("popitem", new BoxedFunction(boxRTFunction((void*)dictPopitem, BOXED_TUPLE, 1)));
dict_cls->giveAttr("get", new BoxedFunction(boxRTFunction((void*)dictGet, UNKNOWN, 3, 1, false, false), { None })); dict_cls->giveAttr("get", new BoxedFunction(boxRTFunction((void*)dictGet, UNKNOWN, 3, 1, false, false), { None }));
......
...@@ -114,3 +114,52 @@ print d2, d ...@@ -114,3 +114,52 @@ print d2, d
d = {} d = {}
print d.__init__((('a', 1), ('b', 2)), b=3) print d.__init__((('a', 1), ('b', 2)), b=3)
print sorted(d.items()) print sorted(d.items())
# clear
d = {1:2, 'a': 'b'}
d.clear()
print d
d = {}
d.clear()
print d
# fromkeys
d = {1:2, 3:4}
print sorted(d.fromkeys([1,2]).items())
print sorted(d.fromkeys([]).items())
print sorted(d.fromkeys([3,4], 5).items())
try:
print d.fromkeys()
assert 0
except TypeError, e:
print 'ok'
# has_key
d = {1:2, 3:4, 'a': 5}
print d.has_key(1)
print d.has_key(42)
print d.has_key('a')
print d.has_key('b')
# popitem - actual order is implementation defined
d = {1:2, 3:4, 'a': 5}
l = []
l.append(d.popitem())
l.append(d.popitem())
l.append(d.popitem())
print sorted(l)
try:
d.popitem()
assert 0
except KeyError, e:
print 'ok'
# expected: fail
# (del statement)
# Not sure how important or necessary this behavior is, but # Not sure how important or necessary this behavior is, but
# CPython's behavior around sys.modules seems to be that it saves # CPython's behavior around sys.modules seems to be that it saves
# an internal reference to the object, so if you set sys.modules # an internal reference to the object, so if you set sys.modules
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment