EnumerableMap.test.js 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  1. const { ethers } = require('hardhat');
  2. const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers');
  3. const { mapValues } = require('../../helpers/iterate');
  4. const { generators } = require('../../helpers/random');
  5. const { MAP_TYPES, typeDescr, toMapTypeDescr } = require('../../../scripts/generate/templates/Enumerable.opts');
  6. const { shouldBehaveLikeMap } = require('./EnumerableMap.behavior');
  7. // Add Bytes32ToBytes32Map that must be tested but is not part of the generated types.
  8. MAP_TYPES.unshift(toMapTypeDescr({ key: typeDescr({ type: 'bytes32' }), value: typeDescr({ type: 'bytes32' }) }));
  9. async function fixture() {
  10. const mock = await ethers.deployContract('$EnumerableMap');
  11. const env = Object.fromEntries(
  12. MAP_TYPES.map(({ name, key, value }) => [
  13. name,
  14. {
  15. key,
  16. value,
  17. keys: Array.from({ length: 3 }, generators[key.type]),
  18. values: Array.from({ length: 3 }, generators[value.type]),
  19. zeroValue: generators[value.type].zero,
  20. methods: mapValues(
  21. MAP_TYPES.filter(map => map.key.name == key.name).length == 1
  22. ? {
  23. set: `$set(uint256,${key.type},${value.type})`,
  24. get: `$get(uint256,${key.type})`,
  25. tryGet: `$tryGet(uint256,${key.type})`,
  26. remove: `$remove(uint256,${key.type})`,
  27. contains: `$contains(uint256,${key.type})`,
  28. clear: `$clear_EnumerableMap_${name}(uint256)`,
  29. length: `$length_EnumerableMap_${name}(uint256)`,
  30. at: `$at_EnumerableMap_${name}(uint256,uint256)`,
  31. keys: `$keys_EnumerableMap_${name}(uint256)`,
  32. }
  33. : {
  34. set: `$set(uint256,${key.type},${value.type})`,
  35. get: `$get_EnumerableMap_${name}(uint256,${key.type})`,
  36. tryGet: `$tryGet_EnumerableMap_${name}(uint256,${key.type})`,
  37. remove: `$remove_EnumerableMap_${name}(uint256,${key.type})`,
  38. contains: `$contains_EnumerableMap_${name}(uint256,${key.type})`,
  39. clear: `$clear_EnumerableMap_${name}(uint256)`,
  40. length: `$length_EnumerableMap_${name}(uint256)`,
  41. at: `$at_EnumerableMap_${name}(uint256,uint256)`,
  42. keys: `$keys_EnumerableMap_${name}(uint256)`,
  43. },
  44. fnSig =>
  45. (...args) =>
  46. mock.getFunction(fnSig)(0, ...args),
  47. ),
  48. events: {
  49. setReturn: `return$set_EnumerableMap_${name}_${key.type}_${value.type}`,
  50. removeReturn: `return$remove_EnumerableMap_${name}_${key.type}`,
  51. },
  52. error: key.memory || value.memory ? `EnumerableMapNonexistent${key.name}Key` : `EnumerableMapNonexistentKey`,
  53. },
  54. ]),
  55. );
  56. return { mock, env };
  57. }
  58. describe('EnumerableMap', function () {
  59. beforeEach(async function () {
  60. Object.assign(this, await loadFixture(fixture));
  61. });
  62. for (const { name, key, value } of MAP_TYPES) {
  63. describe(`${name} (enumerable map from ${key.type} to ${value.type})`, function () {
  64. beforeEach(async function () {
  65. Object.assign(this, this.env[name]);
  66. [this.keyA, this.keyB, this.keyC] = this.keys;
  67. [this.valueA, this.valueB, this.valueC] = this.values;
  68. });
  69. shouldBehaveLikeMap();
  70. });
  71. }
  72. });