blob: ac2b2410e3c99d3f401d2b6db49a810816de713b [file] [log] [blame]
// Copyright (c) 2018 Kenton Varda and contributors
// Licensed under the MIT License:
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
#include "map.h"
#include <kj/test.h>
namespace kj {
namespace _ {
namespace {
KJ_TEST("HashMap") {
HashMap<String, int> map;
map.insert(kj::str("foo"), 123);
map.insert(kj::str("bar"), 456);
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("foo"_kj)) == 123);
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("bar"_kj)) == 456);
KJ_EXPECT(map.find("baz"_kj) == nullptr);
map.upsert(kj::str("foo"), 789, [](int& old, uint newValue) {
KJ_EXPECT(old == 123);
KJ_EXPECT(newValue == 789);
old = 321;
});
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("foo"_kj)) == 321);
KJ_EXPECT(
map.findOrCreate("foo"_kj,
[]() -> HashMap<String, int>::Entry { KJ_FAIL_ASSERT("shouldn't have been called"); })
== 321);
KJ_EXPECT(map.findOrCreate("baz"_kj,
[](){ return HashMap<String, int>::Entry { kj::str("baz"), 654 }; }) == 654);
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("baz"_kj)) == 654);
KJ_EXPECT(map.erase("bar"_kj));
KJ_EXPECT(map.erase("baz"_kj));
KJ_EXPECT(!map.erase("qux"_kj));
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("foo"_kj)) == 321);
KJ_EXPECT(map.size() == 1);
KJ_EXPECT(map.begin()->key == "foo");
auto iter = map.begin();
++iter;
KJ_EXPECT(iter == map.end());
map.erase(*map.begin());
KJ_EXPECT(map.size() == 0);
}
KJ_TEST("TreeMap") {
TreeMap<String, int> map;
map.insert(kj::str("foo"), 123);
map.insert(kj::str("bar"), 456);
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("foo"_kj)) == 123);
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("bar"_kj)) == 456);
KJ_EXPECT(map.find("baz"_kj) == nullptr);
map.upsert(kj::str("foo"), 789, [](int& old, uint newValue) {
KJ_EXPECT(old == 123);
KJ_EXPECT(newValue == 789);
old = 321;
});
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("foo"_kj)) == 321);
KJ_EXPECT(
map.findOrCreate("foo"_kj,
[]() -> TreeMap<String, int>::Entry { KJ_FAIL_ASSERT("shouldn't have been called"); })
== 321);
KJ_EXPECT(map.findOrCreate("baz"_kj,
[](){ return TreeMap<String, int>::Entry { kj::str("baz"), 654 }; }) == 654);
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("baz"_kj)) == 654);
KJ_EXPECT(map.erase("bar"_kj));
KJ_EXPECT(map.erase("baz"_kj));
KJ_EXPECT(!map.erase("qux"_kj));
KJ_EXPECT(KJ_ASSERT_NONNULL(map.find("foo"_kj)) == 321);
KJ_EXPECT(map.size() == 1);
KJ_EXPECT(map.begin()->key == "foo");
auto iter = map.begin();
++iter;
KJ_EXPECT(iter == map.end());
map.erase(*map.begin());
KJ_EXPECT(map.size() == 0);
}
KJ_TEST("TreeMap range") {
TreeMap<String, int> map;
map.insert(kj::str("foo"), 1);
map.insert(kj::str("bar"), 2);
map.insert(kj::str("baz"), 3);
map.insert(kj::str("qux"), 4);
map.insert(kj::str("corge"), 5);
{
auto ordered = KJ_MAP(e, map) -> kj::StringPtr { return e.key; };
KJ_ASSERT(ordered.size() == 5);
KJ_EXPECT(ordered[0] == "bar");
KJ_EXPECT(ordered[1] == "baz");
KJ_EXPECT(ordered[2] == "corge");
KJ_EXPECT(ordered[3] == "foo");
KJ_EXPECT(ordered[4] == "qux");
}
{
auto range = map.range("baz", "foo");
auto iter = range.begin();
KJ_EXPECT(iter->key == "baz");
++iter;
KJ_EXPECT(iter->key == "corge");
++iter;
KJ_EXPECT(iter == range.end());
}
map.eraseRange("baz", "foo");
{
auto ordered = KJ_MAP(e, map) -> kj::StringPtr { return e.key; };
KJ_ASSERT(ordered.size() == 3);
KJ_EXPECT(ordered[0] == "bar");
KJ_EXPECT(ordered[1] == "foo");
KJ_EXPECT(ordered[2] == "qux");
}
}
#if !KJ_NO_EXCEPTIONS
KJ_TEST("HashMap findOrCreate throws") {
HashMap<int, String> m;
try {
m.findOrCreate(1, []() -> HashMap<int, String>::Entry {
throw "foo";
});
KJ_FAIL_ASSERT("shouldn't get here");
} catch (const char*) {
// expected
}
KJ_EXPECT(m.find(1) == nullptr);
m.findOrCreate(1, []() {
return HashMap<int, String>::Entry { 1, kj::str("ok") };
});
KJ_EXPECT(KJ_ASSERT_NONNULL(m.find(1)) == "ok");
}
#endif
template <typename MapType>
void testEraseAll(MapType& m) {
m.insert(12, "foo");
m.insert(83, "bar");
m.insert(99, "baz");
m.insert(6, "qux");
m.insert(55, "corge");
auto count = m.eraseAll([](int i, StringPtr s) {
return i == 99 || s == "foo";
});
KJ_EXPECT(count == 2);
KJ_EXPECT(m.size() == 3);
KJ_EXPECT(m.find(12) == nullptr);
KJ_EXPECT(m.find(99) == nullptr);
KJ_EXPECT(KJ_ASSERT_NONNULL(m.find(83)) == "bar");
KJ_EXPECT(KJ_ASSERT_NONNULL(m.find(6)) == "qux");
KJ_EXPECT(KJ_ASSERT_NONNULL(m.find(55)) == "corge");
}
KJ_TEST("HashMap eraseAll") {
HashMap<int, StringPtr> m;
testEraseAll(m);
}
KJ_TEST("TreeMap eraseAll") {
TreeMap<int, StringPtr> m;
testEraseAll(m);
}
} // namespace
} // namespace _
} // namespace kj