mirror of
https://github.com/alliedmodders/sourcemod.git
synced 2025-12-06 18:08:36 +00:00
* Add new IntHashMap and IntMap natives This is a continuation of #579. [HashMap] Adds new IntHashMap and IntMap natives. This patch makes the following changes * Refactors StringHashMap to a generic HashMap template * Adds the IntHashMap class * Adds new IntMap natives * Adds IntMap tests to the tries test suite [HashMap] Reverted rename of CharsAndLength [HashMap] Use more descriptive template names [HashMap] Removed old-style natives [HashMap] Removed IntHash class [HashMap] Reverted some search & replace errors Co-authored-by: Geoffrey McRae <geoff@hostfission.com> * Fix spelling mistake + include * Fix tries test * Update tests with clone + ContainsKey --------- Co-authored-by: Geoffrey McRae <geoff@hostfission.com> Co-authored-by: Nicholas Hastings <nshastings@gmail.com>
350 lines
10 KiB
SourcePawn
350 lines
10 KiB
SourcePawn
// vim: set sts=2 ts=8 sw=2 tw=99 et ft=c :
|
|
#include <sourcemod>
|
|
|
|
public Plugin:myinfo =
|
|
{
|
|
name = "Trie test",
|
|
author = "AlliedModders LLC",
|
|
description = "Trie tests",
|
|
version = "1.0.0.0",
|
|
url = "http://www.sourcemod.net/"
|
|
};
|
|
|
|
public OnPluginStart()
|
|
{
|
|
RegServerCmd("test_maps", RunTests);
|
|
RegServerCmd("test_int_maps", RunIntTests);
|
|
}
|
|
|
|
public Action:RunTests(argc)
|
|
{
|
|
StringMap map = new StringMap();
|
|
|
|
for (new i = 0; i < 64; i++) {
|
|
char buffer[24];
|
|
Format(buffer, sizeof(buffer), "%d", i);
|
|
|
|
if (!map.SetValue(buffer, i))
|
|
ThrowError("set map to %d failed", i);
|
|
|
|
new value;
|
|
if (!map.GetValue(buffer, value))
|
|
ThrowError("get map %d", i);
|
|
if (value != i)
|
|
ThrowError("get map %d == %d", i, i);
|
|
}
|
|
|
|
// Setting 17 without replace should fail.
|
|
new value;
|
|
if (map.SetValue("17", 999, false))
|
|
ThrowError("set map 17 should fail");
|
|
if (!map.GetValue("17", value) || value != 17)
|
|
ThrowError("value at 17 not correct");
|
|
if (!map.SetValue("17", 999))
|
|
ThrowError("set map 17 = 999 should succeed");
|
|
if (!map.GetValue("17", value) || value != 999)
|
|
ThrowError("value at 17 not correct");
|
|
|
|
// Check size is 64.
|
|
if (map.Size != 64)
|
|
ThrowError("map size not 64");
|
|
|
|
// Check "cat" is not found.
|
|
int array[64];
|
|
char string[64];
|
|
if (map.GetValue("cat", value) ||
|
|
map.GetArray("cat", array, sizeof(array)) ||
|
|
map.GetString("cat", string, sizeof(string)))
|
|
{
|
|
ThrowError("map should not have a cat");
|
|
}
|
|
|
|
// Check that "17" is not a string or array.
|
|
if (map.GetArray("17", array, sizeof(array)) ||
|
|
map.GetString("17", string, sizeof(string)))
|
|
{
|
|
ThrowError("entry 17 should not be an array or string");
|
|
}
|
|
|
|
// Strings.
|
|
if (!map.SetString("17", "hellokitty"))
|
|
ThrowError("17 should be string");
|
|
if (!map.GetString("17", string, sizeof(string)) ||
|
|
strcmp(string, "hellokitty") != 0)
|
|
{
|
|
ThrowError("17 should be hellokitty");
|
|
}
|
|
if (map.GetValue("17", value) ||
|
|
map.GetArray("17", array, sizeof(array)))
|
|
{
|
|
ThrowError("entry 17 should not be an array or string");
|
|
}
|
|
|
|
// Arrays.
|
|
new data[5] = { 93, 1, 2, 3, 4 };
|
|
if (!map.SetArray("17", data, 5))
|
|
ThrowError("17 should be string");
|
|
if (!map.GetArray("17", array, sizeof(array)))
|
|
ThrowError("17 should be hellokitty");
|
|
for (new i = 0; i < 5; i++) {
|
|
if (data[i] != array[i])
|
|
ThrowError("17 slot %d should be %d, got %d", i, data[i], array[i]);
|
|
}
|
|
if (map.GetValue("17", value) ||
|
|
map.GetString("17", string, sizeof(string)))
|
|
{
|
|
ThrowError("entry 17 should not be an array or string");
|
|
}
|
|
|
|
if (!map.SetArray("17", data, 1))
|
|
ThrowError("couldn't set 17 to 1-entry array");
|
|
// Check that we fixed an old bug where 1-entry arrays where cells
|
|
if (!map.GetArray("17", array, sizeof(array), value))
|
|
ThrowError("couldn't fetch 1-entry array");
|
|
if (value != 1)
|
|
ThrowError("array size mismatch (%d, expected %d)", value, 1);
|
|
// Check that we maintained backward compatibility.
|
|
if (!map.GetValue("17", value))
|
|
ThrowError("backwards compatibility failed");
|
|
if (value != data[0])
|
|
ThrowError("wrong value (%d, expected %d)", value, data[0]);
|
|
|
|
// Remove "17".
|
|
if (!map.Remove("17"))
|
|
ThrowError("17 should have been removed");
|
|
if (map.Remove("17"))
|
|
ThrowError("17 should not exist");
|
|
if (map.GetValue("17", value) ||
|
|
map.GetArray("17", array, sizeof(array)) ||
|
|
map.GetString("17", string, sizeof(string)))
|
|
{
|
|
ThrowError("map should not have a 17");
|
|
}
|
|
|
|
map.Clear();
|
|
|
|
if (map.Size)
|
|
ThrowError("size should be 0");
|
|
|
|
map.SetString("adventure", "time!");
|
|
map.SetString("butterflies", "bees");
|
|
map.SetString("egg", "egg");
|
|
|
|
StringMapSnapshot keys = map.Snapshot();
|
|
{
|
|
if (keys.Length != 3)
|
|
ThrowError("map snapshot length should be 3");
|
|
|
|
bool found[3];
|
|
for (new i = 0; i < keys.Length; i++) {
|
|
new size = keys.KeyBufferSize(i);
|
|
char[] buffer = new char[size];
|
|
keys.GetKey(i, buffer, size);
|
|
|
|
if (strcmp(buffer, "adventure") == 0)
|
|
found[0] = true;
|
|
else if (strcmp(buffer, "butterflies") == 0)
|
|
found[1] = true;
|
|
else if (strcmp(buffer, "egg") == 0)
|
|
found[2] = true;
|
|
else
|
|
ThrowError("unexpected key: %s", buffer);
|
|
}
|
|
|
|
if (!found[0] || !found[1] || !found[2])
|
|
ThrowError("did not find all keys");
|
|
}
|
|
delete keys;
|
|
|
|
PrintToServer("All tests passed!");
|
|
|
|
delete map;
|
|
return Plugin_Handled;
|
|
}
|
|
|
|
public Action:RunIntTests(argc)
|
|
{
|
|
IntMap map = new IntMap();
|
|
|
|
for (new i = 0; i < 64; i++) {
|
|
if (!map.SetValue(i, i))
|
|
ThrowError("set map to %d failed", i);
|
|
|
|
if (!map.ContainsKey(i))
|
|
ThrowError("map contains %d failed", i)
|
|
|
|
new value;
|
|
if (!map.GetValue(i, value))
|
|
ThrowError("get map %d", i);
|
|
if (value != i)
|
|
ThrowError("get map %d == %d", i, i);
|
|
}
|
|
|
|
// Setting 17 without replace should fail.
|
|
new value;
|
|
if (map.SetValue(17, 999, false))
|
|
ThrowError("set map 17 should fail");
|
|
if (!map.GetValue(17, value) || value != 17)
|
|
ThrowError("value at 17 not correct");
|
|
if (!map.SetValue(17, 999))
|
|
ThrowError("set map 17 = 999 should succeed");
|
|
if (!map.GetValue(17, value) || value != 999)
|
|
ThrowError("value at 17 not correct");
|
|
|
|
// Check size is 64.
|
|
if (map.Size != 64)
|
|
ThrowError("map size not 64");
|
|
|
|
// Check 100 is not found.
|
|
int array[64];
|
|
char string[64];
|
|
if (map.ContainsKey(100) ||
|
|
map.GetValue(100, value) ||
|
|
map.GetArray(100, array, sizeof(array)) ||
|
|
map.GetString(100, string, sizeof(string)))
|
|
{
|
|
ThrowError("map should not have 100");
|
|
}
|
|
|
|
// Check that 17 is not a string or array.
|
|
if (map.GetArray(17, array, sizeof(array)) ||
|
|
map.GetString(17, string, sizeof(string)))
|
|
{
|
|
ThrowError("entry 17 should not be an array or string");
|
|
}
|
|
|
|
// Strings.
|
|
if (!map.SetString(17, "hellokitty"))
|
|
ThrowError("17 should be string");
|
|
if (!map.GetString(17, string, sizeof(string)) ||
|
|
strcmp(string, "hellokitty") != 0)
|
|
{
|
|
ThrowError("17 should be hellokitty");
|
|
}
|
|
if (map.GetValue(17, value) ||
|
|
map.GetArray(17, array, sizeof(array)))
|
|
{
|
|
ThrowError("entry 17 should not be an array or string");
|
|
}
|
|
|
|
// Arrays.
|
|
new data[5] = { 93, 1, 2, 3, 4 };
|
|
if (!map.SetArray(17, data, 5))
|
|
ThrowError("couldn't set 17 to 5-entry array");
|
|
if (!map.GetArray(17, array, sizeof(array)))
|
|
ThrowError("couldn't fetch 5-entry array");
|
|
for (new i = 0; i < 5; i++) {
|
|
if (data[i] != array[i])
|
|
ThrowError("17 slot %d should be %d, got %d", i, data[i], array[i]);
|
|
}
|
|
if (map.GetValue(17, value) ||
|
|
map.GetString(17, string, sizeof(string)))
|
|
{
|
|
ThrowError("entry 17 should not be a value or string");
|
|
}
|
|
|
|
if (!map.SetArray(17, data, 1))
|
|
ThrowError("couldn't set 17 to 1-entry array");
|
|
// Check that we fixed an old bug where 1-entry arrays where cells
|
|
if (!map.GetArray(17, array, sizeof(array), value))
|
|
ThrowError("couldn't fetch 1-entry array");
|
|
if (value != 1)
|
|
ThrowError("array size mismatch (%d, expected %d)", value, 1);
|
|
// Check that we maintained backward compatibility.
|
|
if (!map.GetValue(17, value))
|
|
ThrowError("backwards compatibility failed");
|
|
if (value != data[0])
|
|
ThrowError("wrong value (%d, expected %d)", value, data[0]);
|
|
|
|
// Remove "17".
|
|
if (!map.Remove(17))
|
|
ThrowError("17 should have been removed");
|
|
if (map.Remove(17))
|
|
ThrowError("17 should not exist");
|
|
if (map.ContainsKey(17) ||
|
|
map.GetValue(17, value) ||
|
|
map.GetArray(17, array, sizeof(array)) ||
|
|
map.GetString(17, string, sizeof(string)))
|
|
{
|
|
ThrowError("map should not have a 17");
|
|
}
|
|
|
|
map.Clear();
|
|
|
|
if (map.Size)
|
|
ThrowError("size should be 0");
|
|
|
|
map.SetString(42, "time!");
|
|
map.SetString(84, "bees");
|
|
map.SetString(126, "egg");
|
|
|
|
IntMapSnapshot keys = map.Snapshot();
|
|
{
|
|
if (keys.Length != 3)
|
|
ThrowError("map snapshot length should be 3");
|
|
|
|
bool found[3];
|
|
for (new i = 0; i < keys.Length; i++) {
|
|
decl key = keys.GetKey(i);
|
|
|
|
if (key == 42)
|
|
found[0] = true;
|
|
else if (key == 84)
|
|
found[1] = true;
|
|
else if (key == 126)
|
|
found[2] = true;
|
|
else
|
|
ThrowError("unexpected key: %d", key);
|
|
}
|
|
|
|
if (!found[0] || !found[1] || !found[2])
|
|
ThrowError("did not find all keys");
|
|
}
|
|
delete keys;
|
|
|
|
map.SetValue(10240, 6744);
|
|
map.SetValue(8, 13);
|
|
|
|
new cloneData[5] = { 12, 23, 55, 1, 2 };
|
|
new cloneArr[5];
|
|
map.SetArray(9102, cloneData, 5);
|
|
|
|
IntMap clone = map.Clone();
|
|
|
|
if (clone.Size != map.Size)
|
|
ThrowError("cloned map size mismatch (%d, expected %d)", clone.Size, map.Size);
|
|
|
|
if (!clone.GetString(42, string, sizeof(string)))
|
|
ThrowError("cloned map entry 42 should be a string");
|
|
if (strcmp(string, "time!") != 0)
|
|
ThrowError("cloned map entry 42 should be \"time!\"");
|
|
if (!clone.GetString(84, string, sizeof(string)))
|
|
ThrowError("cloned map entry 84 should be a string");
|
|
if (strcmp(string, "bees") != 0)
|
|
ThrowError("cloned map entry 84 should be \"bees\"");
|
|
if (!clone.GetString(126, string, sizeof(string)))
|
|
ThrowError("cloned map entry 126 should be a string");
|
|
if (strcmp(string, "egg") != 0)
|
|
ThrowError("cloned map entry 126 should be \"egg\"");
|
|
if (!clone.GetValue(10240, value))
|
|
ThrowError("cloned map entry 10240 should be a value");
|
|
if (value != 6744)
|
|
ThrowError("cloned map entry 10240 should be 6744")
|
|
if (!clone.GetValue(8, value))
|
|
ThrowError("cloned map entry 8 should be a value");
|
|
if (value != 13)
|
|
ThrowError("cloned map entry 8 should be 13")
|
|
if (!clone.GetArray(9102, cloneArr, 5))
|
|
ThrowError("cloned map entry 9102 should be an array");
|
|
for (new i = 0; i < 5; i++) {
|
|
if (cloneData[i] != cloneArr[i])
|
|
ThrowError("cloned map entry 9102 slot %d should be %d, got %d", i, cloneData[i], cloneArr[i]);
|
|
}
|
|
|
|
delete clone;
|
|
|
|
PrintToServer("All tests passed!");
|
|
|
|
delete map;
|
|
return Plugin_Handled;
|
|
} |