From 8b895c7a81708c8fb7920a084fe4c953dddecfd6 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:36:11 -0600 Subject: [PATCH 01/23] docs(store/encoding): fix an error (change 0x0010 to 0x000d) (#2075) --- docs/pages/store/encoding.mdx | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/docs/pages/store/encoding.mdx b/docs/pages/store/encoding.mdx index 2b300f4e00..1f2119621e 100644 --- a/docs/pages/store/encoding.mdx +++ b/docs/pages/store/encoding.mdx @@ -325,17 +325,17 @@ It is used as an onchain gas optimization for storage operations on static lengt Example: `FieldLayout` with 3 fields (`uint64`, `uint40`, `address[]`). ``` -0x 0010 02 01 08080000000000000000000000000000000000000000000000000000 +0x 000D 02 01 08050000000000000000000000000000000000000000000000000000 |----|--|--|--------------------------------------------------------| | | | | Up to 28 static byte lengths, one byte each | - | | | | 8 (uint64), 5 (uint40), 0 (address[]) | + | | | | 8 (uint64), 5 (uint40) | |----|--|--|--------------------------------------------------------| | | | Number of dynamic length fields | | | | 1 | - |----|--|--|--------------------------------------------------------| + |----|--|-----------------------------------------------------------| | | Number of static length fields | | | 2 | - |----|--|--|--------------------------------------------------------| + |----|--------------------------------------------------------------| | Total byte length of all static length fields | | 13 | ``` From 248a172a31234abb5ffaa18850fb6cc902ea2ab2 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:38:09 -0600 Subject: [PATCH 02/23] docs(cli): add `mud test` (#2005) --- docs/pages/cli/test.mdx | 114 +++++++++++++++++++++++++++++++++++++++- 1 file changed, 113 insertions(+), 1 deletion(-) diff --git a/docs/pages/cli/test.mdx b/docs/pages/cli/test.mdx index 3f17bf7649..1a6c4ff5bd 100644 --- a/docs/pages/cli/test.mdx +++ b/docs/pages/cli/test.mdx @@ -1,3 +1,115 @@ # mud test -**Coming soon** +This command runs the tests in a MUD project. Internally, it runs the following steps: + +1. Starts an [`anvil`](https://book.getfoundry.sh/reference/anvil/) instance. + {/* This could be either an empty blockchain, or one that can be forked for a [fork test](https://book.getfoundry.sh/forge/fork-testing). */} +1. Deploys the `World` and all related `System`s using [`mud deploy`](./deploy). +1. Runs tests using [`forge test`](https://book.getfoundry.sh/forge/tests) and passes the deployed world address to the tests via the `WORLD_ADDRESS` environment variable. + +## Command line options + +| Option | Meaning | Type | Default value | +| ----------------------- | ---------------------------------------------- | ------- | ---------------------------------------------------------- | +| `--version` | Show version number | boolean | `false` | +| `--configPath` | Path to the config file | string | `mud.config.ts` | +| `--printConfig` | Print the resolved config | boolean | `false` | +| `--saveDeployment` | Save the deployment info to a file | boolean | `true` | +| `--profile` | The foundry profile to use | string | `local` | +| `--srcDir` | Source directory | string | Foundry `src` directory | +| `--skipBuild` | Skip rebuilding the contracts before deploying | boolean | `false` | +| `--alwaysRunPostDeploy` | Run `PostDeploy.s.sol` after each deploy | boolean | `false` (run the script only when deploying a new `World`) | +| `--port` | Port for the testing `anvil` instance | number | 4242 | +| `--help` | Show help | boolean | `false` | + +## Examples + +```sh copy +pnpm mud test +``` + +## Writing MUD tests + +MUD test contracts inherit from [`MudTest`](https://github.com/latticexyz/mud/blob/main/packages/world/test/MudTest.t.sol). +This contract gets the `World` address from the `$WORLD_ADDRESS` environment variable and sets it as the `Store` address. + +
+ +Line by line explanation of a test + +This is an explanation of [the test](https://github.com/latticexyz/mud/blob/main/templates/react/packages/contracts/test/TasksTest.t.sol) for the [React template](https://github.com/latticexyz/mud/tree/main/templates/react) contracts. + +```solidity +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +import "forge-std/Test.sol"; +import { MudTest } from "@latticexyz/world/test/MudTest.t.sol"; +``` + +Import the general definitions required in all MUD tests. + +```solidity +import { IWorld } from "../src/codegen/world/IWorld.sol"; +import { Tasks, TasksData } from "../src/codegen/index.sol"; +``` + +Import the definitions required for this test, the `World` we can access and the tables we'll use. + +```solidity +contract TasksTest is MudTest { + function testWorldExists() public { +``` + +MUD tests are [Foundry tests](https://book.getfoundry.sh/forge/tests). +Any public function that starts with `test` is a test that gets executed. + +```solidity + uint256 codeSize; + address addr = worldAddress; +``` + +The `World` address comes from the [`MudTest`](https://github.com/latticexyz/mud/blob/main/packages/world/test/MudTest.t.sol#L11). + +```solidity + assembly { + codeSize := extcodesize(addr) + } + assertTrue(codeSize > 0); + } +``` + +Use [`extcodesize`](https://www.evm.codes/#3b?fork=shanghai) to get the size of the `World` contract. +If the deploy process failed, there wouldn't be any code there. + +```solidity + function testTasks() public { + // Expect task to exist that we created during PostDeploy script + TasksData memory task = Tasks.get("1"); +``` + +Use the structure for a table entry's values that is created as part of code generation. + +```solidity + assertEq(task.description, "Walk the dog"); + assertEq(task.completedAt, 0); +``` + +Verify the information that is prepopulated by [the `PostDeploy.s.sol` script](https://github.com/latticexyz/mud/blob/main/templates/react/packages/contracts/script/PostDeploy.s.sol). + +```solidity + // Expect the task to be completed after calling completeTask from our TasksSystem + IWorld(worldAddress).completeTask("1"); +``` + +Call a `System` to modify the table data. + +```solidity + assertEq(Tasks.getCompletedAt("1"), block.timestamp); + } +} +``` + +Verify that the call changed the data correctly. + +
From c6e1eaa981701e2cbc3ae6b26a2696a123e6f1ce Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:41:04 -0600 Subject: [PATCH 03/23] docs(world/systems): explain `SwitchSystem` (#1954) Co-authored-by: alvarius --- docs/pages/world/systems.mdx | 64 ++++++++++++++++++++++++++++++++++++ 1 file changed, 64 insertions(+) diff --git a/docs/pages/world/systems.mdx b/docs/pages/world/systems.mdx index 097290e05b..209ec0af64 100644 --- a/docs/pages/world/systems.mdx +++ b/docs/pages/world/systems.mdx @@ -54,6 +54,70 @@ There are two ways to call one `System` from another one. | `_msgValue()` | zero | can use [`WorldContextProvider`](https://github.com/latticexyz/mud/blob/main/packages/world/src/WorldContext.sol#L180-L203) to transfer the correct information | | Can be used by systems in the root namespace | No (it's a security measure) | Yes | +#### Calling from a root `System` + +If you need to call a `System` from a `System` in the root namespace you can use [`SystemSwitch`](https://github.com/latticexyz/mud/blob/main/packages/world-modules/src/utils/SystemSwitch.sol). + +1. Import `SystemSwitch`. + + ```solidity copy + import { SystemSwitch } from "@latticexyz/world-modules/src/utils/SystemSwitch.sol"; + ``` + +1. Import the interface for the system you wish to call. + + ```solidity copy + import { IIncrementSystem } from "../codegen/world/IIncrementSystem.sol"; + ``` + +1. Call the function using `SystemSwitch.call`. + For example, here is how you can call [`IncrementSystem.increment()`](/templates/typescript/contracts#incrementsystemsol). + + ```solidity copy + uint32 returnValue = abi.decode( + SystemSwitch.call( + abi.encodeCall(IIncrementSystem.increment, ()) + ), + (uint32) + ); + ``` + +
+ + Explanation + + ```solidity + abi.encodeCall(IIncrementSystem.increment, ()) + ``` + + Use [`abi.encodeCall`](https://docs.soliditylang.org/en/latest/cheatsheet.html#abi-encoding-and-decoding-functions) to create the calldata. + The first parameter is a pointer to the function. + The second parameter is a [tuple](https://docs.soliditylang.org/en/latest/control-structures.html#destructuring-assignments-and-returning-multiple-values) with the function parameters. + In this case, there aren't any. + + The advantage of `abi.encodeCall` is that it checks the types of the function parameters are correct. + + ```solidity + SystemSwitch.call( + abi.encodeCall(...) + ) + ``` + + Using `SystemSwitch.call` with the calldata created by `abi.encodeCall`. + `SystemSwitch.call` takes care of figuring out details, such as what type of call to use. + + ```solidity + uint32 retval = abi.decode( + SystemSwitch.call(...), + (uint32) + ); + ``` + + Use [`abi.decode`](https://docs.soliditylang.org/en/latest/cheatsheet.html#abi-encoding-and-decoding-functions) to decode the call's return value. + The second parameter is the data type (or types if there are multiple return values). + +
+ ## Registering systems For a `System` to be callable from a `World` it has to be [registered](https://github.com/latticexyz/mud/blob/main/packages/world/src/modules/core/implementations/WorldRegistrationSystem.sol#L97-L201). From 545486865772abb8b1ce2b68decba5fb87e92fa0 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:42:46 -0600 Subject: [PATCH 04/23] docs(contrib): add how to use the modified source code (#2025) --- docs/pages/contribute.mdx | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/docs/pages/contribute.mdx b/docs/pages/contribute.mdx index 002d942b9f..c8747ff53d 100644 --- a/docs/pages/contribute.mdx +++ b/docs/pages/contribute.mdx @@ -1,11 +1,11 @@ -## Contribute +# Contribute We'd love your support in improving MUD! [The MUD monorepo](https://github.com/latticexyz/mud) includes all of MUD's source code, and pull requests are always welcome. To discuss new features or changes [join our Discord](https://lattice.xyz/discord). You can find a list of good first issues (that don't require a lot of context on the full system) at [contribute.mud.dev](https://contribute.mud.dev). -### Local development setup +## Local development setup The following steps are only necessary if you want to contribute to MUD. To use MUD in your project, [set up a new project with the MUD CLI](/quick-start). @@ -32,7 +32,18 @@ The following steps are only necessary if you want to contribute to MUD. To use pnpm build ``` -### Pull requests +### Using your repository in an application + +To check the changes in an application, run these commands at the application's root. + +```sh copy +pnpm mud set-version --link +pnpm install +``` + +While the application's `pnpm dev` is running, any time you `pnpm build` the MUD repository the application gets updated automatically. + +## Pull requests MUD follows the [conventional commit specification](https://www.conventionalcommits.org/en/v1.0.0/) for PR titles. Please keep the scope of your PR small (rather open multiple small PRs than one huge PR). From b14237bc43f9e1416e331f2852d28d108e8698d9 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:47:17 -0600 Subject: [PATCH 05/23] docs(guides): add client code to the extending guide (#1961) Co-authored-by: alvarius --- docs/pages/guides/extend-ui-with-msgs.png | Bin 0 -> 91051 bytes docs/pages/guides/extending-world.mdx | 355 ++++++++++++++++++++-- 2 files changed, 336 insertions(+), 19 deletions(-) create mode 100644 docs/pages/guides/extend-ui-with-msgs.png diff --git a/docs/pages/guides/extend-ui-with-msgs.png b/docs/pages/guides/extend-ui-with-msgs.png new file mode 100644 index 0000000000000000000000000000000000000000..0553f613bb9976d3fd194d4dfc67d4dd9c3c4368 GIT binary patch literal 91051 zcmZU)1y~%LMVEmGXw-6;--JH_3#XmKs>?(XjH?k=UcQ>^&m4*%=NzQ5;v_u0Lj zolG*BBxA{3n4-J{B0MfU7#J9$l%%K<7#K7z7#MgbEHo&Gddg=U4D6$jrHF{4l!yqi zqJy28rL`#-m}FRzI*f+$7-qJ1e5?>Gija`>Z#k4Q@WQ}Zzy&NP0F^iq8_hqQKv%{7 z%m^H>oi^X#!(jBXCc39SeJS;cQD1eZ6u9nv(#6#MRMz{O2G11fxi48S~g|2xlQkEnkSpR=%=S3@oe<1~?Y)dYig1p`Bg7$d{hX-+dTX zz5Ehkz))enQR$xnU^K(?I!BA2Lv3gzsak<VV;OYtHakG( zS@HQfi9f&R&;V2Cz!;v7NuL;g-KP9D}nLJ`r6n8s5ko6WqDfQYV}VdfOpPQm{bpqk4LV`{HS0^>ntH4KfRYSd>iJ>OY4Nuo;lWQH<`k_c$O+e&C>=5psH(=7iGWt(q$juJR%S*i94fuciKtNJ z;Q%J^FTRNW;E*zmp_6gXce zd!5*Pb+B;acjs7Dy)2>R{1!B7#!5bHJAulp zH=h~g+ASPRlYEs(>t3<|P6MfrB+zIBVR@pS>#+AD-j~#Fh=B_j=D#)}uODm`+yK9c z`N(z1oKS6}u#zRehqK~30bUhV8CiWMKiEQY@B@k0k;_F)b%?kiF+p#_DSeBlSGMXgv1>#FKKf4ze2NaLwhVv~69!TI=9T;@41n>(!o4?0; z3s{r+G!GsaKi9G?!=2*0U|T_bJjGrNY%-MCbF+8mSF5ZBw}T6JyUL9Q61_oZSMnt& zrD&ztrZA0n1NuxJd2ws0LQ5nASGa>dV!n{VV&i|BamOB57+OFCd%PEWeDXl6d9WpZ zve|0m_5s^=t?cP}pLJ#Yg$?G}xqT=2kkw$^uMolmW{!Mi5>HAIGQkA(keu6>WdMR zB&;U5tS~eZ$s|^Ba*~MkFx;V;LenG!qYQiCb5Wh*0(scwe#iNgrHDNErup>jsB(aj z{FUFLjo4Wci~#k5h2Qdx;QW4iLN$3=zY!XdAyavZW73S}vC)i;>oj|+vG`RF@?Gm~ZqsL#c*Yna2incFx zTM|fO4_g4c5-1g@1o&V`w*$6=zAmF1=qgJ3CC5VM#@~}ADpe*^T1QexqLPY`$_0h! z)3YRbJjqzxosdPr8(R=`8k$3QzfjUfb{nCwDbzCQmyuM$tuw*fKABim!^rY z3JU@WO{0q@L=N<63fTZX*-d~gr9Qc9oS|&1Mc=g!9$76)%bO z2tG!-B^bxk;&i&0T^lZ~3;5=Lsj@pYY%}alSxjk2X_UZ0NhRkYdn*Si=P0M1O2~1> zS&;FV0XLb06YLaH72+AL@SleH4nY zuhHTKWceu)TTyk6o+At~4Rvv|`n2Aal178(wk3*`Xilp>dCT!F7or3jYdmi>*zPD&oqPm4F*KS&DH zqt9`7S&E$|Z|j;C(9@G2s2v3 z@BN8J_4_U`AK;SVvdtb%874fE&r-VhI(VswOGry3Mlq`~y2lY`He#Ozm^kgU^?bg6 z{qF6jB`n@292ZTkFwfKH)V~KuU?lOatkp4pW-tlOlOis@N=cMQZB?Zd!DnyH#HC)} z1UdXAf;K`xRywWGuKv#C$6hlHk}{^OyB5CI%bJQ8t=-oN_G`M{}1c_>3lRPE(i7(=8bt>Exu&tOTAC_v*2AO-G_v)H@lQGPAP6n_<2!cf5n$ z>!9m3tX2k%hHY(}o>MQ-up;)D64V)V=tFRMa;*Q1WyNMpZzw*4bPnGVezS0`P$gz zb+TlWZA#l*${<`>Ffi#Pp$U%&#jO&5th|8NXm(GJEj&E&wXCq)g`Z(J>a1cTl^P51^ zE78H}{KuQ|(8yDCM9h8WsK8~Zsn)VO_ZIDAcp(#>Cx;@s+rnpeBKk+J0N$C zBg_oj43{3Tb9VONhs(BW%ocK^XET@1P5ZLb*t3%3$Y^vy3M6ArE2`K2jq3CQ!je(7 zvyMXd_v_(QyezIquAGhWp4(%WA7?YJW*Zdi3>_uTHNUzqUylV(-%BrUPWi5gAB5Ki z218s4FA2Nc=)GTxtj^Jn6T(Z(6s)q_1bEj8R=tlNe;?{H4KfMbC%%=vnlH$7q&&aF zzf@GYs@N^jja9rYA#7oHuiP!Z+Q{hZY`waExDe=JdNz14e3m#EVarhxWpuN#|3e3i&y5F^v@vxuBzCj0wsquj<0t*A1P>_vC!2|s_^%>P zR{W$Ia*D(vb`GY*oQ%wj%%lSF#KgpW4kl(iN}}Tbpo6~nNiCe5?0J}&TwPrmUD+7z z9L$+mxVgERm|2-vSs6ej7#!VgoebR=Y#qt|Ch{*GQBy}_2TOY=OFLWQKXeU^?3|tW zNlE`y^xyS2pQdh>|EtN?@tgH!+kw$rY|$b}&OzFozA0Rvf=yBna-@8m*xeKGc$9Df=My4S%$EW3im zw8Z;jftYmz4-@-`d0#01a1bN8xJaDApr*+C0s^w2@Bd~|lGyD}b=h?WGJph7p;>(O zyYPRS21o;CC=%-4MM0r{1`&tz0$cI=Tad97Aofi{k_iw{LE`W?AzuFYpCoHQLIC1o zAjE@okPf0Cgm!rSuNZkjr8{>AmB>;4i1C+n^nKi4%!BwJ!~#`uDV9A$0gf6P2g1Yt zRQDeY;SbC|3mK6Ngw}6SyQ2C}b+Dg682}UF6+~26u|LUgDl1ff^QcG+;t^du&KuSY z1n_rvTIT$ZM^})9S_}FeL2?up21!}*BP5>iKSKTj5=f|jAeCfq$seVoQ_+$AZ9d?r zuwWoQz1I5PnIN=` zk~@pSDKZrek`9OnIWq_eb=kMB4K5l)4vNdfn)x4n2J(Z*7{0@!fW*iIO7`(`TI2i$ z6LJGt6Uyf9%;s|rSP=9(+X32tn)Z?bY1huZ{!dehP;+OuMN%vO)u)#v2!?D}Pm5{~ zGPwXeYCf@lnoN!|4{9pZEwJ_*Im#v|8F>HDaQ`>^eISY=z-UBmOx!vE#1^~r`@5r||LLtmUrclS!$1H7T*T&}e~=`5 z@j!a%%o}6{)pZE~382&9P(-Si2?~`;o_?W1Ein>@ZS*BbW`Dg6;k!E$9mr;3KZ2U< zw=-E1_(xkiymzPE|7en2<&Tn&N9#7oQJ|rI$yki0vUH&{ChA}9k5Rm@chY#?ou=6E zx=id(sQ3Hwbfm_@8L$d!&K1U!kMTR4EHbn?+@`OqT-Y-&TQPAZK|Ed`7q7qE>mSW| z=h_-GH%8gmQ~2J)8VC8L%^&XkFs3pYneKAf-Zp%^=y}&c2)X<+jlWVjuvVi1B|sn` zjzq{~L=f^M?*)&zPsm8IKarf==}s8iXsI+drfhC{@fN^pIWF5axV4b?b{r)44$|?I zVne~dN@x9}^hw2fGIA6TCF*DdcB?Nk(kV%ZFa6E15P&nzk_;%>Ok z$5oklCOgIooDP?Zgqfa91A>vmX_$+9oml3HU|f1R&C zdQZ5x?-aG{J?Z;!GhL=ap`wG-0Qa?Z`9I487we@03^d!Une6ljU0u%zPQ;N(>>V#u z_%GN8!DGa~JYLK6`F|(^1i{DCsWTqU6-7Le-KNTB^QA$e3_>B{ecn`UE1%1A^P|$xxq3|lvoh}DNwWN^-3}3?1NMTta?6kef&NiqnBh# zf(nZc4r>yyb+S*I*yot**aH6D1~6ABwbuo{QDr`vd7V)C;9liy?V{Dd_iG4Vs(~_!{0r$lprQ*i^y7edWZW*5{EEWK>-R2R>GDLHo44~W zn7!*CTcR{*v{cmWg^0fuqj5Mi%K6p$D$M&UcFKmC=$JBrm&}*&ve=j&Z)qN^r9SEv zhx-+ri9gM!HAnh7Me;~eB|>ce8};lZwxa`stF&9B@6R_(&)3@+oz2G+rC3cyiPu`} zT)FL#Xw@pIE7Yo3L)M{j{AL86WmlVRDzq0`zZsY5wAEWL*9s3u;uOs6cw}(e_*X7i z%TuGmCeF^5y;{ZdYhz*srqt*pIUX`e{t^wN2+-Qk;B#k9IYGc;NckYh;L1G$i5jW^ zfia#EC$6v*F&7xNH3O``lCL)z1JRiBVEoPqyxwW-IB%Ds71WZSD^Z||NU1iVW~It? zIm|Xzn-09=3=v5D0jnE9nxf4H5G4ClGLu6~MVGYLFbo zSGIu7oCV(gCV9FZc)NP3*!)2z5kof<)U*E`WB9oH`MgOVQ^#(xy{l5QrEi}Pxj#av z;$TLmruvi5Qtn~WSAmLwJ_;y!P?ztgu8aDo+Xz9pKzzf*?S4GKv)t@#Q^gt#0vVt5 zQ{nLO0=EiB7Isu`_AvWgsaQhnS#vw1vtAPXnp>e z+lOqSkQV)QTF9my1T${D45XeE6KZ~W3>8t7d<-p1FPZz5)yk5Qy62ZA& z?ZtUN-=$5EGz<;$JuQMzpjM~?WdP5HC;{6;zOk$>6{;1<6u4(wJ^BuZ(>6Vm-l#wt zKOtB|6qI8+@V?ZNh2_@G%CQEea+5F0Dx0qcshRwqdPF5kEK?1pQ@V|Q>TFW<`YdbkoG}@1 za!sZazfh5GRQ^)d|56dVffL`6t~w^E^^~Fomrk##M5Eju$OjZc>KeC;Xn*e?l3{GY zqJ7BaOJ!b-p+*a`CiVJ{&PXKtd0_7MmgA#ZtI zDbww-&Nppdkm%G3bGKi+oF40u3LrkQ6bBAVowrE@!A%wj3RN|AP=0JEJ`FCuscW`5?n&U?D7HSEmp7_;6qFx}3Oey22K(Emz*zy5Y^ zfBqttb~k|U{kGa9Sm5Pygm5gVA9e42d|fsvGDKmuNbj)SOQLzdMtKcul|gm?`gBbe z;L!pk%;s~}TCCYRy`2eBjav!vfG>wyH~MC4S+^}O$>DmkscE0!63 zWl_MKBFfoXYrdFkV+DFbel$dxxIgF3y?=ex&kYRL_sK32M3D6XHat8e$W7%H$()Pz1Y0#C7D#jBWTqT#yo5%r! z2IT|c_Ow%XX~+3u3g=6=YEV%ZC5@_N{qZrcjO8u=-f)!99x}&U!y$u`zriHj>W=A6 z;Q3VQXO zr%b8dK5mKT%Jy|OM6#jcb8o>|G%EQMO5EnVyZkHz9nJmI{+R`}+b*m9W9NGFEn&|( z-_xH3B*dwYDF3b;K?~Z@rdflWNO4|bR(j2P(Y7(Fs5(s)JZ2o^ixB0K{NdZEI)7ru zTI&_oGPSC7B&PX#(+OhRvcqCwKL1dt09W`cmd`ez(d%m6apvgf*;-2qiAeD02w*Cd zmpCyBVBja>$uZPy@ej$&2&ne+k7yMNrGnN`3@?^x22JFZmyQa@f?jnof=|58o-iF> zPP4kR81lf;aC5%jC8-{C`E5|FoXse`Ty}dO61vyF?Jn-s(X^_+)su;%JxMc@sliGBTG@1!Os=ga@4qa8IB zO^n6`wAj?_uq}|y5?E`8_Co^bI+-bmZ40JcwyfGF^KAm~=>C+P&N8ZzQx$|lQG$F?i9LihIayWsG zM~BXd@-T@G_2EZ^6qDXuh02kQ9JY=|+eiO3SA+{L(J;``Rdc^<gD#u-n-Rfk&Ei>(kB;3|TTWN%)*U1XyHKxw z*B(p-vLsOFypF2C3tHpDy!>rB#(u>oBCQf;;my<)l_WVqimkt#h^mNe1AyuV*A$q+Boq>b$o9M zdFy`W<)a9DAA`>?99*7BCvWxRhWqH}tIdg194m*(`OP=RUtMjm;4NkF%KK8E-H(RE z=j6P_Q)|B-2w1ZednbncX1~oEw3J_LIABb^-#o%CVuDTJ00zvMytF7+(H_h^Xdx^v z!g29-Js~HDC6GDVk_;+Mn9Ydc$gGENY1_G;(;;A1*()db$bYc_BC9V-|GPQc7KEt7 z#64cDqPsqvY0+wQ*q3&{`5{rY``ZOY8jfyTAxGeB;kfGeF?_L=q;1sTl6tj{R4o9q zQ+}>UrsNr{870wSe;nh(wOCU=kIiamEgX$vs{{@84D1H=Oj+R(=b7XjXM^S6Wamg++NlO;t!2Y;9p#5f<0>xunN3gE zaM%uIyg9z)EVI-ZGVR(zbckIKdkp1wWp=z8A3qZLT(u!sbe#v(*kl1IqX45iqEsma}lXxs<6pTJ;^Jo~{`Xz%+uFlQKO6=$MUVw81^9ru5$kv$$i?X*Poh>%ClZtx2$&o0vYZK36=FE&(+ z#H72#p`hzIPJ6DsaGTK{CH}7AHgK6d9zcb&MW@O#b^N6xyhre zfxhe25nVgaaB;rQc8K_=cmqMT0FmIEQzQ4yNirZn|FYiUzo|nIB{XT)Kz}N_dA?Fp z`TNt@v*4)xOs;T1q4`Yq>cYatIAbD9Fw?`i+E(hx+p^IdI0<#6VbJ83&wJHCP^#nI z+2!sC2TZ~PE?h#_twnFjaVqw5jAp0rKlRKosbnsPKDe>u}k2A2Y80@^L3o=NB>_yij`>pPPgQ_HitI7de-3ckLl zUY^vx8N6}{n&{F4?8#Eur&39>I6hfar=1N&r0gfcGbBh=xrw^rc91}Y!=>$JMsn;k zONE44R6=tSVjSQ$Kd=i~kX9bJB=;5CF|E;>%KpAR|1sT8kh68_7Pu+kSkS4 zzQpKittM?uBXHckQpug)-KzhG_8{d#`qBiWaw({@K%g z6rTSqpHyOM)U5WMl#whcvQpjzkQevhXk&PW6^f%$Z@R2sc;(rT*<^IYX!2+Ss_y^` z0|Guz1K7Q<1eXxCV}3=(M9C}nXt2epToEOcL~X=@CU^~-*5vf~b^r0(D9?o@42gN| zS0c4i1A&tc#9{lb?$GB?!4tB-o~=56*aoIP%D7i;vwd=>1agD*U2g8hZJnVsTWr0F zNC(Om2&SAG(eibvRXwWSQxv}&q2dCsLvlonEw;G#lLA~%zVBoFLRX@ae-dUsvi^A6 zb89A(a*vr8hvESD+HcPNKCJa4S}g5f(J9KfR}eFnCF0q10O$Jt%ZT^M`d z6(z!zrbZx;j2#f!w%L8cd$oHSbfB0CfMm=?>h$qc6?7=uI=Y(;Y~9RT150DE0(WU5 zf%-9TC!uLG^a`Y1&+li#R5v^Wb_{e5Jn+@rTg#;2Yo2Qi&>?Q&0ULmdW%-N<+1p<( z?TenCR!~Z5`IN*P0R5iEJvPhq4;x?s24ELI;)f6>A}@N_#cuAiSoD2r02+Gpf&)pk zeAey>zRr1b3w*(u^|Y6LNI}p@Fb@ZY5(A(+_qW9jXYrgA$Ld);riD%I%mKi>RQFa>Sc+s}!pz>>%B7^^ZTv(jE_mcMSsvIpU}ZO?jP)aDU!tSfA+O*zu&X%s7)slBd@$N zdqsIDX}QX<$4*qRp=2{V*nD6Sr^6Io>5Q>TfpQ2owl7R&iZB%OM3ckak1pxaAEJz1 z6Y7*zs56^ryIDVy;M5*3b+cmU#0^M9DJMjQRIzq-v@}qTGJji?l9R_~Otq$0Zf|9b z@ceWDT{jp-;wAOm9G4RjTJ>_~`g6+k8$4pFV(+n+E7~5>$Kx`pyl>YyNpz5CGcJzu zjRl0VH-YIZTl$d_Dd&()BjCf&{1~(oveh-39eTy>VajiP9pMG(vRr0?@Q#tKpdFi2 zBcFEWulSa++yXS*)y_@82m{y5peTAL^eHqr0@}bZ3etWE5gEXy)w3mv*l2ur!>VMH ze*h9q&Rb>qqw%W;T@A|q0+=I~z^DL@-J$il!l-trs%{cj$XJ!Z1p?q%M=`%r^6!cllWiT~|Y>*F2j##l_P1m~0;VSdU^gTp2JxZJG);Aw24$bhH znpAq8Jd!Bs?2Q(6mBX(x_gFKPyrR7_*op!qh>y@o8f&6bO0GX3Z*i|`YW z)1zRSYWN#VVqmbCBc*gVUt6WX8r5D^Lfx{vCfq!KVoM53sJtm1*KAJO!Ma3%N7<}T zT!h8lxga&i2?t9`0W4yK%I^yGRi}pPj%(5flLvKHS@!@OJZ`7l`7+f7{Bdo9=igp% zpLgUfdZ9cX%M~=w0!M2t?nD{1Ql*KRxqK*oWr-jKHcgnN$~+zBH;-Ex&af$#sA1r6 z-{@9+gi3*+sL&D=kAUkiP#w@pV?52Q{?M)1 zBya4<4R1wWUh}Ntc)}yl@JVZ8ucbf8YlyaOuL?+sa|0xSRQcWewO!8(+85FdTjY0{ z0?a*I2yLK}U>b*Oobzc70X*Ux>4e8Wp($Ue7cpWrfXKmQ+HGNn<;Ot3O3jA!0p3-k zd?K`KSi!)tM+riAQ40KxBC7YP<3~=LRf`zdz*E1zXE6Uj-P4D=HM=6Igit|f#ii*i z-Xfc|=Bo*&PZc?_65tj0hM>;4009l)#_`nMr;v;z4f;rk&;4jnz0;=MVygHSfwT>9(e6`w_?6Oghi%?x--E6g(>?Ri%~~|*e#RKUCC@R^o7iz z*B5-QLY+x>2IOdV!s~u}vht36k44@+YuRdcAWua|pp=UT4M`In7u{AHF;?5j{GKk# z^s6=vW>Y5pfpF~~9hg6Vz<-{q>3%AYBqZ2jjT~G;k+&efokQ-q@$?E!6bkF#KxWl&-@$rjd0&@N;`NFDE14l`8RDX`bXBZWxw6@}A+n5KYst8UBV}>xo0nJ~{G>gUj4WK#y(D9Iccy z0A%r_$IbUGiq0ri#j|BGj6`QjzGj8FKDNS?acz`vGjr%K*0nZ8UHo409oUZ=n$L@P zaS5X5ln;F$-C1$VMe~O@zifV@6dcQfN7R_&dFa|))&6i>kmap1jd<7*Fj(mA=~6qs zvgHF>1xs=A#e#xKWlIUywXX}=__p||d?XZnOS8(C6YV_v3GN*Dpw($ov_{GT zhxmqV_uR2r$brT5P?;N|WK=C=ZGCpCb>m20e2 zgwd^8#eIKPvm7B;c*KN~sbpJkQcQ}pt7Y4*R#DF=Ln44U!w|ZfhWo_St6}Hnh);IE zo_A^d4=#^mI9;U5h!js9Ps*LZERN6U^r};0U_v;s1Souw92WCRRQ8-Bo3MyKDZf`= zmu?gC9qx?Ze$b=6v;l=-J)s`{j_ZP=aDL0EKrIU)`nQibq(=N1A=?qDX^O0ezv_of z`1vpmQnxSW%Mw(lRvOHkBmzF+5&FD*%OdM?JZV+aRXh~6(LvT9RjJpdMXORuY&;U3 zq*g`RV?nmViE8I*>~;Y1pgVn;zg$q)cke{B7I@r4^Cs}0U2U@BPu~B4+KL9@hmJct z?d9$$GliR!l<^9=Xj0s9XB>1fYp8z=1V3o634%ThtaC@AG?9Z%BjCLs|IMKf|5 z+Kz|xFVdjPs?)uwmnFRo$5!IBp+_QqF(i$~=eI?=8VjEpC?BwOU9m#?XtN*KiU2HAdC0|fVFs?J(xO~qhq5q)} zp`JRih?#?SH^U$JoNgk6Vuj&!DoVxf=AllY`Wwh`zUgx~vrx{_Gj@aHnALJ(eAt0k z4hZOw@@^5z=Wg@7cA%R_;#;RxiFl~K=p4dvfh9!Hs6uW~_)Z4rK(GeysXSA?m7<+( zG_i0|gh~7E`}6L#ZtjWqYxouC$?6He?jQ}j^`mhJ-=mJz;n|bh%kP+dP{iTK^Fs%{ ze%-i-PquWRHL9-hm4HH=#Kz{k!KTht!c`XULRRS@H*uJGtpcq*8i(1`xEqw~ zUIwntQu7WJ^?W))Kz|_s?W~dPnvz=*&9hiKt*(6cYAD%=3= z7++X6<^84_OG?ARw~mWOT_-YQEKjD)k&i>=Eo!MOda41vxAqOF>y5^t@MA`s9bI3= z9zthEPISPh^dfcQ&d(c`*WXWn-WD1?vK}xO^5LEm!u~b@`2p!?ASwpxmREXgk|=~~ zG+Ttz8$&afl+;<_%8fg)q|XCYKU!>%6b<41M*i7p7^=T}}Z5>mOu_spPrZ;;-AyDVT`RP$fPILc9;g7yTer~^Nc`f>7i?AYeA8 z2Qg)uCDF?dle6=^zg+WlDuT9xd!SPQEypapu}u&LRk!>)TV7v%swhfnO>3K>1WOtv*zh@2hDp)+fy9+z+c>5Bj3>*)lxv=|e6=UPw2E#c4&&drWz zS7ofKI5UH0XDGWJZrki&w%y1|t@zBmx|AJJy?jW##DMFrK*H6&w6?d^s~MJNJ-yMl zXy>_!EYYy@t)dcIq?-8PfGS7ipMJ z;mQ_5q12T(uIW^HI~h!ZancwQJP{*qFLkJsA#`icXCbtucc2kBsl|TGL-`1O3Tcvs4va#6sB|f3tZ@}oqm^H z&-)0GPa^YVTq28~%X^?jcmD78bFXw@4a&0nO&X z?W8%2)x}G6nZ0AyS5TB?=WH6fe5~ZzHUtq~EGD>Rtx-*nn7gvt*a(lVZ$`hBIyJG= zr00!ts87EwCj}pLBdv?!gjGLb8v%TG5Fcqz zq`>bJ&XMFrE`_y4_nc*WbvIe7a#y~97Sw-E0Ve!=cJU$o&y>#x3kq}`UcK}zl!N}u zNR|S@f^Wb&Enpa<@gDkkG0mvgNe!Bmn;54uxMtu(_U!V&4XID{SwXIj*a~pMkHgUf zyp-GyqO$EId}vp6gC?n+T>avph;xdbhm`EEqf`g>&T3RMyqQQ^)$&nL)YIq~wI)F* z6BY_!VNvxaq>#XqWV_|k6mORe{|zN;MMP&=y-Q%L)Q?Z8xzV|zB#h8gqd*ZI_Z00V zJ3i&^{D-_;ub3>@M%gUbHh+6_QY7sp!pAUbP?YjmT@@A|yMOm{g6N+?OMk?r1|G1A zJwQI?^PV?Soh&IwUL=Kf8NEB+Cqiqo1gvo)af9iV>0KU_5`?APTF0Rjh?1Dm_j$F! z0CgP_-S)=04$r|sOxj!YXQHLkv8ppd>}jc$RuA@V9QlEZcU(}&qdZ2{366-}Yu+tM z^e(r-&7U_IOCoEeLhFq1c8JSCF39M{WJ;yAN~zgi`_(3_@0-G3hR5su)Hqk6z1PtM z7#e^7vhQ%2Bu2&FX!jg0&6hppfj(&Nb>W=OhQ86h^qJ~%fKPz)}>+nJG)vF;eXB55P-XF*?fBj%1L%J4{Phm_=?FBjT2HP%-U|2+fs z7*;09()5;Gva6)Bqq;2$PXG(IN(XmK4MEbko2iWV@oN5OP@rM#Bb%Y5tstWbr&MCf z#&dv&7Jf|DX9^-joc%hVX*iEcKuG9$=FKhouMcoVXHo^8Wo|oWw=K#Rz(n3bA;Oc0ES7kD2hg?7%LC+kMbNU^XN_h-?!_1=g;p=e;}c zg!7$y_m!*+_!0!CZ^5Knp33e`!rJ#$HY`?GfDHpbA$HBa9sx0hvBKm6Qh+n1(k6Ep z54lnB-*mP_J}{vcjxosiQgl0swM^`dJ|nc{~W=%k@4 zG%qL&mn&$MJZ%p*;wrh%Rj$PA@CA8%*``}MFmkJ9Sm0&P5vK@H-)H9aUX)0$x2zMN zqIX11r}k<@U<{K(b)r5Uhu(5cBTPK5|BJ;ce=}puP9Gle!S<>rWpi%q=nOF^x8C3Y zO2&i-6l5cofUK;7H(b_p$HslWd&Zg5-lug~Z4Aqq5IJV+#dc65#)3En4TgclTHOZq zWm;x*c1_J$Pg)~pRHozTSAtC%kilCryQ=3jWZ`ZDJk5XsyfY7RIB%8v%dg3?otO7j z>OEx5HX7DpXWA`i@z7CP?TMEo;6cznil$lrdGz1NK_2})kSm|n^<1Zd ze;@=og=@>}Cr|WQG#hzbhv@Vj&ly~V`D7{yA|B^_N`R(CZAfPL#PPcj9HPGWZBplp z!{ApY^3Bao_XdKliSJcfO(s9`#h%s!;m`)NQ`d@!ktMg0;6S6Ief9JZ?F-8;r5Id| z-`HnzEO{?pu8-}U9+RRgb&fITMUMOQl#K!pZh@&uTPq9W*DZQ>hC{-`a(R^Oy&{N& zh~lDP6Wz^Wh_d0%=MU|L@<~c;13uj-xQ%XpQmz>OgO*!me7HST?;GwseMC#8dQ#b? zx*NwKB^WQoLaG?hrG5-FRBFz~&RmlM-y(PAXvZD`!+utdX;ik_F5Q`0>3o8=>dnog zE}z1A+E++=xYMNodvSklr^#;t8QD~Zz#;3S>)pCL&dVjW%#BEBrQY}nIGXRU*^Dl^ zOiImSh$}6-$xN~s%J<>ru!NHRff$K&k**Jd_UxQH9Y-W_*tn;gc*_I0`VWJZuciUn z=p#@iEQy|H`(})JKc@9t3paxlWIuH^QCW<@B}feDQ7i67nRgd%R-rbBnY!^ubrQT2 zEM2|7)i`p$N9A%na^&rQ>;Ge)pq16ai zWta*2V5p8#k3K4eJUNY>LZCLr`X=bqN8b8Ck;>`I4&#On=m<*ZN_R>GabPEWLJ!Un zsk9I?y2%k zE9Ze@)M&HFY$q8_rrD8J!837++z#xC^_t6kYidL*2UV>6$Y zES62H6gb=HP`^5uvLfJhMIsWQ2e}eb)b+hp_2Cm|?YDcug*Oq6?Lltcte-#_AyBk= ze>^Ev5X@e`TBj{@$W0bPfCL3L51cLy=I$iFzjfnm?%?EWJ-W|tBUE@p^LWaz%xzKO zLle5`l>yQnY@fmWXwP%0Qx^dBd?55qoBF*={F{r8HU5@L1)4%OAK6$u#gCo%(;a^M z4J-iUvHLSa5+0x`yAkif^<1Dgow-F{=J&J9^n(s3A$<1J{*CWv$Aip|Pe3+ZKD%t4 zxr5sj9b2(1EGAvGN73^pHTQG~<>Hw|v>yl9+Rx?R9N)$lthDz0h`}O2I~z~Gq!r3S z%yDDDZF%{w?JZmA!=uYUxLxJTUtX}|qn{03#PY2yUeNeo=mME4a&}-6mmy*ZjDhXH zH=p_Uy}$j;l%Cc0jK=x->E*@u9V1DdA_*0SrR)s-df1yyZ2vhl)Z6adsz#J1RM`b` zVt4$+?4k7AlhdE^KL9Ko_?1#0V%jk1X~hu>YRj3qdN>1m3AlN8B#7O?qU8jDv~}@_ z*v7wF)M;~&L#V#n;(bVa7NzLpcbIpk;JA_{9@JOC!Dtf0{!U0ZnX-l-oGAgx!}%)# z++X$31+wAial^-5*o#U}f^34;%`fhDQ}%%q4hKj=Sr58VV_tzf4L$#ly|;|2YWw1b z0TB*;2&JUEmF`ABT98fw>5%RgQAAQeX+aTbq`M@gySr1mdDgkt>)(5?_rv?){ql@w z3}x(d&fY8LnsctXX8iUFME2M`%$HWJ# z{9`9<>hzgXpTaNY!uN@{z%s^0d=OF)fo=so zdm#_wec$gwc@kyt0TG(rt_VAWuk}3NJYV!_PdJAsVeP)s7yO|4wwer^<*?@(7{*O+ zN`CsnPQpcVf{-|ygtNjISoJE~kjcFK^zzYJ3unCc!;)V;SMIBHwj$(s@~RdecdyOD z@Os{gQA!vh481?X=!~Ze zEk~`{BE70wxch9?y8-fvynRcReC1xJMQWebWFN%PrxDJ1r3N7hb|w!JB@+NKX|li% zRDRrON|A!JXW1)LZ-9tE!I`?H$u)iy7OO(bA#{k=E-6CH$J#-i&QTn-?~{|*+)Pc2 zZJ0*SNj~kF9Y<6pKRC)a8}2E;8jqr9fP(S$QNZkNMwNmomu!>(mGrv86x0`&gN&In zAa5wA_KtwhL*a*MVvpH9PpD%`o{KmHyMI?3VXL!^0ts#RD=wUxDj#M&9;08u*16c9 zppXNnN_(p2fawRH9A2V!KbUeSJ`p!>yEY_!!(0C13^S_jR3^??9?zHSNTA{T)`MjW z^r=H^kyZm=0hX6-ens)SK24<$#wWw?+z>jy?_+@8?3Q5GR0X#ad%DTBlBDBxU&5~{ zgVv^|5UlZ_f5v?LZdf<@=P>mb4z8bHD7dPZ%F9}ern7Gy$jj;G`nMqHP;YGnY%eRXZe>H{A z8W3=K`h+Yc%Q@N7_8C%>yPak$Fqo3RI@aRGIBK)eFe<7ghs)d%_HE%|oyPAj;zi01 z7a?JxLO7TT0A6U7S-ziEvmq_iDh-~GXm6?0Od&Dd(ZJ{Q!)^4(>-6f!JT~KMw^?;R z;X)DZq}_8|mOnil zF>n@N2YC!yJqJ%#%4U3`i%h9#UyhgAu}K7@zi^lU9^;bHAT)w#x@0%X2ZYJw@CMkR z?85Kpu@s4V*lp$Lu%jX8qgulNYznC--xBRawu-`aF`Ng3!e1QPus*(dI+|Y>H!_5% z%XllCz-4w8o5XG+#{D?ScKGGDq^hHKd;^8MaXGo(OXWh+L61bYMunJ$KOHGPV(K;0 zED1ditI&D7{c23)$;I*B^OqWSJVE~J893O)fvZK=^%RbujMwTWKg;5=Vdjt~-hEmr zI5zvV^BgW)YQ~+1{oo}JdJZz5%jSzuPuod1CRI4a$cD3ZBtC0+53V;AlP!0wm(*#s zCc7l!CZi254mCbSt`qwr40$@YTvQfb7S9G3!YfP$`Wk$j{vKAaAK5Y_(b^SvLhT9bMNtF?|=} z1W7;e(-yNCH_4OW9s9J$E}wF51l83nPpV96Is%jN;#DEYxBc96*tFa`t_~7_2S0z0 z--Et{zRg9mL^e8l^6g!|WCF?2fbB8G%LK$-)`muFRH|FRX;k+HQL?VN``0dh+Ca z+Lms|_uAC|7Q*hzqm7JHrpU(pEVJ_#n%I#B`8V00^y037=eWd^(<>9l?|2h`PIlqI zd^H;^wqnc+&W$UII^K}9G!|C)R#I*)r^WxW)1di?k6AcVOI)zv2RxeBS`mF;;$h>OFAJ8& zOQr~UmMTu8ry+>nn7<%yq80?lXg-I<2$1@>(L8V#bcSWH4N_sVcO=|WK#qC#n{e4K zQ$it+;#-ywI*L1kpHw7(MALzip9=Dnz^S`p`X!c+N{U!0@ooq*&T7Fs(n-S735uR{ zSw4EYJ1chheEr7o_t}QP5;J5)wH!^AST=*aX9OXP-e(7Wjq+y)krepKn8|2Ro?fyb zvH;~G-)oRKgi4X|%Ks}!dCI|>gGb0mRtokJzC0KhUUFRN&H^<$ImUJ_$)4MMVkdM2 zy!ec2ZIu}9RcN=FKj8*QM2ICLtw_Fl6fZzMdjl3M-R$~Gwr~?|c>s60?5gSmuH&{E z(-&io3J4Z+PDJq5jIT_de$^9SNzKhB0T!k0^O86sb#4CW^D~@)*U$54A@b%of}XDq zTSc&uVRJh-q{BtOZTsL4C5ATN{l1)QG^yY)&cvuBt&8rQilJ%xa)u**mM=|wXZBn) z6&>XPHv59e(6RS9*4g1DhFVIsH)bQooEd(Kk%d}$b9AQS#?aAj@{3%tYT2B|);Yc1 z1Jaug)f5lJn59B^LNuUAP#0QgaIfK*nlKc?*JgAhx$p9o=mke~(=Osft#R4JliJ*< zD@sHR^Z-q>TbQ)a+hpQ_YxvV$z`+*rC530x%wg@*% z4nAw&IbjDuIUYtjbadQhnnzcVi*MHIMx9Xp4xJ`5#3BiWS_vtUPIIOVDZ@5%shl>XaFbe*! z1oj)^JlN_ah`FtM+vCGYg)9%TaOXfBVqf+=DE0I?ls@GGf%$OETZsPfX^|1WsZMMC z!$PACizL+u5t;U-0W>ha1f5RV)2xB{^(lkZV)hVQw;N~6isutgz^p4&G) zaa;%Dq#v~vnW?R^8)qt}yg5d$qE+M`@u~?^N^2PLM)`OKOss&cBIP~;52=N^ z6Hq)ijSocx@j;PiU=w|?y;$3j-=mWjau2VwwM+y%1h)rE!Q<19^2{dXb-|S8Q3lz;a1UeEjs3A-uZYuAWU<&(qB{HQ2?QOyW$>o+`^_L7J zuue&h-V4eVr>_HrUR7R4X|wo!iyrn_ZOF)YL|$-x3b*ixCPh&OR6&`()9jed4^vB~ z9&ua7yx^F4E{NN9^3$YY&nFSTZc8yZBmZ5j_P`p9!obnbGQ1b(J9XK-_4vid2oAX9 zX%6zio^xXUyW*PPvF^bKKRSDkMFwh>sWhru)HHPSngkHqbFlRxlrQLRkQFor35(-W zphtgyyz3NRw$kUiFgLd))b~b053+?=NDB%MSQ}-Xj=vM3oCJWHLjz^>DgKBZO5fT^ zRuW0SlpWg3*?aJx@i5DF{A?+1f$aQJ`*c=L=o05#oAq- zCvAl%HUue(u&qXz+Am|ED}lTJlpTkra^k~sU5x8@i(9NHvi)qj)r$^YGk~lY61*1u z)J4X|CNd+e9TIlquiUTP39K+IL0Nf#hubLcU-m3K52(#9_it1MEjPem4eVb@6qHgL zw;b?0L89cz_&fwtUjzicL74!B|Il&72wefsBj$0YquI3d?`IX#X%xjXTt}=|fbm1F z?-`=~L=|`}4I0~YE93-}frC}elUob_g(~nE))BK|Aw5bntiGOCbb;lkJMa4g@LTo# zN)yQFdk;gr;ODQt|6_~&4r}wGa3#7a5InX^V0IMsqxxPPR*tVsq;T{MQT!FSgc2*e zJpbc2auZmc|JyVZ=>S++KRF2jcj%9YWa_Cvy*Fv!F2MH?^@2eR%uv;@{D{K%c>+>R zI?id~?A`)*)l#;5{G1zt6c|UU%sw9&bs;Sn<8YhXjUS7N7zBFpGhCKvgux2jCG~Pi z`q3Q@J*>LiX?D~a2Z9V5Z#0S-lm25J{DT0@f`s@Tu2Mb(!$E$W3KTX42S*8GJgVowR&rgLp;bs1A3Q$<_#L-voGYtQ^eBPSV5RufKX z%s!g&L_+MCB15({Kt7xS~k~ z$A|1yjv$hA%kLOWZtO7=_`nbVX5zoJ{<+R5u=NYNr)X{32tx&#drhMA2XBL4=N{C9 z^VM7wlCLklE??n{4`pw~_Z#Txi*%)F$}ImMz!>z=Fcj^AJ(l`q66iSO-r>EdcE{|D z@YAh6%STUff3TcWW!{&n5%?@Xz!q?b{f%dtKh~EV0^^tfF+*H&7#2wcslLnWA4@16 z1p;#tg|Sw8I3jEWYJQXwGPO8YTX@aE<3cW4SYiRIxyGG3t&klg*KDk^xq~6GLu;^DE1fZ^8ue zT*D9ULLQ)bq`-gi_^Ag{U~1hlBj+wH=cfqa*d&QJ0C_r)X>Qr^2e zZk6A@*dFbyI-GC7=Dxq|Jiwfcs*V7nSrI)V*SfR{@9xWp(1pTJKi z>tAp=KlBh#=m4C=FYva30_j~tKXgKN$%h#-6d=j|g(C+&XEUMR=#EHs~%$D&_yxhsJVc&1{&uHRZg2r?*Nv2NOC^g>-Ex-owSeqsB_FaN+` z03^U{>Q{|u2F_;u(_L>?p!P^Sj}rf);Gb9~KmU@?K_QH3bASjA`^yS7kA*GzGd#$c zR?4i$D8XeUrPNEA*LV09^QyLA#@>xm*cmO9P$pBK0WJw0)LV>rOUai zhyFP~kRDA9D^-vt^HwV|>mmJc0g9Vg0QN|q^NeGU(8F?jqxNXJaB9Uw-UnI-H+r6@ zVXXGRa0%Mvo?e_hWdNaC!IRlQR`1i56kijN##wn~Jys+OtVwFskMb((r70m+3B4lr z)jpS%)>D;_Jn02@>u3F`|3DIU5vb4g#yGB*7dor&yk6~_rJ=u# zi;Cfe@^4=F@eROOLVVK;M*$zb3kv>+a8E#(JY2j0lnkogeSS!Q6;Z zG;;5CnKbeey>=UrAHJ7F+0gDzIwFL+h4s}Ixsm8Bjy3_hWjvx6P-8;XgzNI6|?Fz z9*Aht`!0VYbZj>b<*q%M58VSEtjfePGEpeKPHB#F$Ba#5puXGOO+t`X=(9YSr8@PJ zq@36D$a)L@J(|ZU*)t?Iu|fhNq7%$t8T)V7q6p+*z=3l|ebQvI3oT(cjb8d(9%`J< zTwj?fY&T87=_B6_V)*>-W>VPWSln2J8KAm=1k@RGCw8HD;C~TVgr1*ztxbU;~VBV5HAZ zaD+gR(5sL3;MS`-8iY9$fL4Og&^(%_SEVb?b5A;uqRA!|(wqMGT(qCs_QyqtfPd;O zA|xzQW&oAqvNi_N#tzqg6D4c?fM0K8gP6xQ24Kx>ToQ3vLeys`Fx=52a2p^9Z8H%x za<@S$Tw^n_UA0l2LU*!Ib`w#9BfNNnUEL)u2>!S*+@QEofoBNmDK>0CsM*0I0tulR z&rrQpE|Gg)nkBop+*#*N-)bRGCND!w#_+#!+~hU(@ePy40}%V{qB<-EHIA{LzF!{C zbS{L8*y?5W8m7%E`oVS{sJSq}1pBFT{(iCTC)$gO0pT*i3JVe*r`ypfHV5_c=zV%+ zD?Ij!tHj*#YSNU;fWqy@;X)sAMJI%aV0kLX)1LVZd1VkS#BkL3OkI@nBcROJbkRn6 z4o4?Yu}#K{Ey}IOmOA5Zw#OMk??oJpTgW`XV8aN~V4o(cwTzzn(sVTm>VMm^^J_mV zg%NVxPxd*p*kFH#gm%*vK@wP3exm#Lekk{U=z8$c9vLwJgaKtz08o$A+HD8ic&MyXdgu)DK-F3-*mz}btUN9Z{5 zQII&y5`m`#Td8R3DZxJ(<)@IB-#`#!;3PMz2UTC`PH67fuwv+*iD$uJ0G4()sXc`q z7G+}21(WXLUhduzeQ2ENygn#f@L7bL8ypC-K)G6Iub&rcl7Q>$hhgr*lIV+x_C!(2 zR6Bus=DUv)JPy}sNCaG*gdeUn-kpo0=ag({!i9$!mG$+f|2s7Nb9f;){8M3xoQL13 zImtP(_c)s$nfu*^^RKsnRiojsQ~8R2R?AA1dVhy%Q+ce&R4*vF*89AYT8o?KtCab! zYeIxmf_NYyh3!HDFkntd;e}KFlfxd<0)*=uw8xP;aKS-6u2G`P zuc~R1&_4o7Bz&5yPD%2SSv0ej<(SkJczI)a^$wE;;ynucM8h{B_rwF=T zKZhnWxNc8+%v?NAgn|upa5=BN}ESx2lT>}JxSiDXZ^I3s^i|}9HG$@gq1O)@n zheiIg8qZ^ojAS+gVXac@x3|%i+lbTVfF0KV`4Q7wF00s9kEihM@97j1>1Cs+`<5;k zI8VMPfJ_Mp_E4d*G(KjE&si3y)$q3^U3qY%{X;go`bhpjTP6+U?)O#&HU36wZ@ zK`=0gI3x1wuEzOyY|TKjD1fLUOR^sKA(@M+blu4lVgxqJ6TJXBDPTJs-WxwyPO#G+ z`KAcC5!8)yJS?--V|f_A|77Bm3e^@@B!DIxE4GN{v$47C`?v~1v8e&QhGvJL*iEj^ z=Q4-e{fk6e1Mvgq8^GQ?`NNy>ixv7mi4fhz1$H(RLJfE@g&gSBP|(*#H7ZE@I&4n| zp1ll5!O8Nc5)O-P6hO+On_u9bXKa_N~N-E@cdYU0fu7O`tca7wnV7u&zi!2Zg%88U>^)2+cvsX1CWJ67lrGB z&w@4Q9vl5rFaBhV5PU${`J0|@A~F9zPGh#DS2<@%`*dZQB|8!g=~e_M1CAE#G-?Nd z9eW423p*J-fY-oeV{{MZ=KKe)=ihg7*rdsMQQTNyNBU4}0sEnKsTv21jIQXn4BVQM z<+!-;08P#;If`$CRa+`_QITTc9KY;!g#>^t1Kxkma;3ilBY%nJPqoq@3(ojP->Qw- zPtGM8#?VR`8*o-9Pv9*c@puNvyo)Z8KVmj*a&v5O(jN}l2I)X#=I`K+wJZy9eN1Wm6T{j(_2?CTbxc04q!;=TG z5mgPP=P-ydksD87j?CZx12GQJ&bNK&eT(Tz$D;E#t~kWLWqtsu_8JaHWTnSFWsoH< zc`j-ae!hYp90VoB8gYN$66gWy1as0O>amTI0|#IPKxw6*e*Pxu#{v6abAa3fDWt3e<;icgoqTW_meIEaRG6MwdNBt23|i=4m-V0 z78-Z{mlQO@L>;g=MFfNpxCyJ8vfTP?!xATj+~?iuxnx1nBs&CrP2Sf>yanI zyc8>aW;3$nZLrHd_{V>t!Ty^?MS(-Aegwv==CFIm(~O$_DXu>sst4v(gLnf8%*zV% zVvODo+M@Uud4CYa{}XK4OdC1CvWdaQ9D$CQ@E73gzn8KPW=&w+(Fcnb0%PC#IFUd7 zwg0{qd@~L1p2eVv1-<`J4FY68k#YXTbA>RoP`z9;9?UWVj0!P|JLM-1>%Z=P3Y(hk zq#HiW^9nQA;Y@@B{)5(jh}#ih83hpACcrvh0RL*>^=Q$P|5!!HB|Gt|H+!^OeoaAsYLcp<;fhB?R zzzE(F=K2#E_LXUY@?^jV6b+8gfP?S zpH2SUAw@3$XfCww;?xQ7l{su@K*D7E?~z2n+NQqI@B}m{Ndx^X-gM)5CBj;ZLl#zl z0fVIMPO2l94o`{z*qZI&T=+h~-Vy0|sol8_paY6+SMBJ_0vtbBVyAAcmz2x)G@b9| zVOc^Ir6P^OJ4yBor4@(ElNYM@gawR)4`6t(!`>SC`q6Uh?kx{9A22XHoTLK=StRxG zB$Zi@UPJp@dyoGq*QZ)KM%Ca0!C+L>#jHirnatd(1*X!uKKhjmWa0%m44-TYB>O0;YJ z9}5*z_&gPDCr_lw%)Uh^K2>_Gx9_sDm^HAsQwDEoUkxW2FI%A%jxw|wk9)?=01 zZt3YiFNO9Xvhn6-=Hkd6x`vfba(##r2_FBfU@yv!3Z2!zX-v6Zw1_ z0M4--hTZ7w5$J*Jl3i&Smh98GFJL@{?QGUw^hkU1g~v0q-?KjmECLwhSUM{Jxw=c? z4@z4)gw@3N7=alF0yNdk>e0>H)x@7QvgqH&;0N*`*#$_+sds#pi_?24Kv(}1I~nST z&+N=#CPJ@Q^a)RS`ld^0Xs_Y+wT?UCH?0Edm(VCdE%q$c?3dIEv4ePFE@=5L-O=J0 zjOvF}8yPrf&t7Svy(!0@qcP0q1T_H36#ylN&-pCp^1DZhxC9<}{nqdH%?OzPKt?q1 z>&Gb3a%{-0w7ILqu@9gcQRTUk``t<9?6a-85c#xFe74Sul-RctHO`9nb?_ev@eD~; z3Mjn*?vOSt4*c?v+uI=1?6B$E>E7`SPmbl5K#!o7bkwH^IhXzW=6);wE&&DZ;_>{z znZ_n`YWKR(S@A5v3W^GCflOm63b?PoV%8~aSt?dFua`3O?84|an|>jEL!94cwD^$* z$Qo}ZgOJi@#8?D6T6@p+6}YW2ia}`q0Eqv$4mbe^RwQ5CCzn8}Kc$5-iZ?kbjYLbn zz}32Q_w=>06f!1B9|9b0PqvV{7yxUtd(-e*XfhmgE?xX$R07|q$YaVbVG-1|Rgx~! zxM{j8@@|afi&N&R_HZoQBHS6-j9&c6T)H;!;f*#I2nT1ZR#kDpmC&SFvu&Wc61XQ)td7$ z(~(coDcz0$@X-kpHDO`Ba7wQTzu;F++KCi+rn${~ZBM$N%DTTBW#LbQM+hBAwp7-M zxY0%x5E57 zfXKT?Cx87k`BC?#T9%rA(dWe>#eH@2f|(UOxU11pxl-GStkBdc8Ec!(k9$Y;8MTgw z*8n3^?vl>3+>M8Pa`63!$mrwlm3aO)l<9&}u~k9QFT;BMn)^APuoQd_wQa>UoBU$2 z&plRQmGA5c;2$2fmF{x}Atj8mdv3p{7Z1$kbp;COHxsVUhe_crHYO0~0T2Oc*di8|8TN)rbv4myFIbhBzJImdZSm7U`j#%;v_<@y!7ehz&NH+b zh;rP`@r~{))yJ*vs+qZRirvG{CIoh0pQyN7i|zsgDx=Bxch-b|5X|8xPQ+6B2C?y# zC_My2^Mjqzk@eo~<|p*HG?hB;w+3A+O!tb%_CPqJBJm3uZ`0kaF;CH9yUF*Vyma*x z;xUXWk_Lfh^^0p?a)+*z2J>jM)V7gF=crn#JxiH!G}0)5qoY67R-r=YuyqvR7G&MQ zZCz@puE==R!DfVB0iAV*CJ9|zd*dsGN-@R^YQ4FJTgC%P{8x_{AGwW6wy?U3CKAm%^(diTvkuAL%!dsM_EBUNj#OyQ0!tq01;rOUxyhM$UO^67T7E)*SB*M<`a>LAoG!-4WvyUxlt{2xFiwz$Z z%fK@LXkO>jB2?mMz2bdi9G`7utx+O10Ja%jON)?Vno(?rQ3TKNC8M<}eD(wTL@u*P zP-UtD$~laZj^ld$bn)nvk|W;@kvd}wl9$4)8YpJRC=w0omil}Io$+fVB@?%wLBd}J zk6ajLUSX|!(SkQ}X#ByciosO*2#s`KTO#Pv?upZlS&zS}*b7HpSZGh2^K;pnrX%*f zmcVzV2vBF?Ys&0Ca_PzlL)yqE^H|(p`JIFI3nD6j}V1BqH4nx zJPGeimjC3LH+%G=4Eh#Yno;`RyU`M}I`>4AZxQ;Ih9Bw*2^=_iYXt2l@Rmr7_GH{} zh$1G^j4oUC8hPGIzq4^7#c0dEOBEcU_>q*(tO+`jBpy-GYQ42!+C5fwU|ujVEBrf8 z&EIk-+^C_b;!Wv{Y@*-|0y)A&J#ZUAHgpN7&IhCEOF10)topGxLt{4(}rL*Oei1BKvJd^$W+(*dvn?U^|oxUk%ySXsul7RiP@rWAH>nyTPzc z%#G)-%hf>O^u-|}_Z}sIBdz)oH76cJi~*>*G_n0sy3(x|krI266K+h!+06aDgd+zV zN3|!pP7}JtV;}J(}tZYg~O;V!zTe2ds)9M)pHPVo;ycQz2sJlBHRT z?vhUMESO%#?YHMX@(|czXuiQ{K!$u0B`BcR@1ECZF&luFp4DkjD(EKZ^0^wNjUhr< zPdjV3O@fso7TRIH{>uIF9=zer3)MrbszvI0$&1<>&z?ZnS~yL=C}4SQvUWy z%`9=P_DHI1zT#3mB4#|`l z1yNV2{I?7h!19kashW=>MG0_$X$&tB4?xEs=E=9yM~`m@AMZvQ7kO-XxW+Z>L1@Zu z4z*`9(ls@HWqQ9TVXx>10zjK%MKY#3$srQYUw6E7CeD#Q(J?RASz`XAa;QS%l2MXO zIMp3V)oMC4KJy;$`+f6zg|>Y^FDw_dj9$N$kDwNjW;$EQ13eH1~*9opKJ0YKhwBE1QOv^P*R92oQ3=zx^>OgAv|B}$gp3a}3RV-~_|1!6pQcW+ ze(lHzwAh%y@Mto~6!JW7FSQZyYB3IZQ~3}J5h@A?|F=K&*$9Xj`ZjzWVWM{+&6d4T zWCiJ4*#Eo>JTx!kmrC)PDB&~ZEEjwLQho`+lGH!if_)5;h7SmKvvyv84%4kAFNwo_7;MG+%j|C5X_+BH9a`+LBoN*VtfpDucdiks;*J5-?7c2~Irp-?Rn? zFYTAwkYUT7*Tti>D=4L}q#SDq!e1QU5*YC%tc02V#2KwzuOl8#6%{BuMp8^7qvm!=COTA9vWt9%U^Cs{)W_w?H z>-_s^HJo21U|tDO;%YpRh-#M5UlWz+Wumqw>TTv(LQm9}ut`IG{6V0#J-Oz*(f4t8 z$lvI&SO8WClLsyi5Nhz|V;XaM^_-B!o;bF{mqj;Wosioif};qftgPq!$67VKzx68t z`V}63t4%p8>#qqRa*J+=HV{G$109%T!`X40FtO1||3W4IJ5a!-@J_5R;1JeucRhMj zjoD-9^}l5njf)&#-9TOpS<4sD64-en4tpp>%O8=0KuMT`RVhtX5&P0Ez;nEU>BT+b zN%C)0zo<;VOh8S+DK9)y|Iyo^(b;S6z+m*MC>Pwj%||1#;}qatcmyI|JT~<~I_ufo zU^rU!A0qw-CH$Vlo}YBXBk|Y72hS@PnLKTe7APB#m#sQ(n`@FzG=1{N63YKN-m z!$#|`3~2D$p^nrS<2Rej}iV>fy&R zf6RtF28@B8q!jktUup_C0>B-U9x){T1o&W<-;q9gz(|=0rq-+#sH*o1`TlE{g1}VT zgXcDXwFzr|))ee5Y-OaT(&R0mhsV|aS)?$DF#}Wkf8_nYl)TfgPz485^K>=W1UCwd zsAghVbjr*Q@HWHY5I}Isd{^06MeP@U`rXF;1{c9t8a-k_F0P0*Y4q6i^fFwEPvupJ z?tTtOXZshu3H}j&kArx@p|s3I-lvUPW@|mWf}!p(E%6yJFX} zQ^oez-elx}+^|rTQu&F#2Hd?!AcX%qLR9c3Mv0aBn`t5bvo=h!!;fB9ORmBjPDXnZ zR38A?V)q%wwRKeNR;2*YNVAV%##!l|rTf1s-(nzR{Wy;W5Fm^3K)_!2InF^)Uft=} z=oU=RaW1i~L27&dwUg3msm*{^v$w$VP}|~Ko-CzNSM$`1a95Wn#TgMp_ztt1CQuwr zVbXoSNU|t-253=+l$+8|R1voEW(bemC$C0m2nb$W#C=#MhQ{Y}QSMks+EX0c$(psr zyn!wjh=q(s3e%SM3o1^(&L0A^jiR210dp!ENFiTVPYE_8-%r4t*XyKUu_tAwvi?({ zU1w{G_mZx4N07=e_pvY{1|}AV1tf;Ara19ofgg3k^{G63CR%N=!~m?1DFj_Ic*KWbIb`0@4q@0Nd&w7Xyp?#4*@-366Meo@BZ2xZ z!S?)UE1F@)mzvB+P+EU=N~$~C{`F#alAO!VG?TZ%9aaFLg6h#76ohDTWPYRKllmjR zToaTHqyk~@T1SnjZw9+@uLv$RS`_w%gR+XemWYmx;P*X}6}NQa zJrbERleI_reL9D2*jKVE_R$hMyJ5DG|e1FJjHg4?Gw3_qR-lh4an>Q1 zw8dZG*vi_|9!bt)GbPs4om7>xV+vlgXqszL$E|c+)IB%75;wCfV8t2p;AxO{ zG4}Ai+r9moZks92=}McbZ&!3&vFveJxV$ruEY{&Xb(MjKf_^zn}kU!XyQXPVcL))XR?_Po#UK# z6_l>m1l}>zvBRZy*+1?w(KRCb0_W?%IE#JNDUbE|xA3L5$W1I_2ob`m*56FiLJGjw z#9J~f&=6U$@x9{Nu%a@E>HnZ8w^VO3R$^6#$HQYh6+Nw7Q`d(pJTkq0YxUY}zi}P0 zn%8$)ZVe}4PFs&CGA??Ra98SM&qXs|w@~^5J-SC)rnBQ@&2(}~onho)o>rIkWU+f_ zvH8@{^x{6>;#d9U;1fRKjm>AxwQQX`zRX4VH-tT%t?2Ma@{Dq?=Qk=GfF={*`FiQ< zV>3y8YIwAAm7Tk!&O+2_!(%|^G;wu_ytrDJ4lJby39dVr?=LFMEqcA3O|MrfAH3K_-;AfQ~RE7ItN#xq< zNe|J8@+#*aGtCy-ebW&*zdYT~r->SMEkGfAeQJZv6hpB8z?Z2Wb#zrbZDzkE+|@q! z)Kxt&EGKb6dw*)%q;|nVfhbXM>d;+wyoPQyZ_a$g{$lydnJ0#Jg;QSjah^_>U9lnm z@zK;)x>6l`gO$C`jL?@-$=(Pqk=K9IqN_khgiM#FnLwIrfX-{FL)cCgOR+cDKStu~ z5nRi)gBu+vG;TGSI`j3ib;G*DuU(sT{-q|=UA(s7m9I?!NFkrl%2e9f=;|JL5ED^k zDwlA`uriq7h{a>IvT4?XM;~D@O)N6MNR0mYCLxnXZlvCFmZ$e#wNqA$01SHHA?|f!Yx*YJo_qSm!Mj(=sz}VoZ{8W*DBK z;YT_42&?WbE4T;fu|AD|E(D1aITmv$|)*l6)*icdQ-Inn}xW5Y!#<0_`j z>Gi8WEksAAHq_4E!$q~8EdKggVvId=BD-sX;`xVZ2_^o{Bl#&$+c;^m&vtCuN>>}M zViW7H98ABR`PR`@s=6RyytqW_EbwQzA+|wG&LMIk|51s^Sg5PcY3#x2Mdwap1fwRf z*Qa`Ubh$6Sj8$ssb`MI`86O>NREsQDTs3VCaF4u=UoGb9~3$2xHPKPkINd;l}>dH z@EiN`=1#iK%VeMU8ITZY{bPN zu+EB~92F3uGI8LtGTzI@0U2X~R=za3x9*cVm+a=%&cNsqhRxQxRFzc!+Vd12+%4MH zoW^bAlw}%F9tGnGKaOD`etRD zu+DAb4aJ%=p62sXgd-UyRUVg2j}|H#UQKGP(7IR4QhBP3J1fQvSq<}b@-JXU&?+QT zM1S;}{i1gR`O-GYahQ6Mp&Kt2&eK~D z-qn7axg*q88FC~t^I-*TxcbOhWbvz{mHxGpcWK>_Qq7n7#aHLDn}M{VZnKfTGlIQ( z<7KwA*A?50Ry(Gh8t;f3E)_}NUWJb+XefVBbh&wWlwrYcdbW8$me?)i{0*7JZ`9}A zwJ#R;@!I|Mt&w0EP?VALWvv{(2a90a@yX#3-S-kSk|auO&eef)45Bhjh;1;|I@L;i z8K=`b?gc9MjijvMYT#}U&i`6aE{BNt>08y)V* zge%9e3jHF0gWf`9CtkJJ&u#BowM51#crbgNapmo*wKQ19L&UQKHB2L}a`t$TSH4MU zC9)-+Rp<8j9Rb2^n${xqW z^LKE-@{HICrDHmBr+cN@PRs6{itp^TEH`w;jc zroalVeRObfLdG~Av|H~M9ae{f!;_Ge6jMVV6kMn&9FAT;QqUD5r}2|b+tCc+5IpQ{Pd+zvg#33>FctWC*r!~AD_KQ@b z)0abgPG$9F0^_CZJll4?#K(`r+|%s1Mpd5=cP#~Pvrmec)O4O!zUGsHJ zAw}Py-2BDGL8sU0+Q_LGGKq_eOBWuin#?;M3-Xe#fo}ef;=oDvZ8(N*Os`7o3rXcc zMjc}$oqcT|Z({1v1LeTVhjrUY!yjc3F=o7u)6t2zxl`%4=`QgvDl|Qg z2v;}vAI{hUaIB8#X~%CT$W8Za)~p?SMrUs6#i{qs3bW>(8j1yya?FLLJ&})4O=>bA z;1&0|aMy23og={wPP7z{d?nv7Zr41!lKYfK;%fyM|8wlXBUUqu_RcO1r*A@<%VAar@ySdoh1PbSnQJta`iLmN zjN-AwyAySWRpz8E^abo}Xtr zrb)G1J1QZkSB}!m*|zsV)?!&p^`?{F;`GOTe%@^?k!xD2`{7ut6VuZgDM)tXC-9l! zo}S*8DN3&1im@r1`Ik|lmTh13RFNtbZTw7ydVulAmaENKsz?Y#V*S+^`&E(!{3Ex5 z>8bvLtD0et>%4%;sqyOl?eo3qVlDXA4OGub#H4V^-qaf=Gd`Ud9(;Yob#d4HWdB$YLKJzLAQav>9jbsaCQzS?7+@hkRy}32GZphXdBE^)_ zZ8fgt({wJS_;Po;40!i^UtqYg!=2+BK5%o8L+HFc>H~vbmadP5&DwNDiL=|M)U@tg zp0?;NvcjWKncm%66ryjl?N_RwQs=@nkUK1B!!I;am+H5xex%23eS8 zH`$#`I&`?f9xp&X5LkO+XsJY9wb>+-UV z>TAQbtBEfTn65m^9@9z7RTptJ@rjFXYc}uZUdIpiHV(RaP`kGS{rX4E=Dw1TBg9ymF=Tka#f* zvj>TEvFxMu+y3Fya*cQJK5V?#x+$6CX5byD)mOt+!qNwFo6JL>)k}vh$S%8*H}5!I zz`44L0fXI+yiS!zaSV!!j5u55douI)V%-UwhZS-i6i zKX#~_ZjeI$MdXxywm6)Vb0I07?!b8uiwKZw>@cBcb%!f;UQ06)K*n?2|K>|Wp|Q)_ ze>3NOqz3I>Ltb2t|9Y$3g`Ud7~kz@UAv9(5;!5=1()WvWAy+F)6L-PCMVg74YQ>PD} z5i37Vcn+`2%$%x&EpS^$L+3GxoIaedozf~f#cTl=_xU)-*>K=xvwJZTM$&jmgjIg+ zB%_1nmUgPB&)bV5nJ-EmU1P-oNZ!U74j*Ekk{vXIf~tZ_BGjW8H%ntq&6}uubZWhK zhtYSvDhXG$GJ(^eGuG#bDcL|DNpPmiBmI6p$3{9I`%%{m{)BCYT*q=`cNbTKWncef zz45jW_#;K?=g1h~#3pmGJh1P1nD43{mtTGHn8WgWn9%u_Xi9sq96)_gs_nkY3UMBI;0z=&c8#~u3;3E7ZOy~l z;#o>Ja<+4IZ69+xf6J-rtwstTT`u?VzUs3DFM!hbYb55#hO_Y@t)m(C@0pHOO1;XQ zF<^DA^R&Kv*IRunL(4Uu9(a67^FuzIBDa{cggwJ`jq{miA}V&LxxRyghbxixZNca+ z{8|*(DI|}#Avtr-s*PhGIt?v_8k`-L8=N*`rn^rIy!BUlBHpCx;?PGIHs6H|wtSQN zSrNbU0quLh1`!e_)&yBXrrANDhDW(OuIJH3uW3Qn;U4FB=}DXGT)k~K@sQ80JKgml zOH+W7F1(spGe+O}895qzg8MuTz?4Xv3n)M0VXv+mBK$q~SR@^M-v@r-6@n zY8gVCg}qkUu_dJ4;b)ohN!`-hf%jnVPrppcGcyW%-r+j0^q5)hU&y4%5bt}mH6)IA z==}wmnbK0zT6Zcbjn$|>zQP#bp!RnnVwo%tB%tHm$wK}K)bPwzvfFVY^NDx8K)48D z$T~9*8fWFFokhayHzyQYF@=G%@L0RdD#+hh5g@9mNPN+PX7@_iMlCuxYGN&YsB33= zjF6R~-iz$XBXfDs+4;D}m0a-xzQ*mtbv1TyD=pgjZfw1@o$Esuz0Y*3|41ezfQnh{|8heML>R$RsAavjR5>MbtiJh=fpXZs01 z#lajkEtt&eVIsS?Z_swittHpfOZv3bwy1)srG?3nf@evAerc(H%qFHbEnJ z_Ek(zI!bkxA!l3uyLte7FuGqgXI0<#{DEVDTYt*T3rQEl2#2&6%XjJ;P>b~@s3)o| z%SO!lV+|ulniHrhKyGA>XS3U?*OhbVOTeZ~c0bil z6u!Px_UT>b%h4%$xj}OP#rMVH%uIpjfEe$TXcyUjfBG~g6$FAyU?>ll!N5$?&yEYavl1uDf!xXjOl;Hb$c%=2(6pB1WgK=-v!MFOW-j%q0@`kG&3XU>$73NRp;BkQN^}@W{ zcTj5?fpP`SIV-spabu$n@q=TQ89OsoTo$bP>J>_^iB-fBhJ8QY0^Koh#rTJ6p#s&i zD96kcB&GvP%rv_-j{-UB1o>P_`^!^u=GCd~SH*H(C=ieBUCqzZRy+e)P*YG6te=dl zvQWQcx~G;p-@GVodw0!R!-7fby-W5XR^y!ZXxi-bOUow{HS@bfENfDnI87kEL-3At zmOFX0(Q)KnLCdg*_dv52a`jtNqU1@#;92?(eoVwQtRTSYA*cvajz(Er?;Qznmvoo_ zIWBm$D7%6-&hw5-yupX{+MKkzScA-9P=lOT{za>u{TNiKAFmti`oA~Mx^Yxk!>cF5 zMNXfuceuMC-xO=Vzc5L>#z#E~0#-Wz3Z7T1*nGItr^=e!?v>?V)V$J@hiB0L-X)LZ zGr5O%LfC5U&nyaoM|Kmm-DeoQ#PDo9-e=gUfnDjeMp2G6f>XW+ymoynxZGxR>K4If z;Q2~HlI7?W^J;w6gYwNNofbLs7gL+teYz{_yUQ!I?!LP{@610+zqRsEM z+4(x-<(~bgm4GK=)qRPO^^ahN*!H|JL`jRp%TGelyOC6OO%gKRi7Z2>CVB~gI z2slp5o`Gr|?zgBcw8uA(%0TXVuu9p`B(gzv@Tfy-r|0G&`|P`la+Amd0?jH{&le-} z6CYe`yn^Uc7EQ-B_|JAT=FZOU;dJ0HQXqX)xdl58Iap0v9@2<_y+_rYWwz*$h~E!& zs_1im-kZ9es(bh8)1?EBDzq@}9-HlQ0E-$tURN5eeB+rkg#-ecvIDrkJyIxM&0`yI z?=ori2qaI3Xull)qf7(^yGyBf)FG%7MFDLnmB6jUGyL@WDG##r)BV2I}bBxit0iN`01MO&&_P@;N%U2i`4#WZ*r# zgOAk8&nH=F8qeNcXbOzFt2_`6EKlnuPQpabju`^anf7$@&B!pZwWMCG~L zzYGkNnRr?EMUl{xsooweVcmFnq<4fB6sD*8!LElzWBkEWBV;Zr3c;W2ev|;N@+r(a z1s9aH|AaopC1CWMr<9u{J?Qx`esu}e`ZLft?(iTJ2lxI)0k3aaf^Vog7tOR_21C-p zt(U!iG>p8f-oR2JF$Fi}t+F;LaC6jT{o9 zjskGLgz-l`M;vGGBWk2OVq-ErX4eol0~esO$6Xx>l|vuHB<~cAvRk`_@d>mEnxR&3 zCFEd=s4a_&O`6%dRk*ZCne44X4DR1{O1M?9+T1g#HX|TprcW6|$_wShlI0QTM?aPn zy!j7#gBK8EKQ1x=D4i?FUTm?ipsZsSdwR<_Z4&qQ9XXTC=n8-2RJJp-o(qxMhH&24 z-&1buh}LpTo}|kUX1Y1a+6j3oXbHCF@5Oh2UaS`Yu=ag@C4CS1TP0X5)2gCqV`Zp; zVvdtE0Y}lrgjFZiZEy$8w9>xqW6wlrm|knX^zvd~L(n-jS=X+SEPAW@d12 zLRR)_6R?vB;46n!sJ6p;$SML;(jrdOpPkw?!@RGy#lDlo% zAT4tVx1JW=*i7^(?DpZ|;mmFxlpIV-8yI8>vhw#QS&ieqekT5-bo-CY+I={3LPl2( z8&L-imm5`$IEVjzACAPE+m_z`iI1NENW%NIxyFX~hc?jfi`frYpjrxmP;kVZkLT^5 ztGAy>z_+-@61g6{`S2c8{pED5{A52|y3MUz^Boz^iND;9yDTDxQj;a7M&%)1p&j zn}d%)2qLDv_qXJn{}>_>*oukJ;P{#UTS{VY#H$B4Zd_l>h^h$x?`qMlM5H>*mXo#z z#FCk!Wi)S)ocKyo#$C9%o=_?Q0aM_uwaK9J`0z$R=$qyh_h#8j;B zxkQX9TTM+(2W`#Vd~;7!?8L;bDI0W^-CC;VMfgX=o5;K=h!4Tpr2(>H#T@n=*LqII z*wr2_+_GL2MB*Dakt1jkF6bf(u2xJ*z`uBJAfSJ^fk-S9g&ZMZL@7SgQ^TKPgTZS= zsc|V}^2Zm%95+9DFFB9j`}=D+8BGLa)Xb_~BMwBXq#OJKz4nY0$nef3;6Ik1>FMDA zi!{8KpZ@p%qyGOMXL#Q}f`DV6q|2RV>ses471h+7jY56K2j@I*EQw6-1@Um68kb|k zxiy1Z)7Lf5dw(qU-|H^M4AypV+NqoXfkP^#{2l;-jD;0YD||W`9Mr#VpRCwS+kEzi zjPOp7$kbSo<*1}P9(`mtou>51q4g~+`u9uxH*Xb;11k){n>{KoRngn0@_VvEN>t&+H0~n|M97i z0N{bqb89SQUSfd9MSaVgj{H9|zi5bG3?PMZYxNyq!3LP3oS~4^FY@`nUtm1~%CzvL z&ywVZyEB;LlK-ds|NGnjiNK#cptJ+Mw#?pU0UUfK2SNUIzT>>ni{Sk6DRv143bQBIW1^`cYtlvD=PPkAKUo z6fC$s7fKY8Cy^ON2IMOUSDk;~zZfapn2ej0<00sS-Uaa1jifMtPgDVJCPrpN7D+;; z08_1?ZX8^{<*NHMTv72a+f{gpDd0TStFn^&JySV^0Uz5qBZl00>EHsGIlH9(JMp`L zXcY-YS9tPl^|t$CIQd(N?Nq-k&|hM;^a4yNVOK{A9U(Is{AhifOBKl1Z$6f>gEJ?w zVMT&OW&_@=rAu%6{Xq3Y;LO>wk-|gB^a9Pd>r|6>e@`?DzM#vqtJ5ShU*XL?v^0m` z6GaD3D5Uy6kdcStK)Aozp`v>JpKAQ$C%h4$PpsQuE8rC+{m?Hn==kyEHxq)(6)}8g ze}P#E{!9j6(Yx!&aW{WE(i`qiLH}W2v>yRqq7F3uqIpDo{$l{t;O}y&f`nkdcnw&E zb9$A~(iE8g=TpSELxPbES#z+j0N!MK@%B2a!W{=^}cfE1pHm>1w| z7MPcw-_zs&VyqJ&RoWI62TTS3Lomgsyng?s|HyC_-PA(^jQ?qhzmsfUSvV=N$;bB) z&?(WuxDn=`F#m@I@PkWU$(z*_FxbD><0pgucN2>qkXl&o8vuj-dzF8(JAXIDkin-9 z2QgS~A)tSv_|T8u_mTEL7)$vUus}>TxF610F|c%g9=~q<|NCXcYrt5tN)sboM4+x; zSpV+tA6#$`lLiv9bc#Xykr%$N$QVGAFn=$DltS>%Weo=ukRwM-pIk;ijwvZP|u6L>!Q}%3G`Sm`!k?WF_)x;TWJZ?09hLQl* z5L*r&JTWJ;M>8huVLs}WUiz#REi`ryTD-c$x;&cH!=II*xU>& zPbci+#VR$FaXo+Vv@&Q8%6n?$cCB%@3C2(8{%{iY@*E1DlbqBTes6vC;;r>1O{v{X ztI^q4y5T}ap>s|9_vLZmrq@^CCFCjFkKU9xf4IZHOtbw7nAo$gna03Kz5@%7(pJP0 zk;Ha_m7}UpGc}y&SVt~?t)d2rqYVz=0%!hn-%On!Of_~!tj|3)Jx#%n7h9}1p1Qn; zQxs+5)ELTOL2kEI+FZ$W618Zmh>RLbEJpAf;aGy(P#*p*pf!Gha08J_UwSKADOLMB zs$^!zEXMx#-P<&o@{Lsw!Q)^-@)=^wpJEOc;;omG@X8*p^n61Dwo)bo?Kf+N0=&0W z;_`M7MPgBbsXb}Yc!oq%z~6jVWc%ruhFyw&RCF{2EmygQW?DQH%~@^b zs^Up)QsC{6-lhP`^c%`yKto;$`?vr@R`Rf2F?D{e*)F6^%J|LYngvu+hzeZ0Y&u>o zdZp4JrRlQD?r@Nqthda>eON?UOUtEK-($9qV!hN-a$sS`AWxR=3BF0ker{ z;k8qWk&;^1cav1dsHhXkb(&6e<}5r zps$g;O+DIaa?>=?*DLjs>3pXzQ()?H8TW8zXoAc!k^3Z>XlX*(P4xZvS539ZK$8L* zwQUN?2f6fW_@{3siyUdq#zPEsL!A#soE^@Wleg^IWl1+;U*jiAKu^aDyp6U{>l-S9 zY&Y3dl6GA$KEL!Sjv&k7JsMung*}`70%B!UNl=bO^s)`J_B+mCO}22IV>v959l%i)N=CC|RAN)uom>9~gFQw5jm zrLQ8Uc7TW?LiF4 z<6GNfN@Mj{`GbWpOR7mb7a=7)T^9xdhxQYzW5^`BNj6BWlS07Ay`fiX$xe%Za*|=#J(PB54@4A`Fp*bd;@?3WL5!XJJporU5 z5+NO5if+ZlSa!=wdeix>LU9239jOxmEfg|9`|NVBOU}Bne-oNezpCCp-%@NUr#>cR zr2FsPLj~+iA=vo}z}?jI;4Qq1dKzG)E&N{DH*H{G)GZ!klSZi^e0$SeSHvyPT`e$; z;Jgm6=kq<|HTmXDan6OyJ_nkD5<4G9HSuM!P~01}*FC3h(fbdamrI3tn5kF9jkij?-MYf@#|J}ry z@Y?g5)B+{Ea_HG{wZIGGkCE;9$c+5oR)s@8u_DhoZ>L&)#M>_v6}hBmY${%*S2yd7 z2qy?AsTHoEE#n#HEh>9;Al^d80?eNiOdzdCK8f0ZmYLahcyC7wMxC&hA1#rY(!aM@ zB%aySJ8~;cDd}Rbi|K8-+Hv;l)A>hkUF}eA|F^>Ag#}OxGi{M;5$fLXxhr*&L!82} zTmv?tOHRei!r0i1u}h@nYufYx23!fwGNBnijwX}z8XUKC+t{M`7%BiIqHQ0Cja?RWg1YRV1G zfzYg?28}f~m1uc(4VPkxI5zo)OUsg(LUS{6R1WUA)wEuEl`1iF0n)`CVG&OgIi*K- zj#QJv(IOgyEVCwDu~*YuYZ7;>qU}AY=8@Y2cTVVS=x&=1^Oqp(3aF z-953h#<7JN_Cyz}TQx@)E@qmxhc>o-V~Y)alYP-u)5=*q%xbRZW-1bwwQ}PoG`L#d z_N|d>zaPEKIG%8-8TrTz@jI;j#1`}~kH4K0_y+l;D>uMyUW;fYWF#Z78o(tT7-ZUK z!BSQG`e0@d5!F?jG{&>-eakn;n2Wt`+@n|=$k*xHIa*f(EwXTmW;9UUA&b)ig7f#yibll6P3!7r~pUl?DR92nvRRBy`5%8CsgS$-Z@p>C?i zi>1Wu5MrbGRRNetS~sROWS_wuq|1Uaq8O(*#UBVs)>q`u4)Iatq<*{@%mk3 zoslzD$V`Q_i+3JVaG5n0XrRUq$V`uUnNTGE>dS?=0LOa5up0v;1X%7a`SpLtpJNT# z#BP1_**#favL|?2w6O7lH(n#Kx}eC7qui+K1gT5bXY0-BrK#|9d-gKIi-@}`(d^PnHLX&DE2zEghL z(4RsVH9R7hE2?Y$xM2bp=H=VOFz(;Pp#9Ez^P%{rvJaE@9&|t-QDE<&k18+0n;~DUbH-7Puf)G zB2dla`PJo-ht6yALfYiUq^U1l+tziWVo00M9UVr4LI zeTjZWacXtvbluX>{VU6_(>NT9xrAlKO(}{D9M_Y3Uy7Jplc}+3l!`x(wtjkI68Jof z5?jHj|9d(3|ETQRk5kv0#d4g-PBb>)*uW9)RnoB+Q5>V*qBhcSxF*4aGN+3#Bb?1k z`@WxrW=^s<<6FcRh45&z9s0NmQWKoF3dvvF3vC{VRC#4SnvEen<>sU@Rkn$@SQ|~5 zj*iEJoRPH=&%Eiss)(E`nZ9CZE}v5Fm)0`cGraiWy@S?DFyV|z!51FCy--CGKBmO5 zcD76I+HN@ajDauyI>hi5h`2%EVQi!U?6P^`^rREQiC z1a*8{XK{ICW@3bV4d1(Y!6T1cE5!5$g%!n{{-M@=hJfgo*nbKwl$j zGov2pE+W}Vu9&ZUZ?%;S@cGn*gZj)_c6B`VLBl{%F~30Vo%QcEIDm6s${{J0DO3|{ z4u*PTobM<*=enXy(fOv1*WDDS3Ca1^E8*7L5v^nIzDl?v z(dFUA~rGk_TLV14Y+>_pANoZN}+(>FFtmvpYI`#gNc zV}k?dYKUbLnrhd^B~W#TdDlwRCLU@?H)~VsF6&O+5iEgn-Cm_}NL#egjWc{T2-Xp8zlyH&B=Cn3h-B)VAof?L**!RsDsi;KpO7ynONm-0vN9dd^KP$UZr(Q4j`+f9Rw zT^Ls18*EKO`+Bn1NEUg}Xd8A3D*`aCD4`bVzQLe2dKqz#-_uGVDKx@x8bio=AGV5D zc`ESsVwm;~m;pqz8Mhi=SFg1YVbY0e{cNEdLn44X# z=rDl$prJj5k;3)-m`J7geZ}2p1hIFm1{tA(8VvOpCq>}w2)m{28NQ#cu%|_Y>&&;D zSNh-Tljh2bT9Noq`p8C>B|uEsP>D8K6oid$H|dIR4VhB zHPuG^kHi@er}?kUr{((ySZx-Y;+hXyjKWr_*`)*t*l^R;UcTcvU6L{9AtF1!)T9&J!J*crD5ol?) z&XG58%}dJ3!exUbyPpbNv@~~=+XX${>5}mWVGFX+svzRj3SzQlnJ|IkT8b6L;@_#q zl;_4A8zo(oiWG}{Lf3yjcjE5M>}1)^Xp^a#SfD=`vbcT1keXUubZ?5`>X-b&2i{P4 zUEyK(V^B|d<-=QujCxkX(it4M>rM9T#Dm6d-2v7#064;F&JFh2aE>!u)7+quay1cLDX zapnQ#abFFh#tBw}i9Vf5Vt#aV_8=ZU4di4rPLKjGbObi$jhTjs2Y};=R-}E+0WGSd z_Mz#Ue5u-LjBF+;hLS_cm$Qfut~$W`lgU8VCPyW-)U;RKR(zpd*tAvEiqu8`Amg7z zfEFomT4b2_0NToAX?Aw%FX%E^mvdLaO-OM1KGe<+R*SsY$M&m{YDWWJ$JV<{C6J1| z1@}6}Yd4fyg?g!EpFH3`GjmsOMRaMi-+~!rn?U9mek|wTW-?OG_s^Vy5zs()6nZKb zxfhA9c#8UbD@7mWDl^}))*ZGp*Yx8zddKH*{KHpRa+90qOs5@YuykR>?9srGQ_Sy4 z2x|xxzCXS-yOI;_%G8vIKS56B-@%w>$ULEY`s(5Xh31ceYWFw-dq`7*w2)E1*fdAZ zzOv!lHN(=QwU0x3c_^XJG0iL(%CMCzK6SW$mF0?`U_uhCy6xo+qHupmL%6T@018uG z2vsV{(3n!%&{0verF>&V755KP+0p*(hOQ?)=m`_yz9uHTW2Ac5E>GyDaA;L%x1H)S z#-e2FgQL93dp~9UAwd4scgk-5n#wGN7hFt?a&SSm`I);q1|*{Sue_7m(+#`q)lb)W z?60oBkqpXq!sB$f?_`jCelS;SEClVTS>0!T401?=$GM(q@#jIVbBO*L4>iVa^Hk@z zKC3RgR%3+0G~B8!w@b->n92f;RZFqwK9_HuJr$CO|EeTq){s|jF{5`__+p*wTmT=J z{h0;G!ind(?NX(ayBQcf`iMW3D9f!eCh9m%XP|IuX<$=DPN1!+T{duala%^pG%*^V zp>8$LEbbob<<*$gIupa_WzVQ$Wtioy;BZyeCuK4$TDkn5yC*B5hK603^jFy}nB;Q> zqcr3OU-U4!zL42oGEX2%pJ-emdU>}$^IOS!^%SQ`x9^%bIPO|o^makz>JDvD{VCM> zRDO#U62%i~7W)taq;k9bvXY!!+yy_6bSrM?u+d)7-{^*A{UZ=w4aN*9u#3k%tVxk$ z7h{iCL5=%bI0bxh=+0|N5Etja^uD^QDI!8t=@}@raQ;HVh;nE1$s{;Z_)AKhcoU=y zOM_x=!FSz;~R{oX)0oRdLJQwoZ%uR{lszg#q``O;DjSfvz6R%dwy%`EB z>Eo`#eN3C9Y6zL_w;FM6^KhHFP@fpy&TS21{ZOLSXOYUYu{4p?QIAlc*TH&6G_X&~ zC_j4BNCbE=OXed)+uZqLs%xXsU{ya;{$`%qp$G91T9&9laI|9uX`!Rd+#`cKZSZ)| zdW9~UKyq*9rG1gM%j8(0;s{QCbyruZf!Wh)tKveS;xLw6G6ARvf+g(Orw8pFOxN>) z$bx;{*%c9!^~sZossTHz*>%)RtIPs*J6LQk@Nj1-Mg%E*W>&pYDWNOX zxtg^(ZaU$#!k@j`dw{Q;wqvKb+#PNgZ9FA7{kG~0xl5|83fs~%It{0eE2SGh&Y?mFXYBWukq$NT-Y)n*)@{f`V$Y#b zq!u~11+n7b%DM7%YI?F1*6<~fGq-fZrFKd8nM)1JbU`f(mmboDMW3o{>Jy3g-NO~j zNStG>mnUsjPIS~+TSKpE%& z_aLT2xg(pI+o(l%*t>#`t3pX;f&%dLiK08ELd;V69vrVFwqAzdUGB(1SOkbw||8T9ZjYKw!$4>Ntde_1Y2A|-)3=MVtPg$Tz$V0qO_ zu3M;Mv5u21jJ^0|xXxpuf6hGT6SOc^7y6-(^A_-E391KI2~&d9(^8Y6VTKDrE(8 zk#Rn3>^Y|IQkAzaRT${bHnLqwvf8kOLQZ9jap$beX#(8UJZXH$j^*99$t0Hqnf)}Q zo5LV5o(r8Mj%by;i&)jX?b>{{%^MQsX0Lf4TQbM4*_Fv(;BF#xZCfWRQY(j; z8$|iux&MHipojaQuy5;`JKXErah(t3u-Z*8^H32Oi_E)&I@?xGMfv=wZlrpXF@5VY z!)$G0zdt+X`w3OExxiewuRz@A;UWscdjE2hiev>Q9g)|nx(Etx``i2u-+e6vP?xmz z52i+&kl$0pE*e8#?2vNd3q)UX*IZ;Nr!(vCKxqzL@QP0pZ@qjAguHagxA(sj6`*cl4vl3uI?K%-(?I{2LAeZBHff(JVqs$_v6qK~xTm~iXBWm}xlbfd z3TXo+-ptgHYc~6|Y6R_0W@k?pE_}PnB9N4L5$fQt={dlL#{23NCRNl#zV%T>NpTFd7ggY1>HcmzQUh$Cf4=8F2w*X(EXO!z07YaK_+`CqmwcPf*i?!} z`YNtbY`7)9b`T(^b6MET=hRIjz~ckrFinf4$#rY)Uxa{I1^_uxZ+{R1^GY!SC<-iE ziy%1O=nF+L@I0$p>FC*SPQilDWL#m^4qIrWziy`g0bX>Uev+~iI++Bm+8WA^8qQV{ zrHTibcle9-{3fd5)FaoUq<*x=RiWx zK~))Ds0pBtWClU7Zl?+J0U!#DakYd>lK1%C2K?p3ZcU}^@-0^>jc5Oy9%%Pu|U%NvHM{! zw8O35?z>esE3wR2{@8>^#g%8^Z%P*I1&`jer$vwO>VV^1A5nBTZDWSWS{%fZj7i?o#bsj8Y0KzJ=6Z@R0a0xuYn7K?w zuuuuZZ@s@uwp0o(2w5I@h^B?#1qh4W_FnoHDV_O?pV0S!=T;gJ0fT6s~mDwQXnG5 zqYwqJzM54PvM@vF8cOf&M08FH^`037@LD3HHnq+dW0hL^1Md2_;sBM?^8_-+mMupCWDElia$Q178)A*_$f01Znn-fp zA*;DgZ9Aa(>I#D8kGwKQ&*!VtmtqWj#64&1BU}8VJ7GjFVL>-I=7mJc4jv zUYKwkd=gmrWVCnOa7(1cYz+0vo-syyqSP(J@Os{|EX?D~Pt0iTy{#I@b_Aw}Pt_*4 z=sld!#iwKL5a;8LA?d%AoERS-VDUwA0!&l@I^{k7<{ylndQ}?sm36gtM8rIOXSAeS z*@y`Gfgdm-(+(!QD@3jrLb>J|iV)WE9&5~ zI$m!#kK+&b{teA*hvPiAm{*X2af?H=(lC0y_Tf@brqMjA7rf`X^tF;;=&2AR*a{y( zY!uwU%LMv8aTPe-9M{~6g>~hT8dqPcWgu)Dy_Cg?cKP(-JGcg;T^-;#f{a7SOZ_Vk zuYpSqE)6o;AfCxW9@qtq)$T3kG?>S|2^fgEJ30{EO7V$rw29Aj7p0% zb|);Vfg<{>acg9&*Mza|MAIm1M#ZS0MOQR38ZbzG(H}?sp;iB0=o?h+=~)^uu0>zJ<-o}K8KXt zUZ6{k6ZZ(4H{}_8uw|;6cs>gQNEEMt8PIo+EqJHcMmTUOPZ1otU74ppHu7F;wj1&( ztEbCpy98=lp(wPew7zw3O2{7RQ7J7;dCi__P4@D{NwMxu%oB@x7+>*SpyFKm@n)55 zWz0(AdE%sLg6ibIvC2#mV3^IkIRKN^{@7g|8{%l+NC(MQ5+Sf(8>GmYJMG5EBu8GN zyGg-fI$o^VYvLj~?JEV{Ma$7FhzqQLSHn&-pCdx9hCx!u>w&teX;Wc3UCn4?Q3;+h zj1n9pB+u%iZ)~YpO!6hoO!B2kEg=u{ePVfg;+Jh4?(iUj<>mj9P6F!Txi)mI7L#jymIx5xiIl6E+v z`aW+43J^3dL@P<7>go}W!3~eHXU}X+1`CIJN}bRAw!Wk*AxAJGtfQa?b3Y%bR>%#* zP!<3}&Z%J?o8TyCJyqs%Cn%!G(Dl(+*8_$7wT@E#>sM2D8jC0YPV5nfz(Mg7DPoDhYzQNR>fyVEeG&0!DPX0_O2=ok+zty~uv}89 zlvqV#u#I*BuPMESJc!^4*DjWC(_WKEatAO3ld+gcEECFiPU>ZkoLOuea*{93r-%lu z{yb#!k7%@a=kA%k0PEkAaIaL000Z6yN%bKg8Ye4pUxh3 zyAZ48PLa*g2@&J+)2hqUWBaGE{V86%iO=OWMhh?t+Gw9z`(`)<89QblZH#v4pK>2A z$(4&{5rV2uuR1{_EM zo3G>knN<_Vwb=TRVeh%7Io7owuXqs)gK1fvUJfn$gV$!Z=1OzqTKooAS9%Ba z8tXIDRP4Z&X>-w2t#jK-0zC1AuEjRBD{?uq^{N3O-|wNSzKc$@kC^u~kJ=c?y}i*9 z{yEu6bjNLN|BKyk&P`0fL9-eS!|+2R9=vqOs9w)q{6M78!ppnJumwF$Dm~Ri)d%pm zgQ;mAy^~-biohP3`#t48`ENgldpgzhbfoZJwNSqc1lAWG#yx?03I7^;|0O8I@N+Kr z!IdO8+{57E^3&rxX8+ceKQ=60I8gQ-ohgW>#o#vYVgFs0e`WOlTI)0Tp&KN0DDZ8- ziOGH$=PeGV{}`?uoPX3*2Vj6g@PkF|q1L?r5xe|X!255XotGGnURWoI9OQa zj>Pf5oWMVwhJPAx;FC2+s{UFu{FAf9A#kX5Y=)=?+^T^*fF2e_sbnp~gXEZ&%t+54#KMs9pAU=DRzqi$&gK@`2P}ZF6yl5B>vuH@9y>?hX zj&aG)5J}N#Daa5$HU>9WhaFK7|-IKTc;F}D`3ggl@^Jxe~McGB3gXgAsbK4_%W9_xaoAx9fRR=17Si4XaZwkSN$ZOha?PoEaQxLz=Ef83#b z46U?k$xUeR4#T9pm>s3<3nNA2qAX1Nzrgpz)HK`NHJHt;GaQ>n*&zM=p^9O2kg}NB zVfT*~<+*nGz6OOSQ)>`g-f)*?59_|-JW>m2-nZlMD@__{N&31A@9u=-dJQ^JH#ClH z-}`IU>1KmJL{Gc&iW-8V9>CTTiuD?oMfMV0l8xZel2+C>L6==#CK@19lk>s0T0e6FhS zTY)7727w!F&0%C+OW;U*7}stbMC*&`TgM+wR#y|B;a-m@xU@Lkp?t0}?yMl7W^qS` z^oy+IgGhKL!*0gmb4=yRvRT+NjZ$Ybb5iIdEeBFnt*l4G=qWnWp;?!Xvz*@-z2lLV zonj9AI0U4)YHEfH$`aR@r^R$Xov*KIfn-NGe;@N43Z~w;!)Jn}4H{MdS?OHmxzX~H ziwS}@BT8x$ z)4&xBA#y!$Q9pdpK0S9`J_XGoup@y2ymcB0BXU%qbv*U#KUgP}M?Ger>*s@PHTc)iE3mfC z1~OU9@;mH!DKk43-8UNN`1EIwAks%qRi zn}fMYK|JdqakZdMmAjhqHDNJk-bKIfC8Sj`wK00aQFG9fz+OWBt^I_MT)J>Y3~U?a z(22!*)HHVghxNf>!ra1SCjJq-UY{^@%BAsozbap~ev&Rai$_{2*>v|oV~q9IpbM|B z`FTr0Z>Z9WiirGY{Bz;mp>-D3T@FkOBGvC{yDod8Uss=8zO1nyq*PlS>Z{k=i44ZB zzDlEH*lyt3y5Y@gGWse{oGbIzz40B0NWNWAEd!+4ssjG4HiZ zZ{w+UMfLA(>7Kr)pMT=%9ULhgK*AX_EuLr2leZk%G?S@QzbZ)(cyqlt!%U;rG?taJ zP^*#TEw`Q3)tmp_1u!!Kh1psyVfY9hzPz0FAAS93MyB0sm9Xzxwf58ATSQN!OyE84z1cNajdqW<($T)RyI zl$%{Rt<&lmzY{qfVJK0)(;xmZc@TzX&U#dK8ELlEJtU1XQ>D{_vfiI5-D;o*4`bGC zdh}DV$EWjT#kM=vES z>bmTgFp#ETL6`(Hc9#GK;_BJ@ny-IuCrS~y?jkJ8fc44i+U=dgGWv!n5boRO4fNRr z=d{+`t$Gjzj=BhL9y}s+S>Q0KKCRNXnrKX}KiY^-(~&M!Xt}SB%UUk|)hR1VL+1|L zntgdjMw%lUb@!Pu&la?guo8tJ|GTSa85PU@^*Kqjah@j+$&F#3TO&umXzL2t=Y3$F zKOwQxsw^7nzoYji-N}H)gufYWYAn2C(SnwtDR$18q8=&WKCQCq$2*DGR&V3)ZE?If zY7am7A%A`MWwL~+;Qw5OQhqoXdD&pHBGJ(6=rH6ckjq z%a_uH>(exaj;DjUT;Pa(${h}ZGuY(=9^W9PX{RD#tHHY>B7uHdE-%H;(I5nEU*=N0 z!!bOaVDV4Y_&p(24NUY+Wo|zvf(E&lzv=cEtq^RL9IF>XOWdZBd9@G+WY7C9R~2PH z^R6b^st)g+YB5*1CdBIOhFQeG%yxE}x)|h0q_s+~U0Ox$1WQ*&&OsK_A1|@4B{2*I z6=|oj$OTRARI{_<#LifOQ^AI%k=b0zZ7ViFqY)P4iqDv#93ZpOOU9Er|DJh6qftIi z;f2F~B>OTwbGnFO-Bz)S3)E%y;=wmBi z2Tl~y_z+lqILj-nV(a*v>_$-K){1#;*w|>vg;LCSJHe;tlj5ny`SOIbYLOSmisvaV<7Z*P z?g3Z_O;J55*N%)<@AG5CA&*bvp|8eA*>Koq{>TSE!d5+8O6w{tGhYoG;jlfxo#^e> z>CZTQ;Dm~|FQl=sahyID+KVgz4ni@WHGp0sOGuH%B_dnl-JyDP5E3+nHHvF`5-5G+*glWFGHMhH;DAe{X<5&xom!Tc_ms#{6{%~pK}-N-w5{6AR4*7G>;ACJ#W1OS{=VwWGmV)X zk>UCn$R$NkU^O-}WKr?O0LB#O!70dVv?MeJ;kxN<{8;mMylzSl++9@}cyazDslEQ= z`Rb07^%B|3&=Bl`>>Kws3UidJq$)r9eF(ZfTK!w%x359?N4vcJPJ+fH;ieXHF&o^j4@WPqts;$xLs6lQ{?GUInbB90~gEz0s%DdFsN97gyBa85*lDw~pFq!2={fa0#wK0t8L)K!ObJ?(P~A+}+*X z-QC^YhGCH5yLtDw@9sIf-~KgcX70V+U0r><>Zzxyqiv=Px3X7dcpXkmQ6nDlW3(l| z)ETWZ7FDtQ!iw(YcV30b_OEqvl|{ysB{L7+E63TX)ZOvS zP0XCNJ_RqC3EA73P_?oWE2~{VII^iZjBq+@%}47b@5W@x7FtzQU(+QXHF{snm1YIh zv~(OP+;1&cZAy-gBCI2tVTisD)%-P#ntlPzBPfFjiU+n^Zn1lG9Qo1Je~G2@*Lv=t zM~f82hewFdTa}{W(j9hxFH(w2>Fsnf)9t|zi|~_gXt)3IsEIKmPP16{@%~5)3EQ>SK*}F zM(iQ0IX|;{^Y~v%YCrAwyUC$Im5*QjauJf5QEfEU%akYm>DFa`qJP*W-+yG;3aJvf zTkm%BM}7SIwaGGhq^wGT*}TqRI!&lZj2#)&SUbKeWX{CDgoxcQ1S_7ou20lQZ+jpB z<~MFO#im5%QM*to*h8LwBYvOu(Oin#P`DDZWOx5=v7F-#$LMebpP8l1Di^}S`RW9A zRnmh2dScXJ0_w`t5IEnxdiz0V>|vDY2wK?23>dHwyIgVWmmusU)9(mUi`KChl%{wev3p>Z1YxWgAtbJSGw>D9uRJiza z247^4(IDuxd?O9{p;au;nJR@^+7S6S7%_Q?(Sw$BCHE7D|MQnj{^r0Qi16#oRV z=8=lbk#Mlz%7txX3&EF1Z+SM$eMgEz7kA(84zTYM3ocw0lb!v+k7Sp)?!U?&&3Ab? zcy?Njm4fsI< zK9bVo|5m6nQG5UVs_CJM$5H5jUFRqkA`|m?Q+dWM6z}-T-ZU5X9#*V6PXt0-JQzH2 za{(oI(hQQw$jH03AiOAJGLc)YNkhZ;88=CzZps(m(=2PSq0S=O{D-z#qv;R8g1R3G zWg@)M9`=MdL!osB57`A1jqD8o2qv944G9);+Ybo9KXJwKQ*7GrweJo1a*?Nu%iqa<(M<_4z+N4v4Gk)gBGqOtI(AyDDe88p?W$7H3m&0Kuk<$&Evq6U9}|=Pr_o z#GpjGUT-jjW4}>mz^(GA7?;pW?aH)Vmy4J7qR%4yVs#~@89i*#Yuo1BSrS=YFy_`!@P?MRy?yXq7YkSw!bcEH9icp z=B)`?T%C!eCc7%$8%5cAZddY^u6nY;-7=sN;Ke1PZt5|mjx{z%Etq=S57Gu#(QfW0 zEG;8W5Esapy9f5TCj zqv5?5i2ioQ2C_z0gVwg%hK&*%4BzDn*54O*`UR|`|8S@H zj#Sk8y56{#*QZh}ZZ+dqn7PXHr)=9)fLpn~?D;veIxV3lW3PxghWFR7Jnx-AblEo{ z9eNRYE_-7PcJqbBbGoj*Wvy@0-Z0pi+dL;L1x+wu`P7T|T7hUZ(e086BPlGZQv0 zLe>M^@yV_eGm6w{BsE37h(g2b9NH9`$*MIgRgQx^Pv+A_cdrMtSJZHtmdIs0_xpTY zCejL~Hoe<>Wmc2>Uf!ohyN*t%qv+Yy9b;a=>+lMK*k$PEO%64OndvE7$%vF>O9aQk ztGv*%u=)fWDDAzh1L^I~gIl8jX$IvAWJ%?&+tO4d);;61Ua8%g07uRypdF>1F00eh z%V$|lqu=}{*GveX+PPQJus02lUr*N)7t>8NTJ}FY3?0!D|`ecJ^P9h<huR*I@qpjzK@q0Pz&+VcDZrXC~PUSo1v0UW8b-8Z2 z!y;}Z$K7MbW@{wVnyz`2wKp3)rsv;)efx4``O$K&WHZ@u#ssfkPlfUGX}ALZ^KOG{ zPK$(HZJ7)lLmdcRSCh8lwov%oe*QQXf>k}ffDWXz)gj8f7-~Pj1k6^^Y9A2ry&Tc> z(Q7sRJEgP{lB9`yi)Z6{@&e3LWX^*dm1(C#%Qkesne-lcORiAu}Xng%V4$ywPcF=r+0%lzbr%84%9 z|~y^5qRIYIZfV0pW9RX6#$DSFFD>FF+S63HNS*`tQwm9X|8Nh#{A1z%X%BrEaF> z+&^rY34fvb^DC$ldNfmW|0sw7WE8s6oy?(~k&%$XQ)J&u@A32#f~$WP@W# zb=^LwSL1J0*b?$wiotrg09|&7h$0 z#{&WCo-r=L1aGw|15D^zZMZ7s&7`A$fpV{!p4)twpySy@bDG;tjyu+zf!raP;>wmW zW+R1dl_-=r{b`+bB3{~sKgIsQ!?nu~mDDST$(@6uIqe#DPZrMH;5>T*fXe9hDTjWj zs^7Jx;Z@(Eb;@Tc^qVG1su|kD_p!O@rlaeY|Mq+{J??EuuseAtB_(^R&F5w;k^smMnsXf4 zIq@tS{9Y+00yxOf2?qICk2}a!1>b6O%e~Cp7(B>h$As9ts|Eixa^S?l{?E-|K$PNe5gl5ijS{G9`ezlJ; zg(W2I7%A0ln;}K)}7vI_AnEGm#cihx%LjR;2ljJkCf%fCevxBv|-&Xk4Pm} zv!>eD+f#%*E6$C?pIn|);-Xc5%+mA5Ogf(Yqr+uMm1-OgUwpQGI;Ho+TX9TR+}DNPpa*J4C-*tXU^`2xcJ>y|j&#G(@8XKhq_tgGWAT z#`J#a|X6R{pYHOzaUZJv#$Z|hLeT#&Sr=C^= z8(f!LbOAGYOUsOg+F1B_4|_x9j~b=r1DXe4|+n|&}rrIAf`Y)sXWLcQO2 zQmGep7J9pRc|PpAGftiTYHgpmDP*!f^?S6HaU{;5)VBEp;A@p%DdzkFUXBWM4I6!$ zME#*eupsC9ElQqW&y85r<4J#gI9zJvR!Ku1WbP4f+jAGl4N~F z4w(mnBCThJDHZ2qmwz>h%y+9|5?W6rR&EYGO*?g$5En|?O#RNNN3we;d}@;PM}UDx z^_{Bin%)lul5g~g2;IasEWDL#3COmYujl_v!?EvOSHC(CQ>$uU!~6jen(i7r*=*s; zl3#i{wcRF+0@=(j^5II7{2h^I=P9dTWZ#z^U=BGCosNaK(Niq*8E@~Z3s43746K^? z6cmy>a;`Gq(}~~islH|fyCI1xytTJmuF33W4)@$R4L{WQtlfRU^C?cf)f8V<@b-YD z`U%!S=df+d5~(>C?dt{pNPcki%1{0ZZ%0qskVL_X_LI0r=bjjV`wLD-hb$e?G2b zvyX?vsZ~Sr^5!Q^#@ol2_|nS)pWbsI>unkd8@CYZD;X(p%WLR}OaezdTlV0e1j;OF z>4p{-ao8VmxYUPa8KUL{duCCGxWD10Uu(d&gCUI+`I1lPpESX*^!RVj7=RG9q>7N! zn3_PcJ=9TotmCpgcp-g_fZJ6L%`3k?Yrql3fLnh(5+AITP?1x`W!DSjD96PPByJ&V z61W<=!?pnU@?$le7(l8y3S@>%!(2t>S}B(z@%0=Ng<31P6UviDZ54i|N>4j^He)0< zXITfUHaUEBO`GE?l2Hgpi&{IYv7{KA4MXia*U77ivXvG7!JXA@9_hBbpw0pQ!_5eP zRL02=K^ruBTbkmy3}nA8smJkYvO5JoGV z9^JC3V}WMcKj1U2t5m&Biz-9Jvq+wIgXl=&yXR5$Nn&A=nkHZTjfv6**D$p&GeXG7pKIz^fC(O)g2k`p5eQ<&LL?py{IcF?P z-@rt3^sV{pTCKO6zOEQD&>#`t$=HbpO_Om&PhKh|q;wjWdaS#}>B&2Vf{vmz7S!kV z;jd%u^lKVLF1P6YcBWi|SCu*7OhU=l@6>vUM|?zmw}^(*j8?dRyc>)eys;Z)S1UVO zDxc8mX!ci!`3KlNaNjdSJ3hTaF^*$@W8f;`TPv;oE{te-iEE5FHYGis#66>fSOc$N z_7bIOegD3H=v-V`kLLnHRgkv6(OX`2&}Qi3nOP4om?#nXQh95jy)WA|Tb+8pf?i^; znGe`egGZen8uk@Km4NyJFHKmQb8W{IvRc?P4~I_7x6W`uCi%wW=M{?7wd~y^NtVdV zUvZyvy8r|k!M?Z7-INZY;FD=+w9r-e5V$eleYr@Jj_0sQAKEKAdg)BPhS)?%^ z-Wj-C_w7KnNA_PuJtYvo|2d#th2o6$WiLV8e=q~dRZWDlnAAd(O00582LR{zauK6P z?&+Ve40j##t*=|O%KAd(9-_YW`pdDvA7csue_Bo6$ODKZV$UATF2(C^S)}_|*oul# ziL|@NxrrPC{F)?@w{%PNfnh|1TbIM6eF91TGp>iyBjF_dp|$8cr3&lpvfbnVf)pQ7 zuZ1V#zi}TedJ0{!-%>xVw9Kug1sia$iyjT`hM~5UYH9h&FbnE1ecIDu%@iHFLKrN6 zNwL(Z@Yl{=>v`y~*sC@nOv#DGh$zMKLzVQA|Ci2&+i#yC;!Qqtt#I#CGF0$Yd9E2q zrPh7tRa|T#6 z5unoApIs73@;X=?aKUY8kFoAr((#Qr!k;dZ#6t2t3V^=(e0fTE2UVETtk+LCVrX+RtYK9q$ruSkjGhM$pr4_w+EV;Y)8=M(r&FLClQ>lL*K7d5rD z>SB7$DdOyKAp4%LkLq~JaqaRvEN^Fm?unUk$*>JxppBM6ps%Y3Ag!cBX7vP@{=@o+ zjg$(egz@p5t-qOw-I-6PFU}2>fKoyGCKQZPE@zMO+nnsb7swUT+5cWA&fF^k7Fw2t zeE?{mQVx8kR|Gpt!tcWhZ;Y1aV3-hKFs4`}vU`eo{@YpE6L5M;ImhlG$&0^iIh}9n)P0`=Nvc(xq}5Lz6T~(i=?*JP9eB+=%J5sNj9=@a*Nqj@%Ev(WLV14mZJr=eY83_sK-Ov8Wa(~8K0 z0Bx#$CDN3t4v(f#+2|GW7mDYQAX!NxE7S-TiJBe9LsyPP?XSkx#cy5hrieWI5d9tH zywl|#0L>sweiT1Pv1>&(w!f?84D-v7wi6;jV*1nAciguL1tgCJ1fUGqb2cDZ><%7`1Q(j8a!#O z$Diy;%dO?|I>)b;bjHo;9Y6$ij)slNrE5&BhaFFU=ewvDjZ`IusI&b}bY)cv-o!auq`?Je|_} zH?B6MV3vC}A00ZJ<*2z@&|Mx>@P35g>HcP;FKvlprlSzkg%o)RV$(t~R$Fi_-DrO0 z`Scqe=-At+FDj-+!uEl~Zzk=K%l?AP{qQ#a7@*E_Cb*q-O_($7S6Nsg+U?=99WBK# zg|xrwcTXK@qdb3%&~e*=CVs=Tm7UwGrMi;zopC%~>?;Wg;iUuDeN3Dt4Quot(7yz2 z!NY&tgs<_+eDfby`HkBAEsiDeK|QzN4-~Y}11RvN>}K7!@JUsjId%)bYwF#_1;ObfNHUo81U9|-dWCZHRB0RS1brbnQSw$gu}JyVduma-P?HH&@!XBAB4RuWBV-y# zfz$W@7lM13NFDg$Rrwt8j#<(ipsYyxr$64!{`J6|R9GKlH;Ob6k-20!2q zwQ3TSuxl2u>>bn%w9Fs8w}IZLmi!qY1_l3ER5EUIRb-&g5Lw!m$T$j(iMWmk%M7+JR?}sljo^x@#ASrwefx%ZiSUOpwS z`lYrv@G6hV1ACrWN`@LKe7D!mJm@%(7nfXUe1$O5J5l9Ke$7=X+Lo%l4WNxxDzwBf z6_>SP2^-W=@wqIrDQ?C?mvNdz&oZd?fYA@30_*HpYn?1VpCmi4Q!L5Mto;tou z;#;0A>22RP$}Jz;(IAVv+p4l+ou~bP7mTm3{~0?U+Bdc;8(pJLrS!9HBcNY9}=VxRU<)5w85yqzW$kJfo)$m4WDU#PRgM;v3?<^CcG!!H+9lV$p`43 zNUE>foHT%l{TJ2?r@@8k|D>{}Vb?pw=#npMd3ZIpl%)1dIjl)k4oIZ)=KQ_%m+B#_ zLP-KEJm?3x=Uoo&A~llz1FaDwW(r33hqk?_LkX+IT6L>0@9^&DTxtDKJ-;sika1yf z&Uw(9+k$&tid=EFh=Znrk<|qr9`Pe^cD?vqpljV=S(|p&14Mf7d;iIl{F6uYKH2G> zJfz{>OAz9*5YS~%xaLMh7OI+fFlt&(7qc2_2Bnmr{Xf$L=fCoLl^*pZp6Uy3MCuXO z5BkcDzB_6_?shGX6?Qt}Wx1>F@RYw#Y7l_iBWipzc*rUeY}%#!^mKM2JU~tMTJ|4m0Z$DjU5q5mu&zQSRQHA*KlA z=vSB^xtk?TyLaKaLWqZ9vAx@;#q<)5SM8o}yTZ>t10>lt5yTFK4RT5MCY zm)zDW#Uvvl*?r{ylRNk^wclVg7Prp%5@0DxN-m+e6W^OlIW9OVW@WK2Ii7xd|C}v} z6+Atk;r%zI9iMjE&pIefd^Wmv=eVIjhMpjl<6FLhNbZ`@Dl%2+FOb(q=2M(4tO0JP zqeoF+>Zgl6eod|ga?rCQw0P*1i7Q$2UTr=l^Mw6+oU|j)x2#%EZPc>_&CCmKh7Q6bpDJQE)DFmRURt9wcgOIGPo1~7Rj+&lChZ~*)+B*{sQ{+K-1kuroom;a zgX+0d&}T@Au?xTyFqbXS?>k4i5Wm1ui87jWI*VV>M+^redP#Q)bw)4+WY6IrltsmO z$3##Fz;;{~_fM8*yzSZVYb`Ry-YOzLN0P=>WYrbMC9rkMq~O_O+u<7HImZ>R$DfX!<#-Wdb4vj?&&;u3wG0;cw;?<>6@xgRy zLc{5yl<*CO55(h53=M?e8`>{EOj7>>O5K;k8P2zh#~pa~P79k7uKsAPuJ(~xhkxbN z1%+Ia|L;#<7n=(OMljVjD*A8H>y^g^4&#{-V{#I<_J$;*ddm$C)4$<7A|joh_(9hBy{o%ZHTZ#2lGo?&Q_3r9ss&w>i;uk31 zMnpW>aRmwp4ic@41BebslF;KY9yWUMY`urv$JFCbd#wEc;QH>*6r`EYL;ZI` zb9cH0`g42wh79kLdm_d`FVFAnP2x?qu`pQQ>W`~4@|P8U@1gKIDGW@rbDQ|}oG*x@ zG?KCdJXtSYH7rvm(dX$EK=XoviBe~Y0wrplu!R<-v{;OGOlp#l?IvkEe*PDU=#%1B z?vfMw_!|^)KQX{~^Di4B-h7g+%U}(o_^qn-!(S>uL*?UU>3|$Zy1;=bs^G6c zh@3TYfST}%UDvnW*`zW~^vk(ezsiGLyHVvE&~vi_W%*il^!570(&)8LrCAvVS5I27 z7r{%mcNV{mqC{UV>$syuRgHtvfvrx^t8SA z)KG6anVpS97hg{kxNP-exdPHpDhyUUadZr~azAnP z&=77qn_p{R56{b3UN|{EDqF69GvPW@?o$-&_UaA^ zNetRH<8Mv&Wn)GNHF4(iH9rB@oA?^$X2UY-%`^QNq2mxps>S0?no6agIl#}rhE8w z4GgCWT6N+;;yHPsEJ%*-)NBVA%LUFcH#`pV7`<;F${al49|0j8Xj~#DK%?X^9H7K~ z+wJ6VZu)qAUE#fZj{okRZ>@7_-j;9Q7vAZ&#pM;Uk&LyjF*qJR_Vlq0W-v+^5iVPbn zbeF}E7)&zAiof7Iyj>GkO9~@hSPRzi-tvG+RssV3jdIn!id1R#iZ|-j6fIV0h(R5& zh}(IKYls*WpY3UY7UA?=nPYap@jwUZdG-X)KT~9(zkZ#$aU4Q`w~6dcP8*;28u-x} zSxq-K}*Lbg=KR;ZhKc@*sH69FcQ4<$6>Ue3h4TncTY+e3{uynQ| z?v^kv-wv=lRNO3*968PP>b7^rKdx^*>mAsAcsor1x5#|%JgM~Tp6#b~kg&TEHdD4f zFQUE^Rl_-u%q2hOV@(>*6mt*>Ird$s8va2t9c*hBZ_Dk$rGYHEFJ@ttL>$Q{U^JDN zcbsJk^nHl+jnznt>|zcVHMSHUsDDfN*zK+Pz#j4}a99M0HO=xVB((dXRD+A}a5b&X zsKNQRB2RBHCx+6WEqL)lQ@82JAjHV4vvV3_{7x>p*^dtgOC+6QjjocW8GoG(Q*eNX z(^6|`+2bg@nFOZV>Kktqr0I<%sykHzW2v&&ETv#qOcC$PA!%|l*x*I+A^J*Tl_yX3 zGlW3EyGpBV!SLFTm=!sKvD@eUXeeZYV&&e-xlE@{vjA8rr_}11lqn77g>!y=iG$3@ ztyzJv@VkhOrJ-!%x?r_uK^X0wU+Z2{G4-`6;mtGew!P! z!SCFOe$B`WYe_Z_W_Gv_sC`)O&t8Od7iOT;yrpDX+3HEgdZSFS&K6CvUZ#BYW~utV z;qnlhs3XqfnveaC>@xAHVK1cqOqcW}1FG}sY2~gF8GtJL*2Gm z^LlGWXfy7}TrHj5L+i1gzSIdN6T(M85hN*T%Ve4g0~)tC{cypEpB~=M0Xn~#`j!or zt#PwIzZ%RGLVtwAH+eoq_h)@Wj0C=nuJCp z!;wyIOzQ7mTo6CsC7Aah%rQ+@L`0kCtmV-b#_XxLqm@WdWF<76KQ+j;LGTKU)~Oac z@C=&ZAeSrdyGEJ;5AmWZpTm8RE{^o@b<4NIf^s=s^jY<hllOt&`rP~xE={G_Q(-hM+sF$ z?kv|2{w0pkcjhFFp(EsEMMKhmM+F`izwTVoBpT1K*R$=iS5-PN%D_WrcSkiyuyi;T zRcy$e^vuk>i#s!ej{J}q-Y+jwNPzScz|6l(Ap z-aoXGeDQQHvdr>%F05^?P7;kOhM|vHTvCU&sH#HWoff~B*ZP`5Ini&L?dK_K$x7Xa z2$AM(TJR=Hyx$O0cmA`DqrX;ixzVxkB+E8J`%hu_&p?fwkwOd2J=cfi7b0XclqcDI zb#i-&^?kpG1&f6$`w1u>Vw(Iza8gAW;0=U2RdSsg`Bocx_!{$48*2QS-ocZy_#h&F ztr;!N7L#e7&6MCX_TuUXyi*z@k)|~?90ftCD?cZD<&!FnnNYd8h>vgmgZt+z2L??- zADM0&QlWf-`j;ZKojD-C1dEqSZ4T?dQT+&34NfhIUvrY_TqWmOfo-FIXfI7JAC<@F zQY!eP0HJnv(i7+EG1h0z%dqyuTFt%L+7iG%(HQz0A7IP&I|VI=jqW*9yjWa@bDiub z?)UrcCi(ioo3ocQN1E@BbV9T*N#Y4b@_%xOo?@RsW zPc{DqJ$erndI;{?_IVdJ!PYfa`*KR@+}|0T`}dAmetR}2S7nK92*eUV&9Ywt`IP8c zj0Q`85n^C>;Dk;xSjJn&Kx{ehLm@8wJy9X8XO$y^Gm%MP{ROYH@VF+9_fcMNtzX_% zjEZ_WI}{vpWB-c#>0uk0FA_p}P#aS9bLW;y{g0S2Qh|G%N5&Zb_-Qt)uiOG#PPuex zZX<8(5?2>g^7o&A$`y1qlW`=xM-mFjG<@W$)uP6n2QfGt&~wj?XdgeOo@#{64LRj; zTQ0F1Mi8>!al>A|0QINEhDBh!Q9gSsM{wcg)`=V~n|<#IH@ zyhCa0p$nG_dmIOAr>e^Xfv&ljJ3|R;o=;uPU%hpFuXj{(LCrBY<1f&5H>F^sNoLvt zG?ic4I}7a!VoH)|lNrtzWSNwHpR756*X0}ch|^zw{2hqCj^N31Q%ABfoGT?SzGv!Y zove1mZRaZn5`MEqZiTKe8+1BdX_PU*w|U7bldoxP2p9iaQKE6n6rf6vPJc3Jx@Gc0 ze(xfRi^NdHvh3J@v%}Sjg5%)z5MWPO>~;3A57Q#TBC(z6Ruw70McxMc|}Jw)lr=MM4*OxuQWZCYiu0TGyTwRHXY0Tgq4@9 z^}D#Im3yhy9k&ogwYl|blc@j5c9gdZvle$uuaA3e_jBC`sP}#I+ZI!K!Ge1Gq-&an z=g1o=QU{bdqc-&3Q3c`^vmu1cUo0}Xdb>4b@=l!Wt+mhHHkME-!IAVz!K7FW0vwR{FNlHq3MCCweT%U|YERL`pUm7)T_$LO3xxsY z0ZblY+d3|tW{1UMr?;@|(?*Esu&6*(dgxN1LUOeY5(r;ckMwE^&PWh~krRhWZ>La$Gg|=`9&GU!v*g3+JZ|fIH z_gM>{uPFYH7XQ=me}A~)C}eh^5siEY8f=O5!UMTJf%nmY&ZVzYw4X-bhOLuu&B01C z(sM1B)^pvoJpcFZ|Myc~djsR0z~Julm|&X)URV|0SmX>i_!}|MNqb7FQvYGG^ao3HT-I&5JJ_*(hn>f&M-J z+>ZYo*@;2I`v9BgR+q#N=eauPhyT9spA`Q4DF+nk;$pn}#@hdc{2q9Q{_7v+TK`k* z|Eu5s+gzZdJf{JRdMlCulJb`QcmIE;^}lE9|G3TD>R(s2_-iyT7VI`ClsWe>4K<)f6DyC}Y1Gci=K9VxSqNfUgYn z|M!vqM~uz^|KO@PC?d z{Ws_vs@ugHK>FR1fJWZl?koPkn-TgaTaS87UR2<^b8+EdRCzTH<^Sj0^sD}pKp9gY z*1x6k4(OT`huyzL@E@W6Uq1TN{@o%}78>ydxDFlY<~7&<_4;BPfNK{(tI^Q`$|-9_^3?q>}uB-54pP^=U)03Nl7E#^#YANot?X<8}D-FjVI8`BGbOl$(!_Prlc_ zpAs!8E0TBc`0n&R`+n%?e%%WR%j(qk>G8}{I?~bdTYJ@GAGv2@+Y=mO38LJ31mfFd z9=AVth0w^1JmAo}?A`>Ew1Qa~6RXnBy;R#=Vy^tWk(4Uk0_Woah&IjU*{1r^3XA|wB~R&Dt-m89AP5Wh~F|E zy?YJO8`wGhuk|_!cu4r5yf_B)0Icqz)+qmcnKH`VCGE8wq_|mN3H@8Z zIOxf4N*K2D8G-RY?ST(jKAUVgUsimKvvKGxo?@9azTlroyD6l3A%Cs_Mw5vP)8!lvGw`5%T8auX1wfg?CX8L&eU!JXKZ$NZ$S<6!eBGDLd2#&M zRQmr`AN%h_GF070UiQU&YHy%(Z7Q4sb+p3c*W_MhF-7;6805l%PQiQ;+k>)8d zb6l1!(jGM0&z=cgNQ;Z8TzMyOnPxy*LYeu;rVFjJ{_Rw@<05E0239LV^XC-_EC!`{ zIO&5ugz<@-7A(LX&t_7*u;4me)^@@r6DOpf>UPJI*HX?+f&Prx5hI*@4`h?Anf4Q- zsla|oZa2WRjD_ws&*)pLc5g#&jLcfI$K|k8mZ>X#e6D@v_3Yn0a!0#R&?=oIx%<2D z3$Ntg3$dSEfWi#7WtABbwU?+PedpC0eegm$;&1v&B0QX#Cn z58`d{_-Q{%!#>_1?jq6Gh=U8W?V8CjI05L*zjW_nSLg{IX}$%RL7cz=UlOnfs;izj zohAOTyz2tGrIR!dM9=cVRcSn;(b2-3B46SzaM5U?HLd}XXp*a*#mj1dZ!jERHQx+m z`v-@h@Ou2n3CVTd;?30$#y?@do73@}xL+q}^o&i6OeBHSrb%Vhv%_jU`DZw)=n^`1~CDTjfK&9WJ4p#pDdh*7;tw;Qz< zofWp#7_DMn>VUB+U(sAEmmK!wew%XHjn|6^aalu1tYv}7&Dj9TTesOttv1%cLNE7r zb!E;mB2LF$&*NzedvCtAvchD+?B3%xY!uff=ph+A+cGkp(}1PaXMvOgv!mDR<%9HQ zM@y@^??CI^_degI4yW;q*234RP4~4M*bZ*4GCO~xC>OuzO7lb*A95WKIWZ*-&I492 z3TH_c@Je&`5f-5ZzZpx>j}W*aLhBn9K#jKAXMpH%|K}w;DW(f`uab_k!@)qQL&1BW zySTIfN%<+UN+4T~IJRX@`oR7sd-2<{O9B;`><5oLV`n&VEayVAoR5*?=4XaW%aWB& zcc--Zv!yH~j41j((j$4HCo`0gr-v)cWt&g)ZK`y(1E2T!cF5eer5TEC*(R5rwh`U1 z$%oHFBVCaKq*M6S<9T5uzrG6WWXJ3E-|A20<<5C2D)uq8T~kekA`f5SAzNs>Vuw?H zqNqoq#A9}b5|_C}>6IrNiM@CjVs7fkL(xL(oLdZEyei$ZOnL(v{>c4f%lWu&&4c}< z=@t42)z+P9DGT5PL61Y(pD2EQpW<2>0p!)Ae#4z|Pf?iR#fMWV&}`$eJV_CC_iC7p zgxfq=6AP{EU5Ak0O>(U}!aq+pEi&)&<)qd0_-d{wd}>o=EZ0$0#&uah{!Bnuf4X5E z-oY*Lo6+fvv(T}rK>4W6`tDqC8f&5`yHAr%60lDLu?EOJ)h-3vWZ*a)+!#dyzf&iyGyyep_fV1x%E zDJ9xgQAVw*Atotz!~g09uZ2 zYGW?uv2t&haQp$V=QH{BA`as$rY5;V8d~29z zywsp0d(~{zlZ>E)eCL^xuwSvMU0aiM!c$H zm>`P_!m)4{4qcZ^s?+#*a#Y%yT%8cQ)N0L?(uf@~@$An#^y+JJ=Lce>JYi1DmJZy9 zNAiT(H96XwcDP&hcojP>$zWnOoA zu*RX+)=eK7>l`s)AjdpG2>6yR#p6C^2C)goiHrNb_VDF*zC73)!K&Q%?vVlDp9Mn) zkEWsO)A_(8*&}XI4h~knw#QMyE*-aqu1(dNLI3O%&Bcl2Ue416A*%IFtOO&e9l>?3 z-*AJ>4ON>0Ul^#^P`F*&NrcAMp$F5BTpYQtz5eSKmURv7Pvgqt)B?p>)&vT$p)8pQ zOdZ2DvAoe~V~2Z~@63Y2{6;V6S&07zz35c!sG#`|FLFfM8IB#G~S zzTFjj&3Ea^)m=r>(T$HM8I&R=}0ev*>hd-{-O7+6ctkTqSB<=4&(s9)mee; zbIOLC0`a{-M0gER{#3qLSP{9JvmL%Zwv^EH<3(PYT*72tV&OOXCCenu^Te-dp?lGj zGpx8d49)VX9yc_~nNF;>Q&3f&y>&^mr5Nd}vFhNi;SY4aMk8F%M2AX?y-m&rs~cq7 zx9Y4E8>_W2oM>h%^1ypE7%pQcpi_ar9`p>pr2V3eB>nXg0A2HykKRxrn=T5T^Lx~u zysV^k8FG_3Tb&yCQZ(4Jmy}8OI%7`-uw7Ld&LRSgC!WPw2{cs)6F;a!VjXt{DF_hR z5UpJ9RhKb-s3&tjBZTC88`Ax%F-%qHAv=#PLfMJg!i&OF_kfX4{vmkvL{7tTXx`|L zQ9n`jVmV+KXNL-lyaTgXC-AvfYhR19gA6;R0Jp}q&gh>-uYjD1J5M4m*6eS5y?>J; zz&+s=4I}2D??sUmp#UHLrnQ;K)PK=t^*IbEaq$fKRGF%?6Kt>1(-H*W1F}rS5wsci zXS+evSak+nv|!qBVlz&U7Dn0LaIkQx`QW0*ir;hOo}#lPJ^wkYo~cGX%f1#OKLk6i zpGE2*udIa?}gq=E(uv_mJYqhVF_Nv(I|o zRuHqi+q~&6YhWn+@ylf!o$i9^VBDS}+O$oaATZYs%D^;|PavTZ_}t;Pb*S0^;+26+ z|C>-8scERQ={3_tGe2fK&EmxMs(UoLbXoQr(2sn5CT7{p4$D7QPZ&xB1OFX0oaEzC-uIH(OlH zs4kqVnw(1Ead4=gRFtO(gig7T^G?g>&8iJs)05Gb;Z%hjzU?k?UIO`hOG-FV09T1B zy(Hp$8TH4{Mv$1*`vGduJ6?*SD>CJR5fj zx*@pB4iX5uagE`w8+Uhi_n;w!gy8P(F2UVNg1c-Wz4GsKZr$ptuDTE1Z-#SC-egis`jZ(3NpW{O*)1(%__m5e%>YVCTto%V``sE7AI7-Y2(`;6CFu& zb?$0{w+03n-NoG392;XmDakN*ytKLUf25z!(3Q`w%$d+ok-<@CEvw>qC zy%+QOjDfb0NP=CAhYgZkXq4bs{;_SHTQ^To&RkvuZ-(4ihtF~MAGG5-zP9UlsqXCd zp6NPnn@TUuA{jEk=PVTo}g>F=TqN2 zfjV&3q#O9Zl%@1*Wbj18qOzW|Uw}0@c5K1VcQp)z(+Iq?V44ay?Cehxh-&7+Y~ zkDWn=-U;KVoa3#a#=*0hCxx=>Up1sjMd#6u6iFru(A>M1Avdh;M`rTYG|=5^u6Evv z*7k8vfgvxjdDgp9tirFko?ZRHO=ry|rL(<8J62W{ea1m890Eop{UXe>Lg*>K?3&K3 zsu9>t;)Jg?fu||nyyeB$yhRU+i0EYIUDT=5!N&_rGDtnudWM_(wJL;uU$E3~ywCy_ z{ld>JI^v{cImvZaO0`#)LifkP&t=R5#>R+O*`O1>bzl2>pZe9(FL%-NA|PLhK^*rH z52lW@>z28#ePH^LINNvTf+IX81qqKA$dmS;*VBs)`I&e}&;I?`|I`tfDU7%hq~IGW zyZ7yq>Kx##5P~rNJRf^p)4!4EwnyG0+Z|4;-wzk0XcWLZ0P*DFh~9#QkEJj5))e+R z3HN-itmnBcay*;$)@_PjqOE+~r0R2Sb1{JoKJC;$^rO*|bEmi6b}xYUDqWoNNbff- zE-bWZwwk-nv3EH38XChgxH|h@?8R2*`}mA$3yD|=uhwPg)`-+xOY^zzNq#-z7D$1C z2$gLWfg?*&IzJyDI|E#AznZ~4T)*_bEK2}Tk7k5rD3%TIGhP{Ge4WsRu}L6|7aey# z(_5!l-G%bo;PjhmfWSYTQ-H=#{$d;uKsN zs1mc^^qtFz6ybM+`a7c~vihN$97(qqu12e-ghvyA@U<2?Z98LoZ`S{qhITBJYl-oL z=|G9)IojHYnDg5PN>TVO{7Rc{tow_#n^b7P-ARQ4mR0=5qcdP>!nDNX>-Q08?CO9! z&H~=mjcNB{BdR}V1rv##`q|CFZ%z^~ZWtrEL(Y7%&|TC1`&!>-^m7j=GN_!V2mbLz zdw$hVt@Wa^e2xXV<7x($?f>u9?@#;(d^(fQ{ z%q>cRb(rF)NBa)Bau0Y2Eo64@js|ajBGV6Qw%Vi`OnviVucy%!9ca&bSj&7vJiUM} z30d)2O@$a(8+&_ZB3;HYM5EDbQg)8v29-NU?I%;Izo2v=+r?)m3;ObYdgYgiO@RJP z*?jHiikE>cN5Hp&J$j?7fe?L+g-ZDxP;>+fL+9PGd^ac>O`nL^$kyfP@4(doP5$yI z``nRec(*Hdy!2`GUi8vF-V|y%e64r&~!Y%ozn`zk#Ygz&E9u2=fhoRLi9=x z7M-G$*ZuxGn%=Iqib&h!5mK!x!)?My(-6mE6vCPKVyNzEZ!MV6;G-vyf zaPI->D2`$=4`nN{DFv?6|9Ea+RZuAte`_ATCHZKcbElSX08*93IyOU%)hFI4!XP2` zCmb4Rw)hC^ z^bX%1p|{p^xw-wM{%P=p((H+z4w6S0Qq-DToxO7XV*gGUNfJRp4a~AKG_mftoQ6HL zx;}fnW!iNcOuVxV-;stmj8L;}z~NV2r8Dcwh{(lwh5BwyXeY z#Gm?GDh>*Oi0BItwXyI%ZUm1tAGFZG zvFoU4V`;SffFTne5_!UuRsWG@aR1Jd-evoh!@*>}r$tHXu$obI{rTGSxD9GTchKea z;SeMC0Abnsnb^~9bVL}AZr|O}TvKY(_oXjxV#bN5#CK}zM!ttVRLHvC?3%6JR!xGb zuwY+|V7!0>0!=Zngg8j7OAZFq(MJXSULtqI?H1P_ue1l#7A2M-1__LejK>6QTm@Za zUD4<~D5aW2m6<%CnQOg!?yGd-6;+F*S+sG|Xn9!75XzCem_bRk zZi1~mMBI?kcWLClBe5Ow7hhJb&gSBWYKJB!DLZS`sn@k92K{T_z#sG^ z<*$!zjt)%%<1SpSD}+(st|UOET7=<3NTlu?`eFr$nBF2x?Af7V-0aUc(hxr8_^6I& zw=7sWo^gMfTvDCv?f!0x60zdP+R?fZFY9-Tl!ZDOp;svof!6dbw4QQ}DZ8+9=h~}` zik8OuNMSy%lHivxb971nW9rFPd0SXU89kYN$|?0PM+*vBP&$TI`Zrup5zrJk)b+f5 z$8hsjW#ct&7Yzzagw#grt>2~zP|3(n!}CQgUu}s>28F)sA;A}Tw_?}24s6X*rV&~( z;sQhuUB^3yBE|4Mq4ho+CyA*aKTuJL?DGSV*(IL-NC@)M;xSHvBT0I7f6a<*ktpT1 z%+~`S8P-z!__(l+c%QuBknV<$+ca{UZ7s_63-aoXzw8F=4KR4uef@)*tb3t4wZJrb zh{hA@iQr5dmp2u0ObcUvQ8h_*@>25z@=d3wpR%*z**g1eWgehQzFSi-5c_pGG=a8f zj{Q;g*uv72c3lE9y%bVHqu;X{cHUxq=F+{zy{sPlisSR1;xSHsVc}qY*Agud%t0Qs@1!p zP1f*egatn_)v^F_J3;c&6m;fQXYWeHVHn%FA8P-Ilg?HTc+RH1z&CKF2}~%M{dN0S z>&PZg>FG5K)cn*E8a~BioiT?+13z^^LUp%@=wJ`VN+R*@nrP3CTG;pphnP}Z{zS7O zK=?5$Q&_yB`R#>|jmw#5-C>`RED5I`R&+>zHiCq+cC2nFu9NLS9RklT3gB-iUh{4A z$7Qsp?j`J5Ih|YLXu?RmRT2d-_r=P2c6Gy=iDW7&Jjn(TXqvVNCWXPcq6_7Bl~f@x z1kO(42_YuD@Jf|^`0!&JhBHV1Jn42PB{!jbWSCO^_rF64apu3DfIxXv_P4)Z{xYpo z7sNGWm@Yy`RkMZSGv|X&3k*~;G=HMg_S?GZEKAdKvF`?z1D*}@y_7d0O0*?8p!Ko! z8t$;kyQq;Degdo7g~7~ni=m`YC7(PGtFp|O_K`nIFL%d-h(seo zJI|$mIFalunrqIbFcaGT(phcW*yVqFr+g)1X28BMRBLZ650odX%Pl|KtrkBu2rqNq zFBBj?tBN?};~t?+OnAmoWOKj2&qZeDah5F)Y*qpig-@!t*Wu!bk$FNkih)@(E-R8y zN)+tFXbSU7WH0vu^3c*~wqh1?+5yIVAt%iGMPp-kr>7L!hPTNB*j3&}+TR96cj{-b zE(zP$y)NpCKGfXTlMhq@9XK)ylH#5!<8D|mjD2w~PcYyS%lV&=ws5Fq$5t00t*NeLct3i(nEu23I&>yiFzr-WZ{S&KPkJ=44xMg zV866pF)S}U!MP7&!)3%ddyUp~^^@=qQ%lwIdCLfW#!NL&Op?QYDx?g-vDMbom+q;A0)p_yLg#x{uwk}6{?yykBV*y(o{%YUX zcbCpY0$ciD*SYyMCrLtXi&c^K{`xCdB_)gi|72h|!lrP@sT|(Vt<_=e4e%Z~fkXWg z#amaOPyjHc32!!Y<_a%PJ;ceT6!PLGL6Ol&OjVF*nOF2Un@+kiG)0oWC0^KjPnV0m z1)r#6I1wv*YP7FUfryAj8PE5}Tu*e^P+Bbr89aT}6X278%pRqKO0Nx9j~#n`Qy%ki z`4hn$2Ijs}Ww!@cw5Z=r{^9aTtNs4wf=rM%v@u`GsRe9OqBe?%hMTw1t0d`P*}P+4 zK{|zi%HDV>A?}fR`j{I|CL*NXpuFsI;gT~9MV=33W3jA@vS(V$0Xrscu-0OhH6*N@ z_}?_Jbl_pTRSNkB3_N6RAL`>31e&~1B6~epH4`xMsx&LO@cQPCEasef0-nZP|%97p&0GV^Vnx;Q8WwW zIC!>-lMWkXykTv7=Ty;s^S&q&t@i>b6$gW)%#q=6V;?yX|Y9b=%i}+1k{V{m*{xKHvXd1M4*G9h*e){wA ziY8N{$56twl+jCH2NFOZAZAE^E3ntsI+ib{)XXb{`PQPV)KR@8DKv#0lyah0e)1H$ zf!V&K<-lXQ19jr7>CziL4mR|F^|fZ0fFaG-)K)J!O>n|8@IwGGM6}r~5CJn+i+lKn zarhSfzAyl^+j{K3(~>-MxMMQOel%&D|5Yt~q|u*n^;vbRP6pvjfkgA>uq{^>zyZ0! zMtbc|-V{ieB;WjLh-eCoE}1|LU{Ck32ql%>2D&mm{JYm*8Q3QnfXO||5xO9yHQm(vB7dIL#so86iK54qmZ$2dvQSNE63!+!_fe%zRnwD;DH@5sw^7{C49}a==DF!OFcRBtL>om(d>DMh zKM)r~MG$Vx7CYD3-Y<|zOJ9!Rh#+6#I@Tn7s};<#_Qn9CdeaX31Sj)Vg7hS|(9+CN z%jHEK<1U=@d1y4Ofc8o#l9J2c*^f>-ld;U%WNnalVH^|iT*FE!Rz995R+;KBHq$JNXE#CPT=KbtA&jP?2r|u|G?>wGPb)Ayt6||+~DBH?kZz0Zn)&$AvLy;R! zuGsRtxo$+iy@*5;QoXdo^|a}WvE`zZ80Tt>`>fwAKQ0~I3rlzee3z#$kEat?u=KTN z`TH&+_&LbcN|UV%=1hz#`sll+8qLzl+`G?L-ma<_(0~afqRW?|OH! zI)%r6LyH`ok01ko8&~+S+WmkS}4c=Sj&7c08|wiDim~8?ssO$` zm#bLOp$2}dIni%$L>Y#Bvg+*M*wjZ9Hagd&F2b1?E)AK(NXx4-3BK}>fZp>9MO?a3 z+fPB)yONH4nNc@C1?c1&c+AeDVB!N!i6J_pR0~341x}nzN?EDoK6)9f-vTW^h0~R0*U^JK+JE7CGW@4?Im-JZ)Rd(KFRo-;K5^{WA11_BdMmDv{R)6P@y$}?S( z4!oyaxphJnv4_*-uO$Ma;~nlReuP>OUE|Ms z_lL94BX4SxFh#XST$RhVG^BGSEeLm*z`EM%iVULXVnN8oi;l&)C^QMrwxm|kAv|og z6bZ`mWCbE7q!Ymdr)Nhd)MGrQRsvR}(sDdg;M8IdrHDUO=WB9UU+N}0j2Tt)>~Xuw zR0rzE4(sP%vmM;wW8bP9IDhzXAVbp2-5=)y>04q|*_K9fJTR+cj0{sZ9b)6?vz4Bw z!C{RTib!wKV&Pkyk3?%qGkuGw&J^d$qUAMH?1{-Tw^J_EqaS_dL!3@Kc;%>9F5UuI zhl1oW0qaobn)qK6<@8b1vO(1YjCHC~;F$ZUCLi1hMYitWY%NMqOr74*mQKPrJeN+w zZ3SUQh3e#|l#JyqBh^0T<3}RE{F1F3DwJpX3kkYPXuyF8x-UE#VCp^CNiiiGA8-og z{V@n4==J<69eVY`L(T090v%n_Zb8&`oB=plHxOng{qz;lhZSwT=goeB5RL98m0Hxm6<;Qej%*QEixKQtG zp!(@G=9{X<96{Gy2{kEt7Qj}Yts~zpRFxeIWKS{`IbW?GOTeDZ%TAy7>aDfKTd!1D z$U5tPsmbPFT7BNsuwvG;fGY_ZbVll$E!KOd7&hn~tA{IRBu=({VI@Ui2M+xGRtfZ6 za88fur|(8NX}>7Sd1rjE;kIzd3I6od4yVb~iu~N~0b8C#mI4U^3NGLZv`mK^-F?Wj z>t8mU-zo9I&P6m+0oxDCUHq6W-Sp%;-YC8(^6Vk$Rb;0*;cPV~Lsw}&e!ITD^AYHA zIz;X1Q+esjxR!lkg3z{S-K&K>KXX3V8oCPWpQ6TFFmFtxqG-;An<^cdJbi-* zuoKXAU8RY`HQgJA;MmLyf^4kd_((J{$$&ftdAT|}{k#;x@;c=>GH3Uc5x(Q_eRKUI zeg84y9kgnF&Nt(!x*huTXn)57O-L>pv)}aUo6`>Xw8X8RE?P9GaZR#_IVaT_ImjIh z9Cbq&v8^+O?OKGQ-v4!oIkA0+mPw4(Td?LSK6KMrRgX*6CKGg`P=Ig~+?N0q-W|^0 zBWy78@Xr)ftLXa&PM45SuD0{AS!%DNSodtyO!kt$e#eb?1VU^BpSwD}htKq?WM9I_ ztc!;u*;Q*LaDXvZ)>F=8;_Jo5Ie`kI0tq(K_OZ=y4K$5tNQ&(;rGRI>MGGElt^MpE zzVWc+S=@p@W%ALyrKZWiQP*hlq7&!*e`wp6HLR<`SvrhI6 zJg!%@)TMw4dnG38`>+I#H(m)47DuXS`QqHKEfeoa7LNIY4#VXkg`d09)1k-?nXgli zGlA(f>f6VnoU&!GA6f>`kwEOa=u9Xa6^YQzhYC??dbq^a@qW@R&_3wk9?$o^7wt&m zVBCfddi}DK%Cf&N+?-n_Uvh%Wq#g0S2@n0G(}G*BHO_1>P7;2yd1d+mWrXBb42MrT zBqh;n_@dBN zs|a#9sexBOt1-e&V9Hkz*|LmQ2JFuU!XmhEfA@h#v%OaHgG3FG*wF8Ddf0o`f-6D+ zqOAd3RGQ}?<)M9js~IV>CpN802c&*VO+wDpx%N|5)QBfGMsC)0+D(q}303?;AKx`wfu!Xb8I48vj~TCeKJ+k=zF+2=0{8-z-> zp4WTct!49g7H-S*Pp~HLD}YRwOKnX>6IJrqHG5lr1BSj@lcc37-g-pi0(PA{2i{70 z_K|f(meGq6`cMG%l;Y=_z;*I^V{cz2Zu#~;D`zk8Szcw!+k_+6Cnu#$dh(!gd2^NL zmI_H}q68?#pdI*8`3aSPSfy7`pF->H>u)W9gmpQB9>QdtR{6)P_dUYhp33<>3N884 zlZr`Cn?t@bVIi9l@uLNcg02&t(GyO2h@76XQ$ zG6YH_WJpC_bsFYfHNj#|xz%&Dm~3luR%skyknaW@lg&}vH_`MmyPBJR1PT;#s8O)| zMsN2OYGufdcGNBq`>g1@eN2~^MrR=3p(_2y{9A$dH(ui}7>slLek=R;zT zPzr?^8oj|QtA~=Mv*`@$YCJkQ-cpid z@ku{Ny8*BG%@6*~tMY`R$t0K{;K!5$iS*4~5T;d!)c}P<;NI^OS%d9_a}0jr$YCh9EnQaIoU;yiCLQjyy7)=B0X*y{u7! zoqA%UzPW!C(k+j{rIGLv;#zvn13gK{4smD&B$mOvv!uv1hWuAMEOGKdM)@2tH*k4cF8R#ey zx&ep8+L*mAwo z_~j-+fXzH1NyCN6RLWZk3c~ABhRDU5{ZdD@HD~?eoa?02S{Y0t^w4g>8frPU(E!8b zb<_X0OFR|8Q$_Ns-C*+4+?~Z{Du9;~Wm>foGmsOdMUGdz7-+E1pF+HALlvpHk>OAj zJ^fykdBuJAg?#EAufhUbBwx2fe&%vvF#D)tj#H9FdgX7!SiM#Yyd9z&(+WiVn-{sM zNrNHMgSVxKLcXTS2woop!LRg_%Ei^}?hiwEDjuc3MiGIy^sfUxn~k6r*7KCJp)mdG z|3Eg=5Z!bd6p=ctdc`uBU1%Olp;RN2*vO_&f7WHnbAzxM z?#-3BaR-e;LDwXRh2K_jo%viVLpzm}2T}uv#B`nM z;$)Mrv(5|`XOMxg$dEfB_N1mLlk9f!ME;vS6AX4{!j@`@+$`9j>yfe2O?yv)F7GVe z=Gz}(iQpu=g3VIC5enEe*cY%uANa2xdpyXa(Zv5&QGd1n3nEDoV^Z`o?#7}(;(jV^ zK@oVQQ3<(O#o+l21+nb3JTmL7PME2b_uJOrnO>;NE{`9LXvWdb+Wsm)TgtS3b)y&^ zFlriu1`@{8%tWW=D(J@yk6`45*kUd^2P?8BNXtE=_Q=ybAmFK>|VZxuH5qj|a&_)=$v+!!fcJ)9eu#ioBF|^97!h=`AKRQ0EXY zk74A9CE-TvaH#;!#Q!|NCGW0bv)E(yt_`0Th5BIIUtX-a zEp%F3?|8^3!-+zMHMy#RrX39^jd=s(! zk(SOhyro|9+Ox}<(CyV@Eian7wcJFh?^ORx)OdXjF zWQ}~koyOWC#{u1~z0;hK16q2q5;0E9lHJ|tugiKB8`}|Id2Q5GIjN(hr9xH^B;H1} zw}@pimNg3u&kw34i>0HKf0Nc+1IMWlcf3D9;4zeTyWM46@~uhi6RhiR*<1lwr) zwAvI+XdQx9D-kLTzJ?Xc94FxQNTB8Pbzgk9`hbPBQ0{drKS0VsQW-3_eNRSEN*kL6 z8EL(=a(Q?L)1*V#A~>(Zev5$aH)Xx%h1?`}U$L6ro{%_2KhR6EjCUVTF0k4!6;QvB z5V(rXSdhH}?U7EMaINk9#DEcNgp%00b5SjQg5E+ecZO*&wXlj65FtL zK}y&fxw~jeCF?WgHZG?oGJ+DK$PIQ7)MIN&4=`@RDsqy&n{e z^ye_~1~nwp!1pnh!mbg+qV zvdBWN_3wU+2=)Y5Sm5!2)%VynfA9D!kqu^$oa+a)WIRdrCwQmZ+bB^d)X3S5IeZ2B zTGI$LAHwUyMX4%u{hE3FGxzEt8n-?^)doMfDyvp^J;wQ0=@$Jx@AJDj;I=|Ticeza zK8ZjT{cpBMl+Jme=lvq-*q*9bO)qi~nn1ZhGW)ynU2~R&TV3QXw}KR4669OcM_T_W zh1>dxpk}j2bRkp}jOL;N+9nZ+h*jL-h*0 z`}MdoeJvw{;q=xIE*JM%CrdXLfAp!!zR)jSOb@0HgBpu|Sa4NC3E1_(DZOy=V(SxH z^8<_JBmSqtA76b$@lHK{orfz$ZuN~W;6G;H1g1-;{_y-r@oVc`70|Q^7)m+dVepCm#O)-lRJM z%c=>7*7Rf$S>$f|Nr5SM1tF zsmX=6P>u18upg$4d)r^n`EQjRgtGPCA;m`IO*|Bgb6+~0Zec{u&l}0vgu3hh-*69q z`5KMiPq+qcK&ruoKnxx4Hja8t;J0v?Jy){1h)ge%xDo=1+p!`Pd0e9T<)(8|hhJJ2 z9%p=^;Z)Pfu8pEk)v`eC=DrN$tmq4MY>sccAV-#gNZZk0hCHZ9oP_8|Pk%jx$}M?p z6;A>9KEGr;_&%S|RtV*Jmq);H=Bg@<6ZSbds0&X;sQ}j2KL=0%@kGz_hCu;pXDi)h z8-njARX1t#4G{;Y6)x}|&&2ghllpwBdaGpH?d2EyaKW%U)V=3^^39aa>5c=^2_!ts z(NEyZ{r%|u4M7r;sQ_=-?~qR_G>teOX^pCvp+Jcr@-gCTrevY9HPMuqXe07Fe1-bEWOMm z!1i+w*<%9)Onuq>d_rvA(|1LpAH5&xej@-Ulz2dB@U*}qJwbjdhwt}9?TN!Ph z=Ab)bHti6lbnlEUCXx#>M}7_u_XNfzUfE@=&<*|~+Yz+FF}FgCoGThTdWjjO354QF zB^HhAnGAF>a!JEkPt5dWp+AZ|N9ux~vu8O?HTZN5Z;n@sEdJn|0|AVB<&Mr^2TRAR z)DLAkL^Gjj)4yfsG5|5ms&4v$Bv2FY=3sFO4IvHFwns!R%~m?RFa|}U7w1ePB~2M3P2VDg$iOe~ zrvexYNRQXL18~YHvM9=1k`!byQ^6U(_?Sn0YMeipVMm<$&8$Ub?|BQstZ#fZDQ0b_X^Y4}Y^~9FweJ~kXHVuV zGCseY82FfQ^@FNmb+pCNe^(fE5+C{b8Aapm5V5I@Mz}D+ZTO$x2RT#E0L%dp8%i8X zC9FT^)yj!wIN;0Vo&(;^oTe{KL*Jo_FoazWDB2a!>IcX-9!kbw2yBiXZGTjH!Ey1= zaQo1n>;{qC$(q`~+B5pEJtkP>yVR0|5-~_>Wu2Duna^K$c&Y9cnFYfGYSjfoVx$~< zhP)=r5K`sSAe~0Qsp>HatE2~7+0MRGDVHI~`^M8MbPk(Bhf(Fq;Z2@5QGS2|&xIS> zj;-->UcG|cHO|9g3y#Ic(GlW}u045tgNTxXJ1s)1zxp0ts(xksqs|0V)`*U`-ln>G z$+!d4@3`q zEJ5#$Bp~L~O0wS6N6uIXTsN;PivY(~ovrmuWAU65L(yh-jOqSg&szZI-lQ39je~F; z8{<_aIjdGdW1Y{{J;~6kg)m~1fqG@EmD*j=k~L>c_f;s|feKkeMLJGwyoPyiBLbCy zL$wt{vI-Pr+$$jTa4OX&*oMsQ{`J7stWNP;@rn>$?g%huuzI?O+cWw73;0k*`%pT% zFgv4I(Q4GM9M>w<#F_i+j7W<0zI}wBO+(DiD!~l7T|obMI$s>gsh8vDz$;9|y~Quz z*F_@q!tTaid-xn10KkX%uHiH3YUc21w6}21N69><{8f1qw5)0~1*2qLWmi}B2HUWM zczPDSaAP<8{SvA9z;OYR3Fz3ptxb~*Q6+`?;q(W)@~oBx`0Ld=vWoOG75lY%i&080 z^HWPxASI0Eq5a`+I0=YpFp={{=hm11&Gv!=pVAeizV-z_5gz_cg4Gn zd9d#gk9dy(JcnJGv4S_EI2cxY%sj-Gr1kwnN9T__V9JW&71T_iSm- z#4{e@IFD~Q{UzAK?dZO%`6Q^qUD=xyv9;&Bda)9D$RECn1UN&h25)M9p#QKugYp|X z_fLB0q)rYsDOL$PO0nX^`+!sPZi9k|%P+VT zQ1ox(HgwP%NF%zcf(IoVsWGzHhkJ#(b|*Yj$E$VppReS;F3P9GGCEUyb^v{V@IcvG3@ zw9mGrhd5^XGc>J#)lyY(mSg5}`Lu|uOCd76LBbEucwsh@ZB-V1s@t!F$za9x)6vJC z3kX5A6ahp&*XM225(6pg!(XaNgg*_QnLmv>UcX;)^x}2&1DqL1A8h}I)g>i>Hbu>F zM&_UJT0}GginjCD_&PJuf1w^$&b`!t*ZvinBnk`)==8<>x|D%OV{^NJMM;j|OB`}B z?flOh0r>wX=^^|-q=&zxhc4m&kRJX+diZ}udiby1crFGIBzFE-7ruCpa;<}s0$Z@> zYRhc|+LHeQi2RFh@$W~c=wECHe6nx)6Dty&0{HVodkx)*zZ9T<@9_VI68Yzsh6VuI zBFy*f`>zb*z(qP*5__` (as long as it is [registered](https://github.com/latticexyz/mud/blob/main/packages/world/src/modules/core/implementations/WorldRegistrationSystem.sol#L164-L201)). @@ -406,7 +414,7 @@ The easiest way to create the Solidity code is to use the MUD template: ```sh copy TABLE_ID=0x74626d6573736167696e6700000000004d657373616765730000000000000000 KEY=[`cast to-int256 2`] - RAW_RESULT=`cast call $WORLD_ADDRESS "getRecord(bytes32,bytes32[])" $TABLE_ID $KEY` + RAW_RESULT=`cast call $WORLD_ADDRESS --rpc-url http://localhost:8545 "getRecord(bytes32,bytes32[])" $TABLE_ID $KEY` cast --to-ascii ${RAW_RESULT:322:-2} ``` @@ -463,10 +471,10 @@ The easiest way to create the Solidity code is to use the MUD template: | ----: | ---------------: | --------------------------------------------------------------------------- | | 6-0 | `00000000000005` | The total length of all dynamic fields is five bytes. | | 11-7 | `0000000005` | The length of the first dynamic field is five bytes. | - | 16-12 | `0000000000` | The length of the (non existent in `Messages`) second dynamic field is zero | - | 21-17 | `0000000000` | The length of the (non existent in `Messages`) third dynamic field is zero | - | 26-22 | `0000000000` | The length of the (non existent in `Messages`) fourth dynamic field is zero | - | 31-27 | `0000000000` | The length of the (non existent in `Messages`) fifth dynamic field is zero | + | 16-12 | `0000000000` | The length of the (non-existent in `Messages`) second dynamic field is zero | + | 21-17 | `0000000000` | The length of the (non-existent in `Messages`) third dynamic field is zero | + | 26-22 | `0000000000` | The length of the (non-existent in `Messages`) fourth dynamic field is zero | + | 31-27 | `0000000000` | The length of the (non-existent in `Messages`) fifth dynamic field is zero | MUD tables can only have up to five dynamic fields because `PackedCounter` needs to fit in a 32 byte word. @@ -484,3 +492,312 @@ The easiest way to create the Solidity code is to use the MUD template: The next 320 characters are words 0-4, which are not the actual message (each word is 32 bytes, which is 64 hexadecimal digits). We also don't need the trailing newline. `${RAW_RESULT:322:-2}` removes those characters so we can use [`cast`](https://book.getfoundry.sh/reference/cast/cast-to-ascii) to get the ASCII. + +## Create an updated user interface + +Finally, we create a user interface that supports the messaging system. + +### Run the user interface + +For the user interface to run properly, it cannot require resources that the template user interface is already using, or that are not available because of the changes we made in `packages/contracts`. +Here we modify the template UI to so it will run properly. + +1. Change to the client package of the extension. + + ```sh copy + cd ../client + ``` + +1. Port 3000 is already in use by the `extendMe` application, so edit `vite.config.ts` to use a different port. + + + + ```ts filename="vite.config.ts" copy showLineNumbers {5} + import { defineConfig } from "vite"; + + export default defineConfig({ + server: { + port: 3001, + fs: { + strict: false, + }, + }, + build: { + target: "es2022", + minify: true, + sourcemap: true, + }, + }); + ``` + + + +1. Start the user interface. + + ```sh copy + pnpm vite + ``` + +1. Open the application at [http://localhost:3001](http://localhost:3001). + The client will throw errors because we didn't update the client code to work with the new contracts yet - we'll fix that in the next steps. + +1. The network configuration does not have the correct `World` address. + Get the `World` address from the application being extended. + By default, you can see this value on the application at URL [http://localhost:3000](http://localhost:3000). + Edit `../contracts/worlds.json` to set the `World` address for the chain ID (`31337`). + + Note: If you followed the directions above to create `extendMe`, the `World` address is `0x6e9474e9c83676b9a71133ff96db43e7aa0a4342`. + + ```json copy {3} + { + "31337": { + "address": "0x6e9474e9c83676b9a71133ff96db43e7aa0a4342" + } + } + ``` + +Note that while at this point you can access the user interface, the counter value is inaccurate and if you click **Increment** you get an error. +We will fix these problems now. + +### Increment and send a message + +Edit `src/mud/createSystemCalls.ts` to use the new `messaging_MessageSystem_incrementMessage` function instead of `increment`. + + + + ```ts filename="createSystemCalls.ts" copy showLineNumbers {43-44} + /* + * Create the system calls that the client can use to ask + * for changes in the World state (using the System contracts). + */ + + import { getComponentValue } from "@latticexyz/recs"; + import { ClientComponents } from "./createClientComponents"; + import { SetupNetworkResult } from "./setupNetwork"; + import { singletonEntity } from "@latticexyz/store-sync/recs"; + + export type SystemCalls = ReturnType; + + export function createSystemCalls( + /* + * The parameter list informs TypeScript that: + * + * - The first parameter is expected to be a + * SetupNetworkResult, as defined in setupNetwork.ts + * + * Out of this parameter, we only care about two fields: + * - worldContract (which comes from getContract, see + * https://github.com/latticexyz/mud/blob/main/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L63-L69). + * + * - waitForTransaction (which comes from syncToRecs, see + * https://github.com/latticexyz/mud/blob/main/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L77-L83). + * + * - From the second parameter, which is a ClientComponent, + * we only care about Counter. This parameter comes to use + * through createClientComponents.ts, but it originates in + * syncToRecs + * (https://github.com/latticexyz/mud/blob/main/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L77-L83). + */ + { worldContract, waitForTransaction }: SetupNetworkResult, + { Counter }: ClientComponents + ) { + const increment = async () => { + /* + * Because IncrementSystem + * (https://mud.dev/templates/typescript/contracts#incrementsystemsol) + * is in the root namespace, `.increment` can be called directly + * on the World contract. + */ + const tx = await worldContract.write.messaging_MessageSystem_incrementMessage( + [`message at ${Date()}`]); + await waitForTransaction(tx); + return getComponentValue(Counter, singletonEntity); + }; + + return { + increment, + }; + } + ``` + + + +Now you can click **Increment** in the user interface and create messages. +The user interface does not display those messages, but you can open the **MUD Dev Tools**, click the **Components** tab, and select **messagings:Messages** to view them. + +![The extended UI with the messages shown in the MUD Dev Tools](./extend-ui-with-msgs.png) + +### Read the counter + +We still don't have the `Counter` value available, because the definition in `mud.config.ts` is in the `messaging` namespace, not the root one. +We will now fix this. + +1. Add the `@latticexyz/store` package. + + ```ts sh copy + pnpm add @latticexyz/store@next + ``` + +1. Copy the `mud.config.ts` from the application being extended (`extendMe`) to `src/mud/counter.config.ts`. + + ```ts filename="counter.config.ts" copy + import { mudConfig } from "@latticexyz/world/register"; + + export default mudConfig({ + tables: { + Counter: { + keySchema: {}, + valueSchema: "uint32", + }, + }, + }); + ``` + +1. Edit `src/mud/setupNetwork.ts` to add a `tables` parameter with the `Counter` object to the `syncToRecs` call. + + + + ```ts filename="setupNetwork.ts" copy showLineNumbers {16,27,82} + /* + * The MUD client code is built on top of viem + * (https://viem.sh/docs/getting-started.html). + * This line imports the functions we need from it. + */ + import { + createPublicClient, + fallback, + webSocket, + http, + createWalletClient, + Hex, + parseEther, + ClientConfig, + } from "viem"; + import { createFaucetService } from "@latticexyz/services/faucet"; + import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; + + import { getNetworkConfig } from "./getNetworkConfig"; + import { world } from "./world"; + import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; + import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; + + import { Subject, share } from "rxjs"; + import { resolveConfig } from "@latticexyz/store"; + + /* + * Import our MUD config, which includes strong types for + * our tables and other config options. We use this to generate + * things like RECS components and get back strong types for them. + * + * See https://mud.dev/templates/typescript/contracts#mudconfigts + * for the source of this information. + */ + import mudConfig from "contracts/mud.config"; + import counterConfig from "./counter.config"; + + export type SetupNetworkResult = Awaited>; + + export async function setupNetwork() { + const networkConfig = await getNetworkConfig(); + + /* + * Create a viem public (read only) client + * (https://viem.sh/docs/clients/public.html) + */ + const clientOptions = { + chain: networkConfig.chain, + transport: transportObserver(fallback([webSocket(), http()])), + pollingInterval: 1000, + } as const satisfies ClientConfig; + + const publicClient = createPublicClient(clientOptions); + + /* + * Create a temporary wallet and a viem client for it + * (see https://viem.sh/docs/clients/wallet.html). + */ + const burnerAccount = createBurnerAccount(networkConfig.privateKey as Hex); + const burnerWalletClient = createWalletClient({ + ...clientOptions, + account: burnerAccount, + }); + + /* + * Create an observable for contract writes that we can + * pass into MUD dev tools for transaction observability. + */ + const write$ = new Subject(); + + /* + * Create an object for communicating with the deployed World. + */ + const worldContract = getContract({ + address: networkConfig.worldAddress as Hex, + abi: IWorldAbi, + publicClient, + walletClient: burnerWalletClient, + onWrite: (write) => write$.next(write), + }); + + /* + * Sync on-chain state into RECS and keeps our client in sync. + * Uses the MUD indexer if available, otherwise falls back + * to the viem publicClient to make RPC calls to fetch MUD + * events from the chain. + */ + const { components, latestBlock$, storedBlockLogs$, waitForTransaction } = await syncToRecs({ + world, + config: mudConfig, + tables: resolveConfig(counterConfig).tables, + address: networkConfig.worldAddress as Hex, + publicClient, + startBlock: BigInt(networkConfig.initialBlockNumber), + }); + + /* + * If there is a faucet, request (test) ETH if you have + * less than 1 ETH. Repeat every 20 seconds to ensure you don't + * run out. + */ + if (networkConfig.faucetServiceUrl) { + const address = burnerAccount.address; + console.info("[Dev Faucet]: Player address -> ", address); + + const faucet = createFaucetService(networkConfig.faucetServiceUrl); + + const requestDrip = async () => { + const balance = await publicClient.getBalance({ address }); + console.info(`[Dev Faucet]: Player balance -> ${balance}`); + const lowBalance = balance < parseEther("1"); + if (lowBalance) { + console.info("[Dev Faucet]: Balance is low, dripping funds to player"); + // Double drip + await faucet.dripDev({ address }); + await faucet.dripDev({ address }); + } + }; + + requestDrip(); + // Request a drip every 20 seconds + setInterval(requestDrip, 20000); + } + + return { + world, + components, + playerEntity: encodeEntity({ address: "address" }, { address: burnerWalletClient.account.address }), + publicClient, + walletClient: burnerWalletClient, + latestBlock$, + storedBlockLogs$, + waitForTransaction, + worldContract, + write$: write$.asObservable().pipe(share()), + }; + } + ``` + + + +Now you can click **Increment** and see the update. +Also, you can look in **Components > :Counter** and see the counter value. From b179f309272bea6f8aa8a2a8d28495745786de04 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:50:59 -0600 Subject: [PATCH 06/23] docs(guides/hello-world): how to filter the synchronization (#1998) Co-authored-by: alvarius --- docs/pages/guides/hello-world.mdx | 8 +- docs/pages/guides/hello-world/_meta.js | 1 + docs/pages/guides/hello-world/filter-sync.mdx | 229 ++++++++++++++++++ 3 files changed, 235 insertions(+), 3 deletions(-) create mode 100644 docs/pages/guides/hello-world/filter-sync.mdx diff --git a/docs/pages/guides/hello-world.mdx b/docs/pages/guides/hello-world.mdx index 71448d751f..41eb498950 100644 --- a/docs/pages/guides/hello-world.mdx +++ b/docs/pages/guides/hello-world.mdx @@ -3,10 +3,12 @@ On this page you learn how to modify a `World` before it is deployed to the blockchain. If you want to learn how to modify a `World` that is already deployed, [see the extending world page](./extending-world). -The examples on this page use [the vanilla template](../templates/typescript/vanilla). -[See this page for installation instructions](../templates/typescript/getting-started). -Every example is written under the assumption that it is independent, and built on top of a new vanilla template. +The examples on this page use [the vanilla template](../template/typescript/vanilla). +[See this page for installation instructions](../template/typescript/getting-started). +Most examples are written under the assumption that they are independent, and built on top of a new vanilla template. +The exception is filtering data synchronization, because it needs a table that isn't a singleton. - [Add a table](hello-world/add-table). +- [Filter data synchronization](hello-world/filter-sync). - [Add a system](hello-world/add-system). - [Deploy to a blockchain](../cli/deploy). diff --git a/docs/pages/guides/hello-world/_meta.js b/docs/pages/guides/hello-world/_meta.js index 45d104aca8..a9f17b5005 100644 --- a/docs/pages/guides/hello-world/_meta.js +++ b/docs/pages/guides/hello-world/_meta.js @@ -1,5 +1,6 @@ export default { "add-table": "Add a table", + "filter-sync": "Filter data synchronization", "add-system": "Add a system", "deploy": { "title": "Deploy to a blockchain", diff --git a/docs/pages/guides/hello-world/filter-sync.mdx b/docs/pages/guides/hello-world/filter-sync.mdx new file mode 100644 index 0000000000..d96eef92b7 --- /dev/null +++ b/docs/pages/guides/hello-world/filter-sync.mdx @@ -0,0 +1,229 @@ +import { CollapseCode } from "../../../components/CollapseCode"; + +# Filter data synchronization + +In this tutorial you modify `networkSetup.ts` to filter the information you synchronize. +Filtering information this way allows you to reduce the use of network resources and makes loading times faster. + +## Setup + +To see the effects of filtering we need a table with entries to filter. To get such a table: + +1. [Create a new MUD application from the `vanilla` template](../../templates/typescript/getting-started). +1. [Run the guide to add a table](./add-table). + +## Filtering + +Edit `packages/client/src/mud/setupNetwork.ts`. + +- Import [`pad`](https://viem.sh/docs/utilities/pad.html) from viem. +- Import [`resourceToHex`](https://github.com/latticexyz/mud/blob/main/packages/common/src/resourceToHex.ts) to create resource identifiers. +- Add a `filters` field to the `syncToRecs` call. + + + +```ts filename="setupNetwork.ts" copy showLineNumbers {14-15,85-97} +/* + * The MUD client code is built on top of viem + * (https://viem.sh/docs/getting-started.html). + * This line imports the functions we need from it. + */ +import { createPublicClient, fallback, webSocket, http, createWalletClient, Hex, parseEther, ClientConfig } from "viem"; +import { createFaucetService } from "@latticexyz/services/faucet"; +import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; + +import { getNetworkConfig } from "./getNetworkConfig"; +import { world } from "./world"; +import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { resourceToHex } from "@latticexyz/common"; +import { pad } from "viem"; + +import { Subject, share } from "rxjs"; + +/* + * Import our MUD config, which includes strong types for + * our tables and other config options. We use this to generate + * things like RECS components and get back strong types for them. + * + * See https://mud.dev/templates/typescript/contracts#mudconfigts + * for the source of this information. + */ +import mudConfig from "contracts/mud.config"; + +export type SetupNetworkResult = Awaited>; + +export async function setupNetwork() { + const networkConfig = await getNetworkConfig(); + + /* + * Create a viem public (read only) client + * (https://viem.sh/docs/clients/public.html) + */ + const clientOptions = { + chain: networkConfig.chain, + transport: transportObserver(fallback([webSocket(), http()])), + pollingInterval: 1000, + } as const satisfies ClientConfig; + + const publicClient = createPublicClient(clientOptions); + + /* + * Create a temporary wallet and a viem client for it + * (see https://viem.sh/docs/clients/wallet.html). + */ + const burnerAccount = createBurnerAccount(networkConfig.privateKey as Hex); + const burnerWalletClient = createWalletClient({ + ...clientOptions, + account: burnerAccount, + }); + + /* + * Create an observable for contract writes that we can + * pass into MUD dev tools for transaction observability. + */ + const write$ = new Subject(); + + /* + * Create an object for communicating with the deployed World. + */ + const worldContract = getContract({ + address: networkConfig.worldAddress as Hex, + abi: IWorldAbi, + publicClient, + walletClient: burnerWalletClient, + onWrite: (write) => write$.next(write), + }); + + /* + * Sync on-chain state into RECS and keeps our client in sync. + * Uses the MUD indexer if available, otherwise falls back + * to the viem publicClient to make RPC calls to fetch MUD + * events from the chain. + */ + const { components, latestBlock$, storedBlockLogs$, waitForTransaction } = await syncToRecs({ + world, + config: mudConfig, + address: networkConfig.worldAddress as Hex, + publicClient, + startBlock: BigInt(networkConfig.initialBlockNumber), + filters: [ + { + tableId: resourceToHex({ type: "table", namespace: "", name: "Counter" }), + }, + { + tableId: resourceToHex({ type: "table", namespace: "", name: "History" }), + key0: pad("0x01"), + }, + { + tableId: resourceToHex({ type: "table", namespace: "", name: "History" }), + key0: pad("0x05"), + }, + ], + }); + + /* + * If there is a faucet, request (test) ETH if you have + * less than 1 ETH. Repeat every 20 seconds to ensure you don't + * run out. + */ + if (networkConfig.faucetServiceUrl) { + const address = burnerAccount.address; + console.info("[Dev Faucet]: Player address -> ", address); + + const faucet = createFaucetService(networkConfig.faucetServiceUrl); + + const requestDrip = async () => { + const balance = await publicClient.getBalance({ address }); + console.info(`[Dev Faucet]: Player balance -> ${balance}`); + const lowBalance = balance < parseEther("1"); + if (lowBalance) { + console.info("[Dev Faucet]: Balance is low, dripping funds to player"); + // Double drip + await faucet.dripDev({ address }); + await faucet.dripDev({ address }); + } + }; + + requestDrip(); + // Request a drip every 20 seconds + setInterval(requestDrip, 20000); + } + + return { + world, + components, + playerEntity: encodeEntity({ address: "address" }, { address: burnerWalletClient.account.address }), + publicClient, + walletClient: burnerWalletClient, + latestBlock$, + storedBlockLogs$, + waitForTransaction, + worldContract, + write$: write$.asObservable().pipe(share()), + }; +} +``` + + + +Click **Increment** a few times to see you only see the history for counter values 1 and 5. +You can also go to the MUD Dev Tools and see that when you select **Components > :History** it only has those lines. + +### Explanation + +The `filters` field contains a list of filters. +Only rows that match at least one line are synchronized. +Each filter is a structure that can have up to three fields, and all the fields that are specified must match a row for the filter to match. + +- `tableId`, the table ID to synchronize. + You create this value using [`resourceToHex`](https://github.com/latticexyz/mud/blob/main/packages/common/src/resourceToHex.ts), the type can be either `table`, or `offchainTable`. +- `key0`, the first key value (as a 32 byte hexadecimal string). +- `key1`, the second key value (as a 32 byte hexadecimal string). + +
+ +The filters in the code sample + +```ts + filters: [ + { + tableId: resourceToHex({ type: "table", namespace: "", name: "Counter"}), + }, +``` + +The first filter is for the `:Counter` table (`Counter` in the root namespace). +We don't specify any keys, because we want all the rows of the table. +It's a singleton so there is only one row anyway. + +```ts + { + tableId: resourceToHex({ type: "table", namespace: "", name: "History"}), + key0: pad("0x01"), + }, + { + tableId: resourceToHex({ type: "table", namespace: "", name: "History"}), + key0: pad("0x05"), + }, + ], +``` + +These two filters apply to the `:History` table. +This table has just one key, the counter value which the row documents. +We need a separate filter for every value, and here we have two we care about: `1` and `5`. + +
+ +## Limitations + +There are several limitations on filters. + +- We can only filter on these fields: + - The table ID (`tableId`) + - The first key (`key0`) + - The second key (`key1`) +- We can only filter by checking for equality. + We cannot check ranges, or get all values except for a specific one (inequality). + +Of course, once we have the data we can filter it any way we want. +The purpose of these filters is to restrict the information we get at all, either directly from the blockchain or from the indexer. From d07e2b1666e011611702feee1f577b7d9a1d7d46 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:56:13 -0600 Subject: [PATCH 07/23] docs(cli): add the configuration file (#1940) Co-authored-by: alvarius --- docs/pages/cli/config.mdx | 127 +++++++++++++++++++++++++++++++++++++- 1 file changed, 126 insertions(+), 1 deletion(-) diff --git a/docs/pages/cli/config.mdx b/docs/pages/cli/config.mdx index 46ac1e6a9a..325d0c4c98 100644 --- a/docs/pages/cli/config.mdx +++ b/docs/pages/cli/config.mdx @@ -1 +1,126 @@ -# Config +# The World config + +Certain CLI commands, such as [`mud tablegen`](/cli/tablegen) and [`mud worldgen`](/cli/worldgen) require the MUD configuration file. +This file needs to be named `mud.config.ts` and be in the same folder as your `foundry.toml` file. + +The config is used to define: + +- The tables in your project in the `tables` object of your configuration. +- The [namespace](/world/namespaces-access-control) the [`System`s](/world/systems) and tables will be deployed in. +- The `System`s in your project. + By default, the deployer will find all Solidity matching `*System.sol` (so any file ending in `System.sol`, in any folder) and deploy them as public `System`. + If you want greater control over your systems (to change their public access or their name), you can use the `systems` object in the config. +- The [modules](/world/modules) that will be installed in the `World`. + +The is an example of a `World` config: + +```tsx +import { mudConfig } from "@latticexyz/world/register"; +import { resolveTableId } from "@latticexyz/config"; + +export default mudConfig({ + excludeSystems: ["System3", "System2"], + worldContractName: "CustomWorld", + namespace: "mud", + systems: { + IncrementSystem: { + name: "increment", + openAccess: true, + }, + }, + tables: { + CounterTable: { + valueSchema: { + value: "uint32", + }, + }, + }, + deploysDirectory: "./mud-deploys", + modules: [ + { + name: "KeysWithValueModule", + root: true, + args: [resolveTableId("CounterTable")], + }, + ], +}); +``` + +## Global configuration keys + +The global configuration keys are all optional. + +- **`namespace`**: a `string`: which namespace to deploy the resources defined in the config into. + The default value is the ROOT namespace. + +- **`excludeSystems`**: an array of `string`: which systems to not deploy, even if their name ends with “System”. + +- **`worldContractName`**: a `string`: the name of a contract in your project implementing the `IWorld` interface. + Useful if you want to modify the default World implementation, but potentially dangerous. +- **`deploysDirectory`** a `string`: which folder to put the deployment artifacts into after deployment. + +- **`modules`** an array of module definitions: each module definition has a `name`, `root` (optional), and `args` key. + + - `name`: Name of the module to install. The same module can be installed multiple times. This should be the name of the contract file without `.sol` (eg: if the file is named `DopeModule.sol`, the name of the module is `DopeModule`) + + - `root`: whether to create a `root` module or not. `root` modules have access to all tables and are not bound to namespace restrictions. + + - `args`: a list of arguments to be sent to the `install` function of the module. In this array, you can use the function `resolveTableId`. This function will turn a table name from your config into its low-level ID in the World. It is useful to pass references of a table to a module. + +- **`systems`**: a record of system definitions. The keys in the record are file names without the `.sol` extension. For example, if your system is named `TestSystem.sol`, use `TestSystem` as the key. + + The value is a record of system configuration properties: + + - `fileSelector` (optional): a `string`: the file selector for the system. + - `openAccess` (optional, default `true`): a `bool`: if set to `false`, only the systems in the same namespace and the addresses or systems listed in the the `accessList` array have access. + + - `accessList` (required if openAccess is `false`): an array of `string`. Each address in the array will be granted access to this system, allowing them to call it. + +- **`tables`**: a record of tables. The keys in the record are table names. + The value is a record of [table properties](https://github.com/latticexyz/mud/blob/main/packages/store/ts/config/storeConfig.ts#L110-L135). + + - **`valueSchema`** (record): + The keys of this record are the field names of the value (which should start with a lowercase letter). + The values are strings that contain the data types of the fields. + Note that this is the sole required field, all the others are optional. + + - **`keySchema`** (record): + The keys of this record are the field names of the key (which should start with a lowercase letter). + The values are strings that contain the data types of the fields. + + The default value is: + + ```json + { "key": "bytes32" } + ``` + + For a singleton table (one that contains a single row), use `{}` as the key schema. + + - **`directory`** (string): + Directory in which to create the table. + The default is `tables`, so by default tables are created in `src/codegen/tables`. + + - **`tableIdArgument`** (bool): + Make methods accept `tableId` argument instead of it being a hardcoded constant. + The default is `false` because you can achieve the same result using [`StoreSwitch`](/store/table-libraries#storeswitch). + + - **`storeArgument`** (bool): + Include methods that accept a manual `IStore` argument. + The default is `true`. + + - **`offchainOnly`**: (bool): + Table's information is available offchain (using [events](https://docs.soliditylang.org/en/latest/contracts.html#events)), but don't store in onchain. + These tables require a lot less gas. + + - **`dataStruct`**: (bool): + Include a [data struct](https://docs.soliditylang.org/en/v0.8.23/types.html#structs) and methods for it. + Default is false for 1-column tables; true for multi-column tables. + + Sample code for using a table library with `dataStruct`: + + ```solidity + // no data struct + MyTable.set(keccak256("some.key"), 1, 12, "foo"); + // data struct + MyTable.set(keccak256("some.key"), { field1: 1, field2: 12, stringField: "foo" }); + ``` From 8f41f81f77032b6168f2a4fb8de1dd9bd59e69ec Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Mon, 8 Jan 2024 07:56:52 -0600 Subject: [PATCH 08/23] docs: claify link text (remove `here`) (#1953) --- docs/pages/guides/extending-world.mdx | 2 +- docs/pages/state-query/typescript/zustand.mdx | 3 +-- docs/pages/templates/godot.mdx | 6 +++--- docs/pages/templates/swift.mdx | 2 +- docs/pages/templates/typescript/react.mdx | 2 +- 5 files changed, 7 insertions(+), 8 deletions(-) diff --git a/docs/pages/guides/extending-world.mdx b/docs/pages/guides/extending-world.mdx index 75ede39b31..31748bda84 100644 --- a/docs/pages/guides/extending-world.mdx +++ b/docs/pages/guides/extending-world.mdx @@ -8,7 +8,7 @@ If you want to learn how to modify a `World` before it is deployed, [see the hel ## The sample program To learn how to extend a `World`, we will extend the `Counter` example to allow users to leave a message while incrementing the counter. -The steps to create the example `World` that we'll extend [are here](/templates/typescript/getting-started). +[Follow these steps to create the example `World` that we'll extend](/templates/typescript/getting-started). You can either do it now, or wait until after you've created the extension resources. To extend the `World` with the message functionality requires adding several resources: diff --git a/docs/pages/state-query/typescript/zustand.mdx b/docs/pages/state-query/typescript/zustand.mdx index 50a4e4af89..f27bbcf564 100644 --- a/docs/pages/state-query/typescript/zustand.mdx +++ b/docs/pages/state-query/typescript/zustand.mdx @@ -457,8 +457,7 @@ To read data you use `useStore.getState()`. It supports the same functions as th 1. Browse to the application. -1. Open the console. - [Here are in the instructions on how to do it in Chrome](https://developer.chrome.com/docs/devtools/console/javascript/). +1. [Open the console (see Chrome instructions)](https://developer.chrome.com/docs/devtools/console/javascript/). 1. Run this command to call `zustand` which we defined in `index.ts`: diff --git a/docs/pages/templates/godot.mdx b/docs/pages/templates/godot.mdx index ea5056e8a4..af34f2a13e 100644 --- a/docs/pages/templates/godot.mdx +++ b/docs/pages/templates/godot.mdx @@ -2,8 +2,8 @@ import Disclaimer from "./disclaimer.mdx"; # Godot -[Godot](https://docs.godotengine.org/en/stable/) is a free open source game engine. -[The integration is here](https://github.com/Digital-Avatars-and-Robotics/MUDxGodot). -If you have any question, you can [open an issue in the repository](https://github.com/Digital-Avatars-and-Robotics/MUDxGodot/issues). +[Godot](https://docs.godotengine.org/en/stable/) is a free open-source game engine. +[You can see the source code here](https://github.com/Digital-Avatars-and-Robotics/MUDxGodot). +If you have any questions, you can [open an issue in the repository](https://github.com/Digital-Avatars-and-Robotics/MUDxGodot/issues). diff --git a/docs/pages/templates/swift.mdx b/docs/pages/templates/swift.mdx index f235e5e01a..ce01285340 100644 --- a/docs/pages/templates/swift.mdx +++ b/docs/pages/templates/swift.mdx @@ -3,7 +3,7 @@ import Disclaimer from "./disclaimer.mdx"; # Swift [Swift](https://developer.apple.com/swift/) is a programming language used mainly on Apple platforms (MacOS and iOS). -[The integration is here](https://github.com/Geo-Web-Project/mud-world-base). +[See here for the Swift integration source code](https://github.com/Geo-Web-Project/mud-world-base). If you have any questions, you can ask them [in the Geo Web Discord](https://discord.com/invite/reXgPru7ck). diff --git a/docs/pages/templates/typescript/react.mdx b/docs/pages/templates/typescript/react.mdx index 4e82e045c4..fd4e5a4b77 100644 --- a/docs/pages/templates/typescript/react.mdx +++ b/docs/pages/templates/typescript/react.mdx @@ -1,6 +1,6 @@ # React Note: This page is for the new React template, which includes a todo list application. -The old React template with the counter [is explained here](./react-ecs). +[Here is the old React template with the counter](./react-ecs). **Coming soon** From 9f8b84e733412323103fdd81067f8edc9d681a17 Mon Sep 17 00:00:00 2001 From: yonada Date: Wed, 10 Jan 2024 17:58:54 +0000 Subject: [PATCH 09/23] refactor(store): order load function arguments [N-02] (#2033) --- .changeset/pretty-toys-rescue.md | 12 +++ packages/store/gas-report.json | 102 +++++++++++------------ packages/store/src/Storage.sol | 10 +-- packages/store/src/StoreCore.sol | 22 ++--- packages/store/test/GasStorageLoad.t.sol | 18 ++-- packages/store/test/Storage.t.sol | 6 +- packages/world-modules/gas-report.json | 76 ++++++++--------- packages/world/gas-report.json | 22 ++--- 8 files changed, 140 insertions(+), 128 deletions(-) create mode 100644 .changeset/pretty-toys-rescue.md diff --git a/.changeset/pretty-toys-rescue.md b/.changeset/pretty-toys-rescue.md new file mode 100644 index 0000000000..45878e9dcc --- /dev/null +++ b/.changeset/pretty-toys-rescue.md @@ -0,0 +1,12 @@ +--- +"@latticexyz/store": patch +--- + +Aligned the order of function arguments in the `Storage` library. + +```solidity +store(uint256 storagePointer, uint256 offset, bytes memory data) +store(uint256 storagePointer, uint256 offset, uint256 length, uint256 memoryPointer) +load(uint256 storagePointer, uint256 offset, uint256 length) +load(uint256 storagePointer, uint256 offset, uint256 length, uint256 memoryPointer) +``` diff --git a/packages/store/gas-report.json b/packages/store/gas-report.json index 9f4e108ac7..7e1e3b9316 100644 --- a/packages/store/gas-report.json +++ b/packages/store/gas-report.json @@ -63,7 +63,7 @@ "file": "test/Callbacks.t.sol", "test": "testSetAndGet", "name": "Callbacks: set field", - "gasUsed": 56241 + "gasUsed": 56256 }, { "file": "test/Callbacks.t.sol", @@ -75,7 +75,7 @@ "file": "test/Callbacks.t.sol", "test": "testSetAndGet", "name": "Callbacks: push 1 element", - "gasUsed": 32499 + "gasUsed": 32514 }, { "file": "test/FieldLayout.t.sol", @@ -207,37 +207,37 @@ "file": "test/Gas.t.sol", "test": "testCompareStorageWriteMUD", "name": "MUD storage write (cold, 1 word)", - "gasUsed": 22339 + "gasUsed": 22354 }, { "file": "test/Gas.t.sol", "test": "testCompareStorageWriteMUD", "name": "MUD storage write (cold, 1 word, partial)", - "gasUsed": 22406 + "gasUsed": 22421 }, { "file": "test/Gas.t.sol", "test": "testCompareStorageWriteMUD", "name": "MUD storage write (cold, 10 words)", - "gasUsed": 199795 + "gasUsed": 199810 }, { "file": "test/Gas.t.sol", "test": "testCompareStorageWriteMUD", "name": "MUD storage write (warm, 1 word)", - "gasUsed": 339 + "gasUsed": 354 }, { "file": "test/Gas.t.sol", "test": "testCompareStorageWriteMUD", "name": "MUD storage write (warm, 1 word, partial)", - "gasUsed": 506 + "gasUsed": 521 }, { "file": "test/Gas.t.sol", "test": "testCompareStorageWriteMUD", "name": "MUD storage write (warm, 10 words)", - "gasUsed": 1795 + "gasUsed": 1810 }, { "file": "test/Gas.t.sol", @@ -303,7 +303,7 @@ "file": "test/GasStorageLoad.t.sol", "test": "testCompareStorageLoadMUD", "name": "MUD storage load field (warm, 1 word)", - "gasUsed": 245 + "gasUsed": 2300 }, { "file": "test/GasStorageLoad.t.sol", @@ -315,7 +315,7 @@ "file": "test/GasStorageLoad.t.sol", "test": "testCompareStorageLoadMUD", "name": "MUD storage load field (warm, 1 word, partial)", - "gasUsed": 378 + "gasUsed": 300 }, { "file": "test/GasStorageLoad.t.sol", @@ -369,7 +369,7 @@ "file": "test/KeyEncoding.t.sol", "test": "testRegisterAndGetFieldLayout", "name": "register KeyEncoding table", - "gasUsed": 719017 + "gasUsed": 719032 }, { "file": "test/Mixed.t.sol", @@ -381,13 +381,13 @@ "file": "test/Mixed.t.sol", "test": "testDeleteExternalCold", "name": "delete record from Mixed (external, cold)", - "gasUsed": 24357 + "gasUsed": 24372 }, { "file": "test/Mixed.t.sol", "test": "testDeleteInternalCold", "name": "delete record from Mixed (internal, cold)", - "gasUsed": 19153 + "gasUsed": 19168 }, { "file": "test/Mixed.t.sol", @@ -405,7 +405,7 @@ "file": "test/Mixed.t.sol", "test": "testSetGetDeleteExternal", "name": "delete record from Mixed (external, warm)", - "gasUsed": 8673 + "gasUsed": 8688 }, { "file": "test/Mixed.t.sol", @@ -423,7 +423,7 @@ "file": "test/Mixed.t.sol", "test": "testSetGetDeleteInternal", "name": "delete record from Mixed (internal, warm)", - "gasUsed": 7466 + "gasUsed": 7481 }, { "file": "test/PackedCounter.t.sol", @@ -603,13 +603,13 @@ "file": "test/Storage.t.sol", "test": "testStoreLoad", "name": "store 1 storage slot", - "gasUsed": 22339 + "gasUsed": 22354 }, { "file": "test/Storage.t.sol", "test": "testStoreLoad", "name": "store 34 bytes over 3 storage slots (with offset and safeTrail))", - "gasUsed": 23013 + "gasUsed": 23028 }, { "file": "test/Storage.t.sol", @@ -669,25 +669,25 @@ "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromSecondField", "name": "pop from field (cold, 1 slot, 1 uint32 item)", - "gasUsed": 18017 + "gasUsed": 18032 }, { "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromSecondField", "name": "pop from field (warm, 1 slot, 1 uint32 item)", - "gasUsed": 12024 + "gasUsed": 12039 }, { "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromThirdField", "name": "pop from field (cold, 2 slots, 10 uint32 items)", - "gasUsed": 15785 + "gasUsed": 15800 }, { "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromThirdField", "name": "pop from field (warm, 2 slots, 10 uint32 items)", - "gasUsed": 11792 + "gasUsed": 11807 }, { "file": "test/StoreCoreGas.t.sol", @@ -717,7 +717,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testDeleteData", "name": "delete record (complex data, 3 slots)", - "gasUsed": 8000 + "gasUsed": 8015 }, { "file": "test/StoreCoreGas.t.sol", @@ -747,7 +747,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "register subscriber", - "gasUsed": 57939 + "gasUsed": 57954 }, { "file": "test/StoreCoreGas.t.sol", @@ -759,19 +759,19 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "set static field on table with subscriber", - "gasUsed": 19761 + "gasUsed": 19791 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "delete record on table with subscriber", - "gasUsed": 18587 + "gasUsed": 18617 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "register subscriber", - "gasUsed": 57939 + "gasUsed": 57954 }, { "file": "test/StoreCoreGas.t.sol", @@ -783,31 +783,31 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "set (dynamic) field on table with subscriber", - "gasUsed": 24439 + "gasUsed": 24469 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "delete (dynamic) record on table with subscriber", - "gasUsed": 20253 + "gasUsed": 20283 }, { "file": "test/StoreCoreGas.t.sol", "test": "testPushToDynamicField", "name": "push to field (1 slot, 1 uint32 item)", - "gasUsed": 9444 + "gasUsed": 9459 }, { "file": "test/StoreCoreGas.t.sol", "test": "testPushToDynamicField", "name": "push to field (2 slots, 10 uint32 items)", - "gasUsed": 32114 + "gasUsed": 32129 }, { "file": "test/StoreCoreGas.t.sol", "test": "testRegisterAndGetFieldLayout", "name": "StoreCore: register table", - "gasUsed": 640899 + "gasUsed": 640914 }, { "file": "test/StoreCoreGas.t.sol", @@ -873,7 +873,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set static field (1 slot)", - "gasUsed": 31209 + "gasUsed": 31224 }, { "file": "test/StoreCoreGas.t.sol", @@ -885,7 +885,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set static field (overlap 2 slot)", - "gasUsed": 29862 + "gasUsed": 29877 }, { "file": "test/StoreCoreGas.t.sol", @@ -897,7 +897,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set dynamic field (1 slot, first dynamic field)", - "gasUsed": 53930 + "gasUsed": 53945 }, { "file": "test/StoreCoreGas.t.sol", @@ -909,7 +909,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set dynamic field (1 slot, second dynamic field)", - "gasUsed": 32153 + "gasUsed": 32168 }, { "file": "test/StoreCoreGas.t.sol", @@ -945,13 +945,13 @@ "file": "test/StoreCoreGas.t.sol", "test": "testUpdateInDynamicField", "name": "update in field (1 slot, 1 uint32 item)", - "gasUsed": 8792 + "gasUsed": 8807 }, { "file": "test/StoreCoreGas.t.sol", "test": "testUpdateInDynamicField", "name": "push to field (2 slots, 6 uint64 items)", - "gasUsed": 9238 + "gasUsed": 9253 }, { "file": "test/StoreHook.t.sol", @@ -981,13 +981,13 @@ "file": "test/StoreHooks.t.sol", "test": "testOneSlot", "name": "StoreHooks: set field with one elements (cold)", - "gasUsed": 58247 + "gasUsed": 58262 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: set field (cold)", - "gasUsed": 58247 + "gasUsed": 58262 }, { "file": "test/StoreHooks.t.sol", @@ -999,55 +999,55 @@ "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: push 1 element (cold)", - "gasUsed": 12597 + "gasUsed": 12612 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: pop 1 element (warm)", - "gasUsed": 9931 + "gasUsed": 9946 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: push 1 element (warm)", - "gasUsed": 10614 + "gasUsed": 10629 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: update 1 element (warm)", - "gasUsed": 29840 + "gasUsed": 29855 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: delete record (warm)", - "gasUsed": 10410 + "gasUsed": 10425 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: set field (warm)", - "gasUsed": 30392 + "gasUsed": 30407 }, { "file": "test/StoreHooks.t.sol", "test": "testThreeSlots", "name": "StoreHooks: set field with three elements (cold)", - "gasUsed": 80935 + "gasUsed": 80950 }, { "file": "test/StoreHooks.t.sol", "test": "testTwoSlots", "name": "StoreHooks: set field with two elements (cold)", - "gasUsed": 80847 + "gasUsed": 80862 }, { "file": "test/StoreHooksColdLoad.t.sol", "test": "testDelete", "name": "StoreHooks: delete record (cold)", - "gasUsed": 19268 + "gasUsed": 19283 }, { "file": "test/StoreHooksColdLoad.t.sol", @@ -1071,13 +1071,13 @@ "file": "test/StoreHooksColdLoad.t.sol", "test": "testPop", "name": "StoreHooks: pop 1 element (cold)", - "gasUsed": 18366 + "gasUsed": 18381 }, { "file": "test/StoreHooksColdLoad.t.sol", "test": "testUpdate", "name": "StoreHooks: update 1 element (cold)", - "gasUsed": 20295 + "gasUsed": 20310 }, { "file": "test/StoreSwitch.t.sol", @@ -1143,7 +1143,7 @@ "file": "test/Vector2.t.sol", "test": "testRegisterAndGetFieldLayout", "name": "register Vector2 field layout", - "gasUsed": 442403 + "gasUsed": 442418 }, { "file": "test/Vector2.t.sol", diff --git a/packages/store/src/Storage.sol b/packages/store/src/Storage.sol index b4d0d73396..65f7221f88 100644 --- a/packages/store/src/Storage.sol +++ b/packages/store/src/Storage.sol @@ -27,7 +27,7 @@ library Storage { * @param data Bytes to store. */ function store(uint256 storagePointer, uint256 offset, bytes memory data) internal { - store(storagePointer, offset, Memory.dataPointer(data), data.length); + store(storagePointer, offset, data.length, Memory.dataPointer(data)); } /** @@ -37,7 +37,7 @@ library Storage { * @param memoryPointer Pointer to the start of the data in memory. * @param length Length of the data in bytes. */ - function store(uint256 storagePointer, uint256 offset, uint256 memoryPointer, uint256 length) internal { + function store(uint256 storagePointer, uint256 offset, uint256 length, uint256 memoryPointer) internal { if (offset > 0) { // Support offsets that are greater than 32 bytes by incrementing the storagePointer and decrementing the offset if (offset >= 32) { @@ -153,7 +153,7 @@ library Storage { * @param offset Offset within the storage location. * @return result The loaded bytes of data. */ - function load(uint256 storagePointer, uint256 length, uint256 offset) internal view returns (bytes memory result) { + function load(uint256 storagePointer, uint256 offset, uint256 length) internal view returns (bytes memory result) { uint256 memoryPointer; /// @solidity memory-safe-assembly assembly { @@ -169,7 +169,7 @@ library Storage { // Store length mstore(result, length) } - load(storagePointer, length, offset, memoryPointer); + load(storagePointer, offset, length, memoryPointer); return result; } @@ -180,7 +180,7 @@ library Storage { * @param offset Offset within the storage location. * @param memoryPointer Pointer to the location in memory to append the data. */ - function load(uint256 storagePointer, uint256 length, uint256 offset, uint256 memoryPointer) internal view { + function load(uint256 storagePointer, uint256 offset, uint256 length, uint256 memoryPointer) internal view { if (offset > 0) { // Support offsets that are greater than 32 bytes by incrementing the storagePointer and decrementing the offset if (offset >= 32) { diff --git a/packages/store/src/StoreCore.sol b/packages/store/src/StoreCore.sol index 60e6e993c0..50b2028d7d 100644 --- a/packages/store/src/StoreCore.sol +++ b/packages/store/src/StoreCore.sol @@ -328,8 +328,8 @@ library StoreCore { Storage.store({ storagePointer: staticDataLocation, offset: 0, - memoryPointer: memoryPointer, - length: staticData.length + length: staticData.length, + memoryPointer: memoryPointer }); // Set the dynamic data if there are dynamic fields @@ -350,8 +350,8 @@ library StoreCore { Storage.store({ storagePointer: dynamicDataLocation, offset: 0, - memoryPointer: memoryPointer, - length: dynamicDataLength + length: dynamicDataLength, + memoryPointer: memoryPointer }); memoryPointer += dynamicDataLength; // move the memory pointer to the start of the next dynamic data unchecked { @@ -740,7 +740,7 @@ library StoreCore { for (uint8 i; i < numDynamicFields; i++) { uint256 dynamicDataLocation = StoreCoreInternal._getDynamicDataLocation(tableId, keyTuple, i); uint256 length = encodedLengths.atIndex(i); - Storage.load({ storagePointer: dynamicDataLocation, length: length, offset: 0, memoryPointer: memoryPointer }); + Storage.load({ storagePointer: dynamicDataLocation, offset: 0, length: length, memoryPointer: memoryPointer }); // Advance memoryPointer by the length of this dynamic field memoryPointer += length; } @@ -828,8 +828,8 @@ library StoreCore { return Storage.load({ storagePointer: StoreCoreInternal._getDynamicDataLocation(tableId, keyTuple, dynamicFieldIndex), - length: StoreCoreInternal._loadEncodedDynamicDataLength(tableId, keyTuple).atIndex(dynamicFieldIndex), - offset: 0 + offset: 0, + length: StoreCoreInternal._loadEncodedDynamicDataLength(tableId, keyTuple).atIndex(dynamicFieldIndex) }); } @@ -917,7 +917,7 @@ library StoreCore { // Get the length and storage location of the dynamic field uint256 location = StoreCoreInternal._getDynamicDataLocation(tableId, keyTuple, dynamicFieldIndex); - return Storage.load({ storagePointer: location, length: end - start, offset: start }); + return Storage.load({ storagePointer: location, offset: start, length: end - start }); } } @@ -1078,7 +1078,7 @@ library StoreCoreInternal { // Load the data from storage uint256 location = _getStaticDataLocation(tableId, keyTuple); - return Storage.load({ storagePointer: location, length: length, offset: 0 }); + return Storage.load({ storagePointer: location, offset: 0, length: length }); } /** @@ -1100,8 +1100,8 @@ library StoreCoreInternal { return Storage.load({ storagePointer: _getStaticDataLocation(tableId, keyTuple), - length: fieldLayout.atIndex(fieldIndex), - offset: _getStaticDataOffset(fieldLayout, fieldIndex) + offset: _getStaticDataOffset(fieldLayout, fieldIndex), + length: fieldLayout.atIndex(fieldIndex) }); } diff --git a/packages/store/test/GasStorageLoad.t.sol b/packages/store/test/GasStorageLoad.t.sol index 7db60617b2..cfafdbcd65 100644 --- a/packages/store/test/GasStorageLoad.t.sol +++ b/packages/store/test/GasStorageLoad.t.sol @@ -71,38 +71,38 @@ contract GasStorageLoadTest is Test, GasReporter { bytes32 encodedFieldPartial = valuePartial; startGasReport("MUD storage load (cold, 1 word)"); - encodedSimple = Storage.load(SolidityStorage.STORAGE_SLOT_SIMPLE, encodedSimple.length, 0); + encodedSimple = Storage.load(SolidityStorage.STORAGE_SLOT_SIMPLE, 0, encodedSimple.length); endGasReport(); startGasReport("MUD storage load (cold, 1 word, partial)"); - encodedPartial = Storage.load(SolidityStorage.STORAGE_SLOT_PARTIAL, encodedPartial.length, 16); + encodedPartial = Storage.load(SolidityStorage.STORAGE_SLOT_PARTIAL, 16, encodedPartial.length); endGasReport(); startGasReport("MUD storage load (cold, 10 words)"); - encoded9Words = Storage.load(SolidityStorage.STORAGE_SLOT_BYTES, encoded9Words.length, 0); + encoded9Words = Storage.load(SolidityStorage.STORAGE_SLOT_BYTES, 0, encoded9Words.length); endGasReport(); // warm startGasReport("MUD storage load (warm, 1 word)"); - encodedSimple = Storage.load(SolidityStorage.STORAGE_SLOT_SIMPLE, encodedSimple.length, 0); + encodedSimple = Storage.load(SolidityStorage.STORAGE_SLOT_SIMPLE, 0, encodedSimple.length); endGasReport(); startGasReport("MUD storage load field (warm, 1 word)"); - encodedFieldSimple = Storage.loadField(SolidityStorage.STORAGE_SLOT_SIMPLE, encodedSimple.length, 0); + encodedFieldSimple = Storage.loadField(SolidityStorage.STORAGE_SLOT_SIMPLE, 0, encodedSimple.length); endGasReport(); startGasReport("MUD storage load (warm, 1 word, partial)"); - encodedPartial = Storage.load(SolidityStorage.STORAGE_SLOT_PARTIAL, encodedPartial.length, 16); + encodedPartial = Storage.load(SolidityStorage.STORAGE_SLOT_PARTIAL, 16, encodedPartial.length); endGasReport(); - encodedFieldPartial = Storage.loadField(SolidityStorage.STORAGE_SLOT_PARTIAL, encodedSimple.length, 16); + encodedFieldPartial = Storage.loadField(SolidityStorage.STORAGE_SLOT_PARTIAL, 16, encodedSimple.length); startGasReport("MUD storage load field (warm, 1 word, partial)"); - encodedFieldPartial = Storage.loadField(SolidityStorage.STORAGE_SLOT_PARTIAL, encodedSimple.length, 16); + encodedFieldPartial = Storage.loadField(SolidityStorage.STORAGE_SLOT_PARTIAL, 16, encodedSimple.length); endGasReport(); startGasReport("MUD storage load (warm, 10 words)"); - encoded9Words = Storage.load(SolidityStorage.STORAGE_SLOT_BYTES, encoded9Words.length, 0); + encoded9Words = Storage.load(SolidityStorage.STORAGE_SLOT_BYTES, 0, encoded9Words.length); endGasReport(); // Do something in case the optimizer removes unused assignments diff --git a/packages/store/test/Storage.t.sol b/packages/store/test/Storage.t.sol index 6d8e178a4e..6b2fb40c8f 100644 --- a/packages/store/test/Storage.t.sol +++ b/packages/store/test/Storage.t.sol @@ -59,7 +59,7 @@ contract StorageTest is Test, GasReporter { // Assert we can load the correct partial value from storage startGasReport("load 34 bytes over 3 storage slots (with offset and safeTrail))"); - bytes memory data = Storage.load({ storagePointer: storagePointer, length: 34, offset: 31 }); + bytes memory data = Storage.load({ storagePointer: storagePointer, offset: 31, length: 34 }); endGasReport(); assertEq(Bytes.slice1(data, 0), bytes1(0x01)); @@ -73,7 +73,7 @@ contract StorageTest is Test, GasReporter { vm.assume(storagePointer > 0); Storage.store({ storagePointer: uint256(storagePointer), offset: offset, data: data }); - assertEq(Storage.load({ storagePointer: uint256(storagePointer), length: data.length, offset: offset }), data); + assertEq(Storage.load({ storagePointer: uint256(storagePointer), offset: offset, length: data.length }), data); } function testStoreLoadFieldBytes32Fuzzy(bytes32 data, uint256 storagePointer, uint256 offset) public { @@ -126,7 +126,7 @@ contract StorageTest is Test, GasReporter { uint256 storagePointer = uint256(keccak256("some location")); Storage.store({ storagePointer: storagePointer, offset: 10, data: abi.encodePacked(dataAfterLoad) }); - Storage.load({ storagePointer: storagePointer, length: 10, offset: 10, memoryPointer: memoryPointer }); + Storage.load({ storagePointer: storagePointer, offset: 10, length: 10, memoryPointer: memoryPointer }); assertEq(testData, expectedData); } diff --git a/packages/world-modules/gas-report.json b/packages/world-modules/gas-report.json index 9be66918a1..2e39630ee8 100644 --- a/packages/world-modules/gas-report.json +++ b/packages/world-modules/gas-report.json @@ -3,103 +3,103 @@ "file": "test/ERC20.t.sol", "test": "testApprove", "name": "approve", - "gasUsed": 114329 + "gasUsed": 114344 }, { "file": "test/ERC20.t.sol", "test": "testBurn", "name": "burn", - "gasUsed": 75866 + "gasUsed": 75896 }, { "file": "test/ERC20.t.sol", "test": "testMint", "name": "mint", - "gasUsed": 161705 + "gasUsed": 161735 }, { "file": "test/ERC20.t.sol", "test": "testTransfer", "name": "transfer", - "gasUsed": 92948 + "gasUsed": 92978 }, { "file": "test/ERC20.t.sol", "test": "testTransferFrom", "name": "transferFrom", - "gasUsed": 130250 + "gasUsed": 130295 }, { "file": "test/ERC721.t.sol", "test": "testApproveAllGas", "name": "setApprovalForAll", - "gasUsed": 113956 + "gasUsed": 113971 }, { "file": "test/ERC721.t.sol", "test": "testApproveGas", "name": "approve", - "gasUsed": 87965 + "gasUsed": 87980 }, { "file": "test/ERC721.t.sol", "test": "testBurnGas", "name": "burn", - "gasUsed": 101835 + "gasUsed": 101880 }, { "file": "test/ERC721.t.sol", "test": "testMintGas", "name": "mint", - "gasUsed": 169433 + "gasUsed": 169463 }, { "file": "test/ERC721.t.sol", "test": "testSafeMintToEOAGas", "name": "safeMint", - "gasUsed": 169704 + "gasUsed": 169734 }, { "file": "test/ERC721.t.sol", "test": "testSafeTransferFromToEOAGas", "name": "safeTransferFrom", - "gasUsed": 143638 + "gasUsed": 143698 }, { "file": "test/ERC721.t.sol", "test": "testTransferFromGas", "name": "transferFrom", - "gasUsed": 136798 + "gasUsed": 136858 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallComposite", "name": "install keys in table module", - "gasUsed": 1413089 + "gasUsed": 1413179 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallGas", "name": "install keys in table module", - "gasUsed": 1413089 + "gasUsed": 1413179 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallGas", "name": "set a record on a table with keysInTableModule installed", - "gasUsed": 158814 + "gasUsed": 158829 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallSingleton", "name": "install keys in table module", - "gasUsed": 1413089 + "gasUsed": 1413179 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookCompositeGas", "name": "install keys in table module", - "gasUsed": 1413089 + "gasUsed": 1413179 }, { "file": "test/KeysInTableModule.t.sol", @@ -111,13 +111,13 @@ "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookCompositeGas", "name": "delete a composite record on a table with keysInTableModule installed", - "gasUsed": 155729 + "gasUsed": 155864 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookGas", "name": "install keys in table module", - "gasUsed": 1413089 + "gasUsed": 1413179 }, { "file": "test/KeysInTableModule.t.sol", @@ -129,13 +129,13 @@ "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookGas", "name": "delete a record on a table with keysInTableModule installed", - "gasUsed": 84971 + "gasUsed": 85046 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testGetKeysWithValueGas", "name": "install keys with value module", - "gasUsed": 653436 + "gasUsed": 653541 }, { "file": "test/KeysWithValueModule.t.sol", @@ -153,49 +153,49 @@ "file": "test/KeysWithValueModule.t.sol", "test": "testInstall", "name": "install keys with value module", - "gasUsed": 653436 + "gasUsed": 653541 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testInstall", "name": "set a record on a table with KeysWithValueModule installed", - "gasUsed": 135405 + "gasUsed": 135435 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "install keys with value module", - "gasUsed": 653436 + "gasUsed": 653541 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "change a record on a table with KeysWithValueModule installed", - "gasUsed": 103789 + "gasUsed": 103819 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "delete a record on a table with KeysWithValueModule installed", - "gasUsed": 36471 + "gasUsed": 36501 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "install keys with value module", - "gasUsed": 653436 + "gasUsed": 653541 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "set a field on a table with KeysWithValueModule installed", - "gasUsed": 146626 + "gasUsed": 146671 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "change a field on a table with KeysWithValueModule installed", - "gasUsed": 111385 + "gasUsed": 111430 }, { "file": "test/query.t.sol", @@ -267,31 +267,31 @@ "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromCallboundDelegation", "name": "register a callbound delegation", - "gasUsed": 118295 + "gasUsed": 118325 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromCallboundDelegation", "name": "call a system via a callbound delegation", - "gasUsed": 36688 + "gasUsed": 36703 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromSystemDelegation", "name": "register a systembound delegation", - "gasUsed": 115851 + "gasUsed": 115881 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromSystemDelegation", "name": "call a system via a systembound delegation", - "gasUsed": 33831 + "gasUsed": 33846 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromTimeboundDelegation", "name": "register a timebound delegation", - "gasUsed": 112789 + "gasUsed": 112819 }, { "file": "test/StandardDelegationsModule.t.sol", @@ -303,24 +303,24 @@ "file": "test/UniqueEntityModule.t.sol", "test": "testInstall", "name": "install unique entity module", - "gasUsed": 681732 + "gasUsed": 681852 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstall", "name": "get a unique entity nonce (non-root module)", - "gasUsed": 51026 + "gasUsed": 51041 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstallRoot", "name": "installRoot unique entity module", - "gasUsed": 648129 + "gasUsed": 648249 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstallRoot", "name": "get a unique entity nonce (root module)", - "gasUsed": 51026 + "gasUsed": 51041 } ] diff --git a/packages/world/gas-report.json b/packages/world/gas-report.json index caf93441b1..09c68697d8 100644 --- a/packages/world/gas-report.json +++ b/packages/world/gas-report.json @@ -69,7 +69,7 @@ "file": "test/World.t.sol", "test": "testCallFromUnlimitedDelegation", "name": "register an unlimited delegation", - "gasUsed": 47623 + "gasUsed": 47638 }, { "file": "test/World.t.sol", @@ -81,13 +81,13 @@ "file": "test/World.t.sol", "test": "testDeleteRecord", "name": "Delete record", - "gasUsed": 9837 + "gasUsed": 9852 }, { "file": "test/World.t.sol", "test": "testPushToDynamicField", "name": "Push data to the table", - "gasUsed": 85811 + "gasUsed": 85826 }, { "file": "test/World.t.sol", @@ -99,7 +99,7 @@ "file": "test/World.t.sol", "test": "testRegisterNamespace", "name": "Register a new namespace", - "gasUsed": 120887 + "gasUsed": 120932 }, { "file": "test/World.t.sol", @@ -111,19 +111,19 @@ "file": "test/World.t.sol", "test": "testRegisterSystem", "name": "register a system", - "gasUsed": 164238 + "gasUsed": 164283 }, { "file": "test/World.t.sol", "test": "testRegisterTable", "name": "Register a new table in the namespace", - "gasUsed": 636447 + "gasUsed": 636507 }, { "file": "test/World.t.sol", "test": "testSetField", "name": "Write data to a table field", - "gasUsed": 36881 + "gasUsed": 36896 }, { "file": "test/World.t.sol", @@ -135,25 +135,25 @@ "file": "test/WorldDynamicUpdate.t.sol", "test": "testPopFromDynamicField", "name": "pop 1 address (cold)", - "gasUsed": 25602 + "gasUsed": 25617 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testPopFromDynamicField", "name": "pop 1 address (warm)", - "gasUsed": 12748 + "gasUsed": 12763 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testSpliceDynamicData", "name": "update in field 1 item (cold)", - "gasUsed": 25953 + "gasUsed": 25968 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testSpliceDynamicData", "name": "update in field 1 item (warm)", - "gasUsed": 13154 + "gasUsed": 13169 }, { "file": "test/WorldResourceId.t.sol", From 7f089c7d0650865664ca2cf65c583f14d9dc451a Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Thu, 11 Jan 2024 11:36:42 +0200 Subject: [PATCH 10/23] feat(examples): access `World` from different accounts (#2108) --- examples/multiple-accounts/.eslintrc | 10 + examples/multiple-accounts/.gitattributes | 3 + examples/multiple-accounts/.gitignore | 1 + .../multiple-accounts/.vscode/extensions.json | 3 + .../multiple-accounts/.vscode/settings.json | 1 + examples/multiple-accounts/mprocs.yaml | 7 + examples/multiple-accounts/package.json | 29 + .../packages/client/.eslintrc | 7 + .../packages/client/.gitignore | 3 + .../packages/client/index.html | 12 + .../packages/client/package.json | 37 + .../packages/client/src/App.tsx | 110 + .../packages/client/src/MUDContext.tsx | 21 + .../packages/client/src/index.tsx | 34 + .../client/src/mud/createSystemCalls.ts | 13 + .../client/src/mud/getNetworkConfig.ts | 91 + .../packages/client/src/mud/setup.ts | 18 + .../packages/client/src/mud/setupNetwork.ts | 120 + .../client/src/mud/supportedChains.ts | 20 + .../packages/client/tsconfig.json | 19 + .../packages/client/vite.config.ts | 17 + .../packages/contracts/.gitignore | 9 + .../packages/contracts/.prettierrc | 8 + .../packages/contracts/.solhint.json | 12 + .../packages/contracts/foundry.toml | 26 + .../packages/contracts/mud.config.ts | 15 + .../packages/contracts/package.json | 34 + .../packages/contracts/remappings.txt | 3 + .../contracts/script/PostDeploy.s.sol | 27 + .../packages/contracts/src/codegen/index.sol | 6 + .../contracts/src/codegen/tables/LastCall.sol | 231 + .../src/codegen/world/ILastCallSystem.sol | 12 + .../src/codegen/world/ITasksSystem.sol | 18 + .../contracts/src/codegen/world/IWorld.sol | 18 + .../contracts/src/systems/LastCallSystem.sol | 11 + .../packages/contracts/tsconfig.json | 13 + .../packages/contracts/worlds.json | 5 + .../packages/contracts/worlds.json.d.ts | 2 + examples/multiple-accounts/pnpm-lock.yaml | 4323 +++++++++++++++++ .../multiple-accounts/pnpm-workspace.yaml | 2 + 40 files changed, 5351 insertions(+) create mode 100644 examples/multiple-accounts/.eslintrc create mode 100644 examples/multiple-accounts/.gitattributes create mode 100644 examples/multiple-accounts/.gitignore create mode 100644 examples/multiple-accounts/.vscode/extensions.json create mode 100644 examples/multiple-accounts/.vscode/settings.json create mode 100644 examples/multiple-accounts/mprocs.yaml create mode 100644 examples/multiple-accounts/package.json create mode 100644 examples/multiple-accounts/packages/client/.eslintrc create mode 100644 examples/multiple-accounts/packages/client/.gitignore create mode 100644 examples/multiple-accounts/packages/client/index.html create mode 100644 examples/multiple-accounts/packages/client/package.json create mode 100644 examples/multiple-accounts/packages/client/src/App.tsx create mode 100644 examples/multiple-accounts/packages/client/src/MUDContext.tsx create mode 100644 examples/multiple-accounts/packages/client/src/index.tsx create mode 100644 examples/multiple-accounts/packages/client/src/mud/createSystemCalls.ts create mode 100644 examples/multiple-accounts/packages/client/src/mud/getNetworkConfig.ts create mode 100644 examples/multiple-accounts/packages/client/src/mud/setup.ts create mode 100644 examples/multiple-accounts/packages/client/src/mud/setupNetwork.ts create mode 100644 examples/multiple-accounts/packages/client/src/mud/supportedChains.ts create mode 100644 examples/multiple-accounts/packages/client/tsconfig.json create mode 100644 examples/multiple-accounts/packages/client/vite.config.ts create mode 100644 examples/multiple-accounts/packages/contracts/.gitignore create mode 100644 examples/multiple-accounts/packages/contracts/.prettierrc create mode 100644 examples/multiple-accounts/packages/contracts/.solhint.json create mode 100644 examples/multiple-accounts/packages/contracts/foundry.toml create mode 100644 examples/multiple-accounts/packages/contracts/mud.config.ts create mode 100644 examples/multiple-accounts/packages/contracts/package.json create mode 100644 examples/multiple-accounts/packages/contracts/remappings.txt create mode 100644 examples/multiple-accounts/packages/contracts/script/PostDeploy.s.sol create mode 100644 examples/multiple-accounts/packages/contracts/src/codegen/index.sol create mode 100644 examples/multiple-accounts/packages/contracts/src/codegen/tables/LastCall.sol create mode 100644 examples/multiple-accounts/packages/contracts/src/codegen/world/ILastCallSystem.sol create mode 100644 examples/multiple-accounts/packages/contracts/src/codegen/world/ITasksSystem.sol create mode 100644 examples/multiple-accounts/packages/contracts/src/codegen/world/IWorld.sol create mode 100644 examples/multiple-accounts/packages/contracts/src/systems/LastCallSystem.sol create mode 100644 examples/multiple-accounts/packages/contracts/tsconfig.json create mode 100644 examples/multiple-accounts/packages/contracts/worlds.json create mode 100644 examples/multiple-accounts/packages/contracts/worlds.json.d.ts create mode 100644 examples/multiple-accounts/pnpm-lock.yaml create mode 100644 examples/multiple-accounts/pnpm-workspace.yaml diff --git a/examples/multiple-accounts/.eslintrc b/examples/multiple-accounts/.eslintrc new file mode 100644 index 0000000000..79bd6ef23f --- /dev/null +++ b/examples/multiple-accounts/.eslintrc @@ -0,0 +1,10 @@ +{ + "root": true, + "parser": "@typescript-eslint/parser", + "plugins": ["@typescript-eslint"], + "extends": [ + "eslint:recommended", + "plugin:@typescript-eslint/eslint-recommended", + "plugin:@typescript-eslint/recommended" + ] +} diff --git a/examples/multiple-accounts/.gitattributes b/examples/multiple-accounts/.gitattributes new file mode 100644 index 0000000000..9c70dc52f0 --- /dev/null +++ b/examples/multiple-accounts/.gitattributes @@ -0,0 +1,3 @@ +# suppress diffs for generated files +**/pnpm-lock.yaml linguist-generated=true +**/codegen/**/*.sol linguist-generated=true diff --git a/examples/multiple-accounts/.gitignore b/examples/multiple-accounts/.gitignore new file mode 100644 index 0000000000..b512c09d47 --- /dev/null +++ b/examples/multiple-accounts/.gitignore @@ -0,0 +1 @@ +node_modules \ No newline at end of file diff --git a/examples/multiple-accounts/.vscode/extensions.json b/examples/multiple-accounts/.vscode/extensions.json new file mode 100644 index 0000000000..7d5d7da3d9 --- /dev/null +++ b/examples/multiple-accounts/.vscode/extensions.json @@ -0,0 +1,3 @@ +{ + "recommendations": ["NomicFoundation.hardhat-solidity"] +} diff --git a/examples/multiple-accounts/.vscode/settings.json b/examples/multiple-accounts/.vscode/settings.json new file mode 100644 index 0000000000..0967ef424b --- /dev/null +++ b/examples/multiple-accounts/.vscode/settings.json @@ -0,0 +1 @@ +{} diff --git a/examples/multiple-accounts/mprocs.yaml b/examples/multiple-accounts/mprocs.yaml new file mode 100644 index 0000000000..89cf2bd606 --- /dev/null +++ b/examples/multiple-accounts/mprocs.yaml @@ -0,0 +1,7 @@ +procs: + contracts: + cwd: packages/contracts + shell: pnpm run dev + client: + cwd: packages/client + shell: pnpm run dev diff --git a/examples/multiple-accounts/package.json b/examples/multiple-accounts/package.json new file mode 100644 index 0000000000..efabf8e04f --- /dev/null +++ b/examples/multiple-accounts/package.json @@ -0,0 +1,29 @@ +{ + "name": "mud-template-react", + "private": true, + "scripts": { + "build": "pnpm recursive run build", + "dev": "mprocs", + "dev:client": "pnpm --filter 'client' run dev", + "dev:contracts": "pnpm --filter 'contracts' dev", + "foundry:up": "curl -L https://foundry.paradigm.xyz | bash && bash $HOME/.foundry/bin/foundryup", + "mud:up": "pnpm mud set-version --tag main && pnpm install", + "prepare": "(forge --version || pnpm foundry:up)", + "test": "pnpm recursive run test" + }, + "devDependencies": { + "@latticexyz/cli": "2.0.0-next.15", + "@types/debug": "4.1.7", + "@types/prettier": "2.7.2", + "@typescript-eslint/eslint-plugin": "5.46.1", + "@typescript-eslint/parser": "5.46.1", + "eslint": "8.29.0", + "mprocs": "^0.6.4", + "rimraf": "^3.0.2", + "typescript": "5.1.6" + }, + "engines": { + "node": "18.x", + "pnpm": "8.x" + } +} diff --git a/examples/multiple-accounts/packages/client/.eslintrc b/examples/multiple-accounts/packages/client/.eslintrc new file mode 100644 index 0000000000..930af95967 --- /dev/null +++ b/examples/multiple-accounts/packages/client/.eslintrc @@ -0,0 +1,7 @@ +{ + "extends": ["../../.eslintrc", "plugin:react/recommended", "plugin:react-hooks/recommended"], + "plugins": ["react", "react-hooks"], + "rules": { + "react/react-in-jsx-scope": "off" + } +} diff --git a/examples/multiple-accounts/packages/client/.gitignore b/examples/multiple-accounts/packages/client/.gitignore new file mode 100644 index 0000000000..0ca39c007c --- /dev/null +++ b/examples/multiple-accounts/packages/client/.gitignore @@ -0,0 +1,3 @@ +node_modules +dist +.DS_Store diff --git a/examples/multiple-accounts/packages/client/index.html b/examples/multiple-accounts/packages/client/index.html new file mode 100644 index 0000000000..fca4e978ab --- /dev/null +++ b/examples/multiple-accounts/packages/client/index.html @@ -0,0 +1,12 @@ + + + + + + Show when each address called us last + + +
+ + + diff --git a/examples/multiple-accounts/packages/client/package.json b/examples/multiple-accounts/packages/client/package.json new file mode 100644 index 0000000000..729a030305 --- /dev/null +++ b/examples/multiple-accounts/packages/client/package.json @@ -0,0 +1,37 @@ +{ + "name": "client", + "version": "0.0.0", + "private": true, + "license": "MIT", + "type": "module", + "scripts": { + "build": "vite build", + "dev": "wait-port localhost:8545 && vite", + "preview": "vite preview", + "test": "tsc --noEmit" + }, + "dependencies": { + "@latticexyz/common": "2.0.0-next.15", + "@latticexyz/dev-tools": "2.0.0-next.15", + "@latticexyz/react": "2.0.0-next.15", + "@latticexyz/schema-type": "2.0.0-next.15", + "@latticexyz/services": "2.0.0-next.15", + "@latticexyz/store-sync": "2.0.0-next.15", + "@latticexyz/utils": "2.0.0-next.15", + "@latticexyz/world": "2.0.0-next.15", + "contracts": "workspace:*", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "rxjs": "7.5.5", + "viem": "1.14.0" + }, + "devDependencies": { + "@types/react": "18.2.22", + "@types/react-dom": "18.2.7", + "@vitejs/plugin-react": "^3.1.0", + "eslint-plugin-react": "7.31.11", + "eslint-plugin-react-hooks": "4.6.0", + "vite": "^4.2.1", + "wait-port": "^1.0.4" + } +} diff --git a/examples/multiple-accounts/packages/client/src/App.tsx b/examples/multiple-accounts/packages/client/src/App.tsx new file mode 100644 index 0000000000..f3d5e4bf4e --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/App.tsx @@ -0,0 +1,110 @@ +import { useMUD } from "./MUDContext"; +import { getBurnerPrivateKey, createBurnerAccount, transportObserver, getContract } from "@latticexyz/common"; +import { createPublicClient, createWalletClient, fallback, webSocket, http, ClientConfig, Hex } from "viem"; +import { getNetworkConfig } from "./mud/getNetworkConfig"; +import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; + +// A lot of code that would normally go in mud/getNetworkConfig.ts is part of the application here, +// because `getNetworkConfig.ts` assumes you use one wallet, and here we need several. +const networkConfig = await getNetworkConfig(); + +const clientOptions = { + chain: networkConfig.chain, + transport: transportObserver(fallback([webSocket(), http()])), + pollingInterval: 1000, +} as const satisfies ClientConfig; + +const publicClient = createPublicClient({ + ...clientOptions, +}); + +// Create a structure with two fields: +// client - a wallet client that uses a random account +// world - a world contract object that lets us issue newCall +const makeWorldContract = () => { + const client = createWalletClient({ + ...clientOptions, + account: createBurnerAccount(getBurnerPrivateKey(Math.random().toString())), + }); + + return { + world: getContract({ + address: networkConfig.worldAddress as Hex, + abi: IWorldAbi, + publicClient: publicClient, + walletClient: client, + }), + client, + }; +}; + +// Create five world contracts +const worldContracts = [1, 2, 3, 4, 5].map((x) => makeWorldContract()); + +export const App = () => { + const { + network: { tables, useStore }, + } = useMUD(); + + // Get all the calls from the records cache. + const calls = useStore((state) => { + const records = Object.values(state.getRecords(tables.LastCall)); + records.sort((a, b) => Number(a.value.callTime - b.value.callTime)); + return records; + }); + + // Convert timestamps to readable format + const twoDigit = (str) => str.toString().padStart(2, "0"); + const timestamp2Str = (timestamp: number) => { + const date = new Date(timestamp * 1000); + return `${twoDigit(date.getHours())}:${twoDigit(date.getMinutes())}:${twoDigit(date.getSeconds())}`; + }; + + // Call newCall() on LastCall:LastCallSystem. + const newCall = async (worldContract) => { + const tx = await worldContract.write.LastCall_LastCallSystem_newCall(); + }; + + return ( + <> +

Last calls

+ + + + + + + { + // Show all the calls + calls.map((call) => ( + + + + + )) + } + +
CallerTime
{call.key.caller}{timestamp2Str(Number(call.value.callTime))}
+

My clients

+ { + // For every world contract, have a button to call newCall as that address. + worldContracts.map((worldContract) => ( + <> + +
+ + )) + } + + ); +}; diff --git a/examples/multiple-accounts/packages/client/src/MUDContext.tsx b/examples/multiple-accounts/packages/client/src/MUDContext.tsx new file mode 100644 index 0000000000..7b5637f6a6 --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/MUDContext.tsx @@ -0,0 +1,21 @@ +import { createContext, ReactNode, useContext } from "react"; +import { SetupResult } from "./mud/setup"; + +const MUDContext = createContext(null); + +type Props = { + children: ReactNode; + value: SetupResult; +}; + +export const MUDProvider = ({ children, value }: Props) => { + const currentValue = useContext(MUDContext); + if (currentValue) throw new Error("MUDProvider can only be used once"); + return {children}; +}; + +export const useMUD = () => { + const value = useContext(MUDContext); + if (!value) throw new Error("Must be used within a MUDProvider"); + return value; +}; diff --git a/examples/multiple-accounts/packages/client/src/index.tsx b/examples/multiple-accounts/packages/client/src/index.tsx new file mode 100644 index 0000000000..3ba2a44052 --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/index.tsx @@ -0,0 +1,34 @@ +import ReactDOM from "react-dom/client"; +import { App } from "./App"; +import { setup } from "./mud/setup"; +import { MUDProvider } from "./MUDContext"; +import mudConfig from "contracts/mud.config"; + +const rootElement = document.getElementById("react-root"); +if (!rootElement) throw new Error("React root not found"); +const root = ReactDOM.createRoot(rootElement); + +// TODO: figure out if we actually want this to be async or if we should render something else in the meantime +setup().then(async (result) => { + root.render( + + + + ); + + // https://vitejs.dev/guide/env-and-mode.html + if (import.meta.env.DEV) { + const { mount: mountDevTools } = await import("@latticexyz/dev-tools"); + mountDevTools({ + config: mudConfig, + publicClient: result.network.publicClient, + walletClient: result.network.walletClient, + latestBlock$: result.network.latestBlock$, + storedBlockLogs$: result.network.storedBlockLogs$, + worldAddress: result.network.worldContract.address, + worldAbi: result.network.worldContract.abi, + write$: result.network.write$, + useStore: result.network.useStore, + }); + } +}); diff --git a/examples/multiple-accounts/packages/client/src/mud/createSystemCalls.ts b/examples/multiple-accounts/packages/client/src/mud/createSystemCalls.ts new file mode 100644 index 0000000000..0b8c753a67 --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/mud/createSystemCalls.ts @@ -0,0 +1,13 @@ +/* + * Create the system calls that the client can use to ask + * for changes in the World state (using the System contracts). + */ + +import { Hex } from "viem"; +import { SetupNetworkResult } from "./setupNetwork"; + +export type SystemCalls = ReturnType; + +export function createSystemCalls({ tables, useStore, worldContract, waitForTransaction }: SetupNetworkResult) { + return {}; +} diff --git a/examples/multiple-accounts/packages/client/src/mud/getNetworkConfig.ts b/examples/multiple-accounts/packages/client/src/mud/getNetworkConfig.ts new file mode 100644 index 0000000000..61ad962f25 --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/mud/getNetworkConfig.ts @@ -0,0 +1,91 @@ +/* + * Network specific configuration for the client. + * By default connect to the anvil test network. + * + */ + +/* + * By default the template just creates a temporary wallet + * (called a burner wallet) and uses a faucet (on our test net) + * to get ETH for it. + * + * See https://mud.dev/tutorials/minimal/deploy#wallet-managed-address + * for how to use the user's own address instead. + */ +import { getBurnerPrivateKey } from "@latticexyz/common"; + +/* + * Import the addresses of the World, possibly on multiple chains, + * from packages/contracts/worlds.json. When the contracts package + * deploys a new `World`, it updates this file. + */ +import worlds from "contracts/worlds.json"; + +/* + * The supported chains. + * By default, there are only two chains here: + * + * - mudFoundry, the chain running on anvil that pnpm dev + * starts by default. It is similar to the viem anvil chain + * (see https://viem.sh/docs/clients/test.html), but with the + * basefee set to zero to avoid transaction fees. + * - latticeTestnet, our public test network. + * + * See https://mud.dev/tutorials/minimal/deploy#run-the-user-interface + * for instructions on how to add networks. + */ +import { supportedChains } from "./supportedChains"; + +export async function getNetworkConfig() { + const params = new URLSearchParams(window.location.search); + + /* + * The chain ID is the first item available from this list: + * 1. chainId query parameter + * 2. chainid query parameter + * 3. The VITE_CHAIN_ID environment variable set when the + * vite dev server was started or client was built + * 4. The default, 31337 (anvil) + */ + const chainId = Number(params.get("chainId") || params.get("chainid") || import.meta.env.VITE_CHAIN_ID || 31337); + + /* + * Find the chain (unless it isn't in the list of supported chains). + */ + const chainIndex = supportedChains.findIndex((c) => c.id === chainId); + const chain = supportedChains[chainIndex]; + if (!chain) { + throw new Error(`Chain ${chainId} not found`); + } + + /* + * Get the address of the World. If you want to use a + * different address than the one in worlds.json, + * provide it as worldAddress in the query string. + */ + const world = worlds[chain.id.toString()]; + const worldAddress = params.get("worldAddress") || world?.address; + if (!worldAddress) { + throw new Error(`No world address found for chain ${chainId}. Did you run \`mud deploy\`?`); + } + + /* + * MUD clients use events to synchronize the database, meaning + * they need to look as far back as when the World was started. + * The block number for the World start can be specified either + * on the URL (as initialBlockNumber) or in the worlds.json + * file. If neither has it, it starts at the first block, zero. + */ + const initialBlockNumber = params.has("initialBlockNumber") + ? Number(params.get("initialBlockNumber")) + : world?.blockNumber ?? 0n; + + return { + privateKey: getBurnerPrivateKey(), + chainId, + chain, + faucetServiceUrl: params.get("faucet") ?? chain.faucetUrl, + worldAddress, + initialBlockNumber, + }; +} diff --git a/examples/multiple-accounts/packages/client/src/mud/setup.ts b/examples/multiple-accounts/packages/client/src/mud/setup.ts new file mode 100644 index 0000000000..6d07134612 --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/mud/setup.ts @@ -0,0 +1,18 @@ +/* + * This file sets up all the definitions required for a MUD client. + */ + +import { createSystemCalls } from "./createSystemCalls"; +import { setupNetwork } from "./setupNetwork"; + +export type SetupResult = Awaited>; + +export async function setup() { + const network = await setupNetwork(); + const systemCalls = createSystemCalls(network); + + return { + network, + systemCalls, + }; +} diff --git a/examples/multiple-accounts/packages/client/src/mud/setupNetwork.ts b/examples/multiple-accounts/packages/client/src/mud/setupNetwork.ts new file mode 100644 index 0000000000..5cd26c31b1 --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/mud/setupNetwork.ts @@ -0,0 +1,120 @@ +/* + * The MUD client code is built on top of viem + * (https://viem.sh/docs/getting-started.html). + * This line imports the functions we need from it. + */ +import { createPublicClient, fallback, webSocket, http, createWalletClient, Hex, parseEther, ClientConfig } from "viem"; +import { createFaucetService } from "@latticexyz/services/faucet"; +import { syncToZustand } from "@latticexyz/store-sync/zustand"; +import { getNetworkConfig } from "./getNetworkConfig"; +import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { Subject, share } from "rxjs"; + +/* + * Import our MUD config, which includes strong types for + * our tables and other config options. We use this to generate + * things like RECS components and get back strong types for them. + * + * See https://mud.dev/templates/typescript/contracts#mudconfigts + * for the source of this information. + */ +import mudConfig from "contracts/mud.config"; + +export type SetupNetworkResult = Awaited>; + +export async function setupNetwork() { + const networkConfig = await getNetworkConfig(); + + /* + * Create a viem public (read only) client + * (https://viem.sh/docs/clients/public.html) + */ + const clientOptions = { + chain: networkConfig.chain, + transport: transportObserver(fallback([webSocket(), http()])), + pollingInterval: 1000, + } as const satisfies ClientConfig; + + const publicClient = createPublicClient(clientOptions); + + /* + * Create a temporary wallet and a viem client for it + * (see https://viem.sh/docs/clients/wallet.html). + */ + const burnerAccount = createBurnerAccount(networkConfig.privateKey as Hex); + const burnerWalletClient = createWalletClient({ + ...clientOptions, + account: burnerAccount, + }); + + /* + * Create an observable for contract writes that we can + * pass into MUD dev tools for transaction observability. + */ + const write$ = new Subject(); + + /* + * Create an object for communicating with the deployed World. + */ + const worldContract = getContract({ + address: networkConfig.worldAddress as Hex, + abi: IWorldAbi, + publicClient, + walletClient: burnerWalletClient, + onWrite: (write) => write$.next(write), + }); + + /* + * Sync on-chain state into RECS and keeps our client in sync. + * Uses the MUD indexer if available, otherwise falls back + * to the viem publicClient to make RPC calls to fetch MUD + * events from the chain. + */ + const { tables, useStore, latestBlock$, storedBlockLogs$, waitForTransaction } = await syncToZustand({ + config: mudConfig, + address: networkConfig.worldAddress as Hex, + publicClient, + startBlock: BigInt(networkConfig.initialBlockNumber), + }); + + /* + * If there is a faucet, request (test) ETH if you have + * less than 1 ETH. Repeat every 20 seconds to ensure you don't + * run out. + */ + if (networkConfig.faucetServiceUrl) { + const address = burnerAccount.address; + console.info("[Dev Faucet]: Player address -> ", address); + + const faucet = createFaucetService(networkConfig.faucetServiceUrl); + + const requestDrip = async () => { + const balance = await publicClient.getBalance({ address }); + console.info(`[Dev Faucet]: Player balance -> ${balance}`); + const lowBalance = balance < parseEther("1"); + if (lowBalance) { + console.info("[Dev Faucet]: Balance is low, dripping funds to player"); + // Double drip + await faucet.dripDev({ address }); + await faucet.dripDev({ address }); + } + }; + + requestDrip(); + // Request a drip every 20 seconds + setInterval(requestDrip, 20000); + } + + return { + tables, + useStore, + publicClient, + walletClient: burnerWalletClient, + latestBlock$, + storedBlockLogs$, + waitForTransaction, + worldContract, + write$: write$.asObservable().pipe(share()), + }; +} diff --git a/examples/multiple-accounts/packages/client/src/mud/supportedChains.ts b/examples/multiple-accounts/packages/client/src/mud/supportedChains.ts new file mode 100644 index 0000000000..614412500e --- /dev/null +++ b/examples/multiple-accounts/packages/client/src/mud/supportedChains.ts @@ -0,0 +1,20 @@ +/* + * The supported chains. + * By default, there are only two chains here: + * + * - mudFoundry, the chain running on anvil that pnpm dev + * starts by default. It is similar to the viem anvil chain + * (see https://viem.sh/docs/clients/test.html), but with the + * basefee set to zero to avoid transaction fees. + * - latticeTestnet, our public test network. + * + + */ + +import { MUDChain, latticeTestnet, mudFoundry } from "@latticexyz/common/chains"; + +/* + * See https://mud.dev/tutorials/minimal/deploy#run-the-user-interface + * for instructions on how to add networks. + */ +export const supportedChains: MUDChain[] = [mudFoundry, latticeTestnet]; diff --git a/examples/multiple-accounts/packages/client/tsconfig.json b/examples/multiple-accounts/packages/client/tsconfig.json new file mode 100644 index 0000000000..5bf43bace8 --- /dev/null +++ b/examples/multiple-accounts/packages/client/tsconfig.json @@ -0,0 +1,19 @@ +{ + "compilerOptions": { + "types": ["vite/client"], + "target": "ESNext", + "useDefineForClassFields": true, + "module": "ESNext", + "lib": ["ESNext", "DOM"], + "moduleResolution": "Node", + "strict": true, + "resolveJsonModule": true, + "isolatedModules": true, + "esModuleInterop": true, + "noEmit": true, + "skipLibCheck": true, + "jsx": "react-jsx", + "jsxImportSource": "react" + }, + "include": ["src"] +} diff --git a/examples/multiple-accounts/packages/client/vite.config.ts b/examples/multiple-accounts/packages/client/vite.config.ts new file mode 100644 index 0000000000..a47d70e7a1 --- /dev/null +++ b/examples/multiple-accounts/packages/client/vite.config.ts @@ -0,0 +1,17 @@ +import { defineConfig } from "vite"; +import react from "@vitejs/plugin-react"; + +export default defineConfig({ + plugins: [react()], + server: { + port: 3000, + fs: { + strict: false, + }, + }, + build: { + target: "es2022", + minify: true, + sourcemap: true, + }, +}); diff --git a/examples/multiple-accounts/packages/contracts/.gitignore b/examples/multiple-accounts/packages/contracts/.gitignore new file mode 100644 index 0000000000..aea4e54c82 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/.gitignore @@ -0,0 +1,9 @@ +out/ +cache/ +node_modules/ +bindings/ +artifacts/ +broadcast/ + +# Ignore MUD deploy artifacts +deploys/**/*.json diff --git a/examples/multiple-accounts/packages/contracts/.prettierrc b/examples/multiple-accounts/packages/contracts/.prettierrc new file mode 100644 index 0000000000..bf689e8c48 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/.prettierrc @@ -0,0 +1,8 @@ +{ + "plugins": ["prettier-plugin-solidity"], + "printWidth": 120, + "semi": true, + "tabWidth": 2, + "useTabs": false, + "bracketSpacing": true +} diff --git a/examples/multiple-accounts/packages/contracts/.solhint.json b/examples/multiple-accounts/packages/contracts/.solhint.json new file mode 100644 index 0000000000..f3e0b01ffc --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/.solhint.json @@ -0,0 +1,12 @@ +{ + "extends": ["solhint:recommended", "mud"], + "plugins": ["mud"], + "rules": { + "compiler-version": ["error", ">=0.8.0"], + "avoid-low-level-calls": "off", + "no-inline-assembly": "off", + "func-visibility": ["warn", { "ignoreConstructors": true }], + "no-empty-blocks": "off", + "no-complex-fallback": "off" + } +} diff --git a/examples/multiple-accounts/packages/contracts/foundry.toml b/examples/multiple-accounts/packages/contracts/foundry.toml new file mode 100644 index 0000000000..d03fad8043 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/foundry.toml @@ -0,0 +1,26 @@ +[profile.default] +solc = "0.8.21" +ffi = false +fuzz_runs = 256 +optimizer = true +optimizer_runs = 3000 +verbosity = 2 +src = "src" +test = "test" +out = "out" +allow_paths = [ + # pnpm symlinks to the project root's node_modules + "../../node_modules", + # template uses linked mud packages from within the mud monorepo + "../../../../packages", + # projects created from this template and using linked mud packages + "../../../mud/packages", +] +extra_output_files = [ + "abi", + "evm.bytecode" +] +fs_permissions = [{ access = "read", path = "./"}] + +[profile.lattice-testnet] +eth_rpc_url = "https://follower.testnet-chain.linfra.xyz" diff --git a/examples/multiple-accounts/packages/contracts/mud.config.ts b/examples/multiple-accounts/packages/contracts/mud.config.ts new file mode 100644 index 0000000000..3ba46294c4 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/mud.config.ts @@ -0,0 +1,15 @@ +import { mudConfig } from "@latticexyz/world/register"; + +export default mudConfig({ + namespace: "LastCall", + tables: { + LastCall: { + keySchema: { + caller: "address", + }, + valueSchema: { + callTime: "uint256", + }, + }, + }, +}); diff --git a/examples/multiple-accounts/packages/contracts/package.json b/examples/multiple-accounts/packages/contracts/package.json new file mode 100644 index 0000000000..78dd852e30 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/package.json @@ -0,0 +1,34 @@ +{ + "name": "contracts", + "version": "0.0.0", + "private": true, + "license": "MIT", + "scripts": { + "build": "mud build", + "clean": "forge clean && rimraf src/codegen", + "deploy:local": "pnpm run build && mud deploy", + "deploy:testnet": "pnpm run build && mud deploy --profile=lattice-testnet", + "dev": "pnpm mud dev-contracts", + "lint": "pnpm run prettier && pnpm run solhint", + "prettier": "prettier --write 'src/**/*.sol'", + "solhint": "solhint --config ./.solhint.json 'src/**/*.sol' --fix", + "test": "tsc --noEmit && mud test" + }, + "dependencies": { + "@latticexyz/cli": "2.0.0-next.15", + "@latticexyz/schema-type": "2.0.0-next.15", + "@latticexyz/store": "2.0.0-next.15", + "@latticexyz/world": "2.0.0-next.15", + "@latticexyz/world-modules": "2.0.0-next.15" + }, + "devDependencies": { + "@types/node": "^18.15.11", + "ds-test": "https://github.com/dapphub/ds-test.git#e282159d5170298eb2455a6c05280ab5a73a4ef0", + "forge-std": "https://github.com/foundry-rs/forge-std.git#74cfb77e308dd188d2f58864aaf44963ae6b88b1", + "prettier": "^2.6.2", + "prettier-plugin-solidity": "1.1.3", + "solhint": "^3.3.7", + "solhint-config-mud": "2.0.0-next.15", + "solhint-plugin-mud": "2.0.0-next.15" + } +} diff --git a/examples/multiple-accounts/packages/contracts/remappings.txt b/examples/multiple-accounts/packages/contracts/remappings.txt new file mode 100644 index 0000000000..c4d992480e --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/remappings.txt @@ -0,0 +1,3 @@ +ds-test/=node_modules/ds-test/src/ +forge-std/=node_modules/forge-std/src/ +@latticexyz/=node_modules/@latticexyz/ diff --git a/examples/multiple-accounts/packages/contracts/script/PostDeploy.s.sol b/examples/multiple-accounts/packages/contracts/script/PostDeploy.s.sol new file mode 100644 index 0000000000..de01099e0a --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/script/PostDeploy.s.sol @@ -0,0 +1,27 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +import { Script } from "forge-std/Script.sol"; +import { console } from "forge-std/console.sol"; +import { StoreSwitch } from "@latticexyz/store/src/StoreSwitch.sol"; + +import { IWorld } from "../src/codegen/world/IWorld.sol"; +import { LastCall } from "../src/codegen/index.sol"; + +contract PostDeploy is Script { + function run(address worldAddress) external { + // Specify a store so that you can use tables directly in PostDeploy + StoreSwitch.setStoreAddress(worldAddress); + + // Load the private key from the `PRIVATE_KEY` environment variable (in .env) + uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + + // Start broadcasting transactions from the deployer account + vm.startBroadcast(deployerPrivateKey); + + // Add a last call + IWorld(worldAddress).LastCall_LastCallSystem_newCall(); + + vm.stopBroadcast(); + } +} diff --git a/examples/multiple-accounts/packages/contracts/src/codegen/index.sol b/examples/multiple-accounts/packages/contracts/src/codegen/index.sol new file mode 100644 index 0000000000..525fad9dae --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/src/codegen/index.sol @@ -0,0 +1,6 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +/* Autogenerated file. Do not edit manually. */ + +import { LastCall, LastCallTableId } from "./tables/LastCall.sol"; diff --git a/examples/multiple-accounts/packages/contracts/src/codegen/tables/LastCall.sol b/examples/multiple-accounts/packages/contracts/src/codegen/tables/LastCall.sol new file mode 100644 index 0000000000..f22ae9bd64 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/src/codegen/tables/LastCall.sol @@ -0,0 +1,231 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +/* Autogenerated file. Do not edit manually. */ + +// Import schema type +import { SchemaType } from "@latticexyz/schema-type/src/solidity/SchemaType.sol"; + +// Import store internals +import { IStore } from "@latticexyz/store/src/IStore.sol"; +import { StoreSwitch } from "@latticexyz/store/src/StoreSwitch.sol"; +import { StoreCore } from "@latticexyz/store/src/StoreCore.sol"; +import { Bytes } from "@latticexyz/store/src/Bytes.sol"; +import { Memory } from "@latticexyz/store/src/Memory.sol"; +import { SliceLib } from "@latticexyz/store/src/Slice.sol"; +import { EncodeArray } from "@latticexyz/store/src/tightcoder/EncodeArray.sol"; +import { FieldLayout, FieldLayoutLib } from "@latticexyz/store/src/FieldLayout.sol"; +import { Schema, SchemaLib } from "@latticexyz/store/src/Schema.sol"; +import { PackedCounter, PackedCounterLib } from "@latticexyz/store/src/PackedCounter.sol"; +import { ResourceId } from "@latticexyz/store/src/ResourceId.sol"; +import { RESOURCE_TABLE, RESOURCE_OFFCHAIN_TABLE } from "@latticexyz/store/src/storeResourceTypes.sol"; + +ResourceId constant _tableId = ResourceId.wrap( + bytes32(abi.encodePacked(RESOURCE_TABLE, bytes14("LastCall"), bytes16("LastCall"))) +); +ResourceId constant LastCallTableId = _tableId; + +FieldLayout constant _fieldLayout = FieldLayout.wrap( + 0x0020010020000000000000000000000000000000000000000000000000000000 +); + +library LastCall { + /** + * @notice Get the table values' field layout. + * @return _fieldLayout The field layout for the table. + */ + function getFieldLayout() internal pure returns (FieldLayout) { + return _fieldLayout; + } + + /** + * @notice Get the table's key schema. + * @return _keySchema The key schema for the table. + */ + function getKeySchema() internal pure returns (Schema) { + SchemaType[] memory _keySchema = new SchemaType[](1); + _keySchema[0] = SchemaType.ADDRESS; + + return SchemaLib.encode(_keySchema); + } + + /** + * @notice Get the table's value schema. + * @return _valueSchema The value schema for the table. + */ + function getValueSchema() internal pure returns (Schema) { + SchemaType[] memory _valueSchema = new SchemaType[](1); + _valueSchema[0] = SchemaType.UINT256; + + return SchemaLib.encode(_valueSchema); + } + + /** + * @notice Get the table's key field names. + * @return keyNames An array of strings with the names of key fields. + */ + function getKeyNames() internal pure returns (string[] memory keyNames) { + keyNames = new string[](1); + keyNames[0] = "caller"; + } + + /** + * @notice Get the table's value field names. + * @return fieldNames An array of strings with the names of value fields. + */ + function getFieldNames() internal pure returns (string[] memory fieldNames) { + fieldNames = new string[](1); + fieldNames[0] = "callTime"; + } + + /** + * @notice Register the table with its config. + */ + function register() internal { + StoreSwitch.registerTable(_tableId, _fieldLayout, getKeySchema(), getValueSchema(), getKeyNames(), getFieldNames()); + } + + /** + * @notice Register the table with its config. + */ + function _register() internal { + StoreCore.registerTable(_tableId, _fieldLayout, getKeySchema(), getValueSchema(), getKeyNames(), getFieldNames()); + } + + /** + * @notice Get callTime. + */ + function getCallTime(address caller) internal view returns (uint256 callTime) { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + bytes32 _blob = StoreSwitch.getStaticField(_tableId, _keyTuple, 0, _fieldLayout); + return (uint256(bytes32(_blob))); + } + + /** + * @notice Get callTime. + */ + function _getCallTime(address caller) internal view returns (uint256 callTime) { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + bytes32 _blob = StoreCore.getStaticField(_tableId, _keyTuple, 0, _fieldLayout); + return (uint256(bytes32(_blob))); + } + + /** + * @notice Get callTime. + */ + function get(address caller) internal view returns (uint256 callTime) { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + bytes32 _blob = StoreSwitch.getStaticField(_tableId, _keyTuple, 0, _fieldLayout); + return (uint256(bytes32(_blob))); + } + + /** + * @notice Get callTime. + */ + function _get(address caller) internal view returns (uint256 callTime) { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + bytes32 _blob = StoreCore.getStaticField(_tableId, _keyTuple, 0, _fieldLayout); + return (uint256(bytes32(_blob))); + } + + /** + * @notice Set callTime. + */ + function setCallTime(address caller, uint256 callTime) internal { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + StoreSwitch.setStaticField(_tableId, _keyTuple, 0, abi.encodePacked((callTime)), _fieldLayout); + } + + /** + * @notice Set callTime. + */ + function _setCallTime(address caller, uint256 callTime) internal { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + StoreCore.setStaticField(_tableId, _keyTuple, 0, abi.encodePacked((callTime)), _fieldLayout); + } + + /** + * @notice Set callTime. + */ + function set(address caller, uint256 callTime) internal { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + StoreSwitch.setStaticField(_tableId, _keyTuple, 0, abi.encodePacked((callTime)), _fieldLayout); + } + + /** + * @notice Set callTime. + */ + function _set(address caller, uint256 callTime) internal { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + StoreCore.setStaticField(_tableId, _keyTuple, 0, abi.encodePacked((callTime)), _fieldLayout); + } + + /** + * @notice Delete all data for given keys. + */ + function deleteRecord(address caller) internal { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + StoreSwitch.deleteRecord(_tableId, _keyTuple); + } + + /** + * @notice Delete all data for given keys. + */ + function _deleteRecord(address caller) internal { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + StoreCore.deleteRecord(_tableId, _keyTuple, _fieldLayout); + } + + /** + * @notice Tightly pack static (fixed length) data using this table's schema. + * @return The static data, encoded into a sequence of bytes. + */ + function encodeStatic(uint256 callTime) internal pure returns (bytes memory) { + return abi.encodePacked(callTime); + } + + /** + * @notice Encode all of a record's fields. + * @return The static (fixed length) data, encoded into a sequence of bytes. + * @return The lengths of the dynamic fields (packed into a single bytes32 value). + * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + */ + function encode(uint256 callTime) internal pure returns (bytes memory, PackedCounter, bytes memory) { + bytes memory _staticData = encodeStatic(callTime); + + PackedCounter _encodedLengths; + bytes memory _dynamicData; + + return (_staticData, _encodedLengths, _dynamicData); + } + + /** + * @notice Encode keys as a bytes32 array using this table's field layout. + */ + function encodeKeyTuple(address caller) internal pure returns (bytes32[] memory) { + bytes32[] memory _keyTuple = new bytes32[](1); + _keyTuple[0] = bytes32(uint256(uint160(caller))); + + return _keyTuple; + } +} diff --git a/examples/multiple-accounts/packages/contracts/src/codegen/world/ILastCallSystem.sol b/examples/multiple-accounts/packages/contracts/src/codegen/world/ILastCallSystem.sol new file mode 100644 index 0000000000..ea46b83c17 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/src/codegen/world/ILastCallSystem.sol @@ -0,0 +1,12 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +/* Autogenerated file. Do not edit manually. */ + +/** + * @title ILastCallSystem + * @dev This interface is automatically generated from the corresponding system contract. Do not edit manually. + */ +interface ILastCallSystem { + function LastCall_LastCallSystem_newCall() external; +} diff --git a/examples/multiple-accounts/packages/contracts/src/codegen/world/ITasksSystem.sol b/examples/multiple-accounts/packages/contracts/src/codegen/world/ITasksSystem.sol new file mode 100644 index 0000000000..e9bfddc79e --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/src/codegen/world/ITasksSystem.sol @@ -0,0 +1,18 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +/* Autogenerated file. Do not edit manually. */ + +/** + * @title ITasksSystem + * @dev This interface is automatically generated from the corresponding system contract. Do not edit manually. + */ +interface ITasksSystem { + function addTask(string memory description) external returns (bytes32 key); + + function completeTask(bytes32 key) external; + + function resetTask(bytes32 key) external; + + function deleteTask(bytes32 key) external; +} diff --git a/examples/multiple-accounts/packages/contracts/src/codegen/world/IWorld.sol b/examples/multiple-accounts/packages/contracts/src/codegen/world/IWorld.sol new file mode 100644 index 0000000000..35618e1257 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/src/codegen/world/IWorld.sol @@ -0,0 +1,18 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +/* Autogenerated file. Do not edit manually. */ + +import { IBaseWorld } from "@latticexyz/world/src/codegen/interfaces/IBaseWorld.sol"; + +import { ILastCallSystem } from "./ILastCallSystem.sol"; + +/** + * @title IWorld + * @notice This interface integrates all systems and associated function selectors + * that are dynamically registered in the World during deployment. + * @dev This is an autogenerated file; do not edit manually. + */ +interface IWorld is IBaseWorld, ILastCallSystem { + +} diff --git a/examples/multiple-accounts/packages/contracts/src/systems/LastCallSystem.sol b/examples/multiple-accounts/packages/contracts/src/systems/LastCallSystem.sol new file mode 100644 index 0000000000..e40df89975 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/src/systems/LastCallSystem.sol @@ -0,0 +1,11 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.21; + +import { System } from "@latticexyz/world/src/System.sol"; +import { LastCall } from "../codegen/index.sol"; + +contract LastCallSystem is System { + function newCall() public { + LastCall.set(_msgSender(), block.timestamp); + } +} diff --git a/examples/multiple-accounts/packages/contracts/tsconfig.json b/examples/multiple-accounts/packages/contracts/tsconfig.json new file mode 100644 index 0000000000..270db8fdf6 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/tsconfig.json @@ -0,0 +1,13 @@ +// Visit https://aka.ms/tsconfig.json for all config options +{ + "compilerOptions": { + "target": "ES2020", + "module": "commonjs", + "strict": true, + "resolveJsonModule": true, + "esModuleInterop": true, + "skipLibCheck": true, + "forceConsistentCasingInFileNames": true, + "moduleResolution": "node" + } +} diff --git a/examples/multiple-accounts/packages/contracts/worlds.json b/examples/multiple-accounts/packages/contracts/worlds.json new file mode 100644 index 0000000000..906845ce6d --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/worlds.json @@ -0,0 +1,5 @@ +{ + "31337": { + "address": "0xc44504ab6a2c4df9a9ce82aecfc453fec3c8771c" + } +} \ No newline at end of file diff --git a/examples/multiple-accounts/packages/contracts/worlds.json.d.ts b/examples/multiple-accounts/packages/contracts/worlds.json.d.ts new file mode 100644 index 0000000000..90ffc786f4 --- /dev/null +++ b/examples/multiple-accounts/packages/contracts/worlds.json.d.ts @@ -0,0 +1,2 @@ +declare const worlds: Partial>; +export default worlds; diff --git a/examples/multiple-accounts/pnpm-lock.yaml b/examples/multiple-accounts/pnpm-lock.yaml new file mode 100644 index 0000000000..3a42631ed8 --- /dev/null +++ b/examples/multiple-accounts/pnpm-lock.yaml @@ -0,0 +1,4323 @@ +lockfileVersion: '6.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +importers: + + .: + devDependencies: + '@latticexyz/cli': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(google-protobuf@3.21.2) + '@types/debug': + specifier: 4.1.7 + version: 4.1.7 + '@types/prettier': + specifier: 2.7.2 + version: 2.7.2 + '@typescript-eslint/eslint-plugin': + specifier: 5.46.1 + version: 5.46.1(@typescript-eslint/parser@5.46.1)(eslint@8.29.0)(typescript@5.1.6) + '@typescript-eslint/parser': + specifier: 5.46.1 + version: 5.46.1(eslint@8.29.0)(typescript@5.1.6) + eslint: + specifier: 8.29.0 + version: 8.29.0 + mprocs: + specifier: ^0.6.4 + version: 0.6.4 + rimraf: + specifier: ^3.0.2 + version: 3.0.2 + typescript: + specifier: 5.1.6 + version: 5.1.6 + + packages/client: + dependencies: + '@latticexyz/common': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/dev-tools': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(@latticexyz/common@2.0.0-next.15)(@latticexyz/recs@2.0.0-next.15)(@latticexyz/store-sync@2.0.0-next.15)(@latticexyz/store@2.0.0-next.15)(@latticexyz/utils@2.0.0-next.15)(@latticexyz/world@2.0.0-next.15)(@types/react@18.2.22)(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/react': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/schema-type': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/services': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(google-protobuf@3.21.2) + '@latticexyz/store-sync': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(@types/react@18.2.22)(react@18.2.0)(typescript@5.1.6) + '@latticexyz/utils': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15 + '@latticexyz/world': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6) + contracts: + specifier: workspace:* + version: link:../contracts + react: + specifier: ^18.2.0 + version: 18.2.0 + react-dom: + specifier: ^18.2.0 + version: 18.2.0(react@18.2.0) + rxjs: + specifier: 7.5.5 + version: 7.5.5 + viem: + specifier: 1.14.0 + version: 1.14.0(typescript@5.1.6)(zod@3.22.4) + devDependencies: + '@types/react': + specifier: 18.2.22 + version: 18.2.22 + '@types/react-dom': + specifier: 18.2.7 + version: 18.2.7 + '@vitejs/plugin-react': + specifier: ^3.1.0 + version: 3.1.0(vite@4.5.1) + eslint-plugin-react: + specifier: 7.31.11 + version: 7.31.11(eslint@8.29.0) + eslint-plugin-react-hooks: + specifier: 4.6.0 + version: 4.6.0(eslint@8.29.0) + vite: + specifier: ^4.2.1 + version: 4.5.1 + wait-port: + specifier: ^1.0.4 + version: 1.1.0 + + packages/contracts: + dependencies: + '@latticexyz/cli': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(google-protobuf@3.21.2) + '@latticexyz/schema-type': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/store': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/world': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/world-modules': + specifier: 2.0.0-next.15 + version: 2.0.0-next.15(typescript@5.1.6) + devDependencies: + '@types/node': + specifier: ^18.15.11 + version: 18.19.6 + ds-test: + specifier: https://github.com/dapphub/ds-test.git#e282159d5170298eb2455a6c05280ab5a73a4ef0 + version: github.com/dapphub/ds-test/e282159d5170298eb2455a6c05280ab5a73a4ef0 + forge-std: + specifier: https://github.com/foundry-rs/forge-std.git#74cfb77e308dd188d2f58864aaf44963ae6b88b1 + version: github.com/foundry-rs/forge-std/74cfb77e308dd188d2f58864aaf44963ae6b88b1 + prettier: + specifier: ^2.6.2 + version: 2.8.8 + prettier-plugin-solidity: + specifier: 1.1.3 + version: 1.1.3(prettier@2.8.8) + solhint: + specifier: ^3.3.7 + version: 3.6.2(typescript@5.1.6) + solhint-config-mud: + specifier: 2.0.0-next.15 + version: 2.0.0-next.15 + solhint-plugin-mud: + specifier: 2.0.0-next.15 + version: 2.0.0-next.15 + +packages: + + /@aashutoshrathi/word-wrap@1.2.6: + resolution: {integrity: sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==} + engines: {node: '>=0.10.0'} + dev: true + + /@adraffy/ens-normalize@1.9.4: + resolution: {integrity: sha512-UK0bHA7hh9cR39V+4gl2/NnBBjoXIxkuWAPCaY4X7fbH4L/azIi7ilWOCjMUYfpJgraLUAqkRi2BqrjME8Rynw==} + + /@ampproject/remapping@2.2.1: + resolution: {integrity: sha512-lFMjJTrFL3j7L9yBxwYfCq2k6qqwHyzuUl/XBnif78PWTJYyL/dfowQHWE3sp6U6ZzqWiiIZnpTMO96zhkjwtg==} + engines: {node: '>=6.0.0'} + dependencies: + '@jridgewell/gen-mapping': 0.3.3 + '@jridgewell/trace-mapping': 0.3.20 + dev: true + + /@babel/code-frame@7.23.5: + resolution: {integrity: sha512-CgH3s1a96LipHCmSUmYFPwY7MNx8C3avkq7i4Wl3cfa662ldtUe4VM1TPXX70pfmrlWTb6jLqTYrZyT2ZTJBgA==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/highlight': 7.23.4 + chalk: 2.4.2 + dev: true + + /@babel/compat-data@7.23.5: + resolution: {integrity: sha512-uU27kfDRlhfKl+w1U6vp16IuvSLtjAxdArVXPa9BvLkrr7CYIsxH5adpHObeAGY/41+syctUWOZ140a2Rvkgjw==} + engines: {node: '>=6.9.0'} + dev: true + + /@babel/core@7.23.7: + resolution: {integrity: sha512-+UpDgowcmqe36d4NwqvKsyPMlOLNGMsfMmQ5WGCu+siCe3t3dfe9njrzGfdN4qq+bcNUt0+Vw6haRxBOycs4dw==} + engines: {node: '>=6.9.0'} + dependencies: + '@ampproject/remapping': 2.2.1 + '@babel/code-frame': 7.23.5 + '@babel/generator': 7.23.6 + '@babel/helper-compilation-targets': 7.23.6 + '@babel/helper-module-transforms': 7.23.3(@babel/core@7.23.7) + '@babel/helpers': 7.23.8 + '@babel/parser': 7.23.6 + '@babel/template': 7.22.15 + '@babel/traverse': 7.23.7 + '@babel/types': 7.23.6 + convert-source-map: 2.0.0 + debug: 4.3.4 + gensync: 1.0.0-beta.2 + json5: 2.2.3 + semver: 6.3.1 + transitivePeerDependencies: + - supports-color + dev: true + + /@babel/generator@7.23.6: + resolution: {integrity: sha512-qrSfCYxYQB5owCmGLbl8XRpX1ytXlpueOb0N0UmQwA073KZxejgQTzAmJezxvpwQD9uGtK2shHdi55QT+MbjIw==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.6 + '@jridgewell/gen-mapping': 0.3.3 + '@jridgewell/trace-mapping': 0.3.20 + jsesc: 2.5.2 + dev: true + + /@babel/helper-compilation-targets@7.23.6: + resolution: {integrity: sha512-9JB548GZoQVmzrFgp8o7KxdgkTGm6xs9DW0o/Pim72UDjzr5ObUQ6ZzYPqA+g9OTS2bBQoctLJrky0RDCAWRgQ==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/compat-data': 7.23.5 + '@babel/helper-validator-option': 7.23.5 + browserslist: 4.22.2 + lru-cache: 5.1.1 + semver: 6.3.1 + dev: true + + /@babel/helper-environment-visitor@7.22.20: + resolution: {integrity: sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==} + engines: {node: '>=6.9.0'} + dev: true + + /@babel/helper-function-name@7.23.0: + resolution: {integrity: sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/template': 7.22.15 + '@babel/types': 7.23.6 + dev: true + + /@babel/helper-hoist-variables@7.22.5: + resolution: {integrity: sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.6 + dev: true + + /@babel/helper-module-imports@7.22.15: + resolution: {integrity: sha512-0pYVBnDKZO2fnSPCrgM/6WMc7eS20Fbok+0r88fp+YtWVLZrp4CkafFGIp+W0VKw4a22sgebPT99y+FDNMdP4w==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.6 + dev: true + + /@babel/helper-module-transforms@7.23.3(@babel/core@7.23.7): + resolution: {integrity: sha512-7bBs4ED9OmswdfDzpz4MpWgSrV7FXlc3zIagvLFjS5H+Mk7Snr21vQ6QwrsoCGMfNC4e4LQPdoULEt4ykz0SRQ==} + engines: {node: '>=6.9.0'} + peerDependencies: + '@babel/core': ^7.0.0 + dependencies: + '@babel/core': 7.23.7 + '@babel/helper-environment-visitor': 7.22.20 + '@babel/helper-module-imports': 7.22.15 + '@babel/helper-simple-access': 7.22.5 + '@babel/helper-split-export-declaration': 7.22.6 + '@babel/helper-validator-identifier': 7.22.20 + dev: true + + /@babel/helper-plugin-utils@7.22.5: + resolution: {integrity: sha512-uLls06UVKgFG9QD4OeFYLEGteMIAa5kpTPcFL28yuCIIzsf6ZyKZMllKVOCZFhiZ5ptnwX4mtKdWCBE/uT4amg==} + engines: {node: '>=6.9.0'} + dev: true + + /@babel/helper-simple-access@7.22.5: + resolution: {integrity: sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.6 + dev: true + + /@babel/helper-split-export-declaration@7.22.6: + resolution: {integrity: sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.6 + dev: true + + /@babel/helper-string-parser@7.23.4: + resolution: {integrity: sha512-803gmbQdqwdf4olxrX4AJyFBV/RTr3rSmOj0rKwesmzlfhYNDEs+/iOcznzpNWlJlIlTJC2QfPFcHB6DlzdVLQ==} + engines: {node: '>=6.9.0'} + dev: true + + /@babel/helper-validator-identifier@7.22.20: + resolution: {integrity: sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==} + engines: {node: '>=6.9.0'} + dev: true + + /@babel/helper-validator-option@7.23.5: + resolution: {integrity: sha512-85ttAOMLsr53VgXkTbkx8oA6YTfT4q7/HzXSLEYmjcSTJPMPQtvq1BD79Byep5xMUYbGRzEpDsjUf3dyp54IKw==} + engines: {node: '>=6.9.0'} + dev: true + + /@babel/helpers@7.23.8: + resolution: {integrity: sha512-KDqYz4PiOWvDFrdHLPhKtCThtIcKVy6avWD2oG4GEvyQ+XDZwHD4YQd+H2vNMnq2rkdxsDkU82T+Vk8U/WXHRQ==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/template': 7.22.15 + '@babel/traverse': 7.23.7 + '@babel/types': 7.23.6 + transitivePeerDependencies: + - supports-color + dev: true + + /@babel/highlight@7.23.4: + resolution: {integrity: sha512-acGdbYSfp2WheJoJm/EBBBLh/ID8KDc64ISZ9DYtBmC8/Q204PZJLHyzeB5qMzJ5trcOkybd78M4x2KWsUq++A==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/helper-validator-identifier': 7.22.20 + chalk: 2.4.2 + js-tokens: 4.0.0 + dev: true + + /@babel/parser@7.23.6: + resolution: {integrity: sha512-Z2uID7YJ7oNvAI20O9X0bblw7Qqs8Q2hFy0R9tAfnfLkp5MW0UH9eUvnDSnFwKZ0AvgS1ucqR4KzvVHgnke1VQ==} + engines: {node: '>=6.0.0'} + hasBin: true + dependencies: + '@babel/types': 7.23.6 + dev: true + + /@babel/plugin-transform-react-jsx-self@7.23.3(@babel/core@7.23.7): + resolution: {integrity: sha512-qXRvbeKDSfwnlJnanVRp0SfuWE5DQhwQr5xtLBzp56Wabyo+4CMosF6Kfp+eOD/4FYpql64XVJ2W0pVLlJZxOQ==} + engines: {node: '>=6.9.0'} + peerDependencies: + '@babel/core': ^7.0.0-0 + dependencies: + '@babel/core': 7.23.7 + '@babel/helper-plugin-utils': 7.22.5 + dev: true + + /@babel/plugin-transform-react-jsx-source@7.23.3(@babel/core@7.23.7): + resolution: {integrity: sha512-91RS0MDnAWDNvGC6Wio5XYkyWI39FMFO+JK9+4AlgaTH+yWwVTsw7/sn6LK0lH7c5F+TFkpv/3LfCJ1Ydwof/g==} + engines: {node: '>=6.9.0'} + peerDependencies: + '@babel/core': ^7.0.0-0 + dependencies: + '@babel/core': 7.23.7 + '@babel/helper-plugin-utils': 7.22.5 + dev: true + + /@babel/template@7.22.15: + resolution: {integrity: sha512-QPErUVm4uyJa60rkI73qneDacvdvzxshT3kksGqlGWYdOTIUOwJ7RDUL8sGqslY1uXWSL6xMFKEXDS3ox2uF0w==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/code-frame': 7.23.5 + '@babel/parser': 7.23.6 + '@babel/types': 7.23.6 + dev: true + + /@babel/traverse@7.23.7: + resolution: {integrity: sha512-tY3mM8rH9jM0YHFGyfC0/xf+SB5eKUu7HPj7/k3fpi9dAlsMc5YbQvDi0Sh2QTPXqMhyaAtzAr807TIyfQrmyg==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/code-frame': 7.23.5 + '@babel/generator': 7.23.6 + '@babel/helper-environment-visitor': 7.22.20 + '@babel/helper-function-name': 7.23.0 + '@babel/helper-hoist-variables': 7.22.5 + '@babel/helper-split-export-declaration': 7.22.6 + '@babel/parser': 7.23.6 + '@babel/types': 7.23.6 + debug: 4.3.4 + globals: 11.12.0 + transitivePeerDependencies: + - supports-color + dev: true + + /@babel/types@7.23.6: + resolution: {integrity: sha512-+uarb83brBzPKN38NX1MkB6vb6+mwvR6amUulqAE7ccQw1pEl+bCia9TbdG1lsnFP7lZySvUn37CHyXQdfTwzg==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/helper-string-parser': 7.23.4 + '@babel/helper-validator-identifier': 7.22.20 + to-fast-properties: 2.0.0 + dev: true + + /@esbuild/android-arm64@0.17.19: + resolution: {integrity: sha512-KBMWvEZooR7+kzY0BtbTQn0OAYY7CsiydT63pVEaPtVYF0hXbUaOyZog37DKxK7NF3XacBJOpYT4adIJh+avxA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + requiresBuild: true + optional: true + + /@esbuild/android-arm64@0.18.20: + resolution: {integrity: sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-arm@0.17.19: + resolution: {integrity: sha512-rIKddzqhmav7MSmoFCmDIb6e2W57geRsM94gV2l38fzhXMwq7hZoClug9USI2pFRGL06f4IOPHHpFNOkWieR8A==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + requiresBuild: true + optional: true + + /@esbuild/android-arm@0.18.20: + resolution: {integrity: sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-x64@0.17.19: + resolution: {integrity: sha512-uUTTc4xGNDT7YSArp/zbtmbhO0uEEK9/ETW29Wk1thYUJBz3IVnvgEiEwEa9IeLyvnpKrWK64Utw2bgUmDveww==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + requiresBuild: true + optional: true + + /@esbuild/android-x64@0.18.20: + resolution: {integrity: sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/darwin-arm64@0.17.19: + resolution: {integrity: sha512-80wEoCfF/hFKM6WE1FyBHc9SfUblloAWx6FJkFWTWiCoht9Mc0ARGEM47e67W9rI09YoUxJL68WHfDRYEAvOhg==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + requiresBuild: true + optional: true + + /@esbuild/darwin-arm64@0.18.20: + resolution: {integrity: sha512-bxRHW5kHU38zS2lPTPOyuyTm+S+eobPUnTNkdJEfAddYgEcll4xkT8DB9d2008DtTbl7uJag2HuE5NZAZgnNEA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /@esbuild/darwin-x64@0.17.19: + resolution: {integrity: sha512-IJM4JJsLhRYr9xdtLytPLSH9k/oxR3boaUIYiHkAawtwNOXKE8KoU8tMvryogdcT8AU+Bflmh81Xn6Q0vTZbQw==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + requiresBuild: true + optional: true + + /@esbuild/darwin-x64@0.18.20: + resolution: {integrity: sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /@esbuild/freebsd-arm64@0.17.19: + resolution: {integrity: sha512-pBwbc7DufluUeGdjSU5Si+P3SoMF5DQ/F/UmTSb8HXO80ZEAJmrykPyzo1IfNbAoaqw48YRpv8shwd1NoI0jcQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + requiresBuild: true + optional: true + + /@esbuild/freebsd-arm64@0.18.20: + resolution: {integrity: sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/freebsd-x64@0.17.19: + resolution: {integrity: sha512-4lu+n8Wk0XlajEhbEffdy2xy53dpR06SlzvhGByyg36qJw6Kpfk7cp45DR/62aPH9mtJRmIyrXAS5UWBrJT6TQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + requiresBuild: true + optional: true + + /@esbuild/freebsd-x64@0.18.20: + resolution: {integrity: sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-arm64@0.17.19: + resolution: {integrity: sha512-ct1Tg3WGwd3P+oZYqic+YZF4snNl2bsnMKRkb3ozHmnM0dGWuxcPTTntAF6bOP0Sp4x0PjSF+4uHQ1xvxfRKqg==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-arm64@0.18.20: + resolution: {integrity: sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-arm@0.17.19: + resolution: {integrity: sha512-cdmT3KxjlOQ/gZ2cjfrQOtmhG4HJs6hhvm3mWSRDPtZ/lP5oe8FWceS10JaSJC13GBd4eH/haHnqf7hhGNLerA==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-arm@0.18.20: + resolution: {integrity: sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-ia32@0.17.19: + resolution: {integrity: sha512-w4IRhSy1VbsNxHRQpeGCHEmibqdTUx61Vc38APcsRbuVgK0OPEnQ0YD39Brymn96mOx48Y2laBQGqgZ0j9w6SQ==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-ia32@0.18.20: + resolution: {integrity: sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-loong64@0.17.19: + resolution: {integrity: sha512-2iAngUbBPMq439a+z//gE+9WBldoMp1s5GWsUSgqHLzLJ9WoZLZhpwWuym0u0u/4XmZ3gpHmzV84PonE+9IIdQ==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-loong64@0.18.20: + resolution: {integrity: sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-mips64el@0.17.19: + resolution: {integrity: sha512-LKJltc4LVdMKHsrFe4MGNPp0hqDFA1Wpt3jE1gEyM3nKUvOiO//9PheZZHfYRfYl6AwdTH4aTcXSqBerX0ml4A==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-mips64el@0.18.20: + resolution: {integrity: sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-ppc64@0.17.19: + resolution: {integrity: sha512-/c/DGybs95WXNS8y3Ti/ytqETiW7EU44MEKuCAcpPto3YjQbyK3IQVKfF6nbghD7EcLUGl0NbiL5Rt5DMhn5tg==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-ppc64@0.18.20: + resolution: {integrity: sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-riscv64@0.17.19: + resolution: {integrity: sha512-FC3nUAWhvFoutlhAkgHf8f5HwFWUL6bYdvLc/TTuxKlvLi3+pPzdZiFKSWz/PF30TB1K19SuCxDTI5KcqASJqA==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-riscv64@0.18.20: + resolution: {integrity: sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-s390x@0.17.19: + resolution: {integrity: sha512-IbFsFbxMWLuKEbH+7sTkKzL6NJmG2vRyy6K7JJo55w+8xDk7RElYn6xvXtDW8HCfoKBFK69f3pgBJSUSQPr+4Q==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-s390x@0.18.20: + resolution: {integrity: sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-x64@0.17.19: + resolution: {integrity: sha512-68ngA9lg2H6zkZcyp22tsVt38mlhWde8l3eJLWkyLrp4HwMUr3c1s/M2t7+kHIhvMjglIBrFpncX1SzMckomGw==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + requiresBuild: true + optional: true + + /@esbuild/linux-x64@0.18.20: + resolution: {integrity: sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/netbsd-x64@0.17.19: + resolution: {integrity: sha512-CwFq42rXCR8TYIjIfpXCbRX0rp1jo6cPIUPSaWwzbVI4aOfX96OXY8M6KNmtPcg7QjYeDmN+DD0Wp3LaBOLf4Q==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + requiresBuild: true + optional: true + + /@esbuild/netbsd-x64@0.18.20: + resolution: {integrity: sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/openbsd-x64@0.17.19: + resolution: {integrity: sha512-cnq5brJYrSZ2CF6c35eCmviIN3k3RczmHz8eYaVlNasVqsNY+JKohZU5MKmaOI+KkllCdzOKKdPs762VCPC20g==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + requiresBuild: true + optional: true + + /@esbuild/openbsd-x64@0.18.20: + resolution: {integrity: sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/sunos-x64@0.17.19: + resolution: {integrity: sha512-vCRT7yP3zX+bKWFeP/zdS6SqdWB8OIpaRq/mbXQxTGHnIxspRtigpkUcDMlSCOejlHowLqII7K2JKevwyRP2rg==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + requiresBuild: true + optional: true + + /@esbuild/sunos-x64@0.18.20: + resolution: {integrity: sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-arm64@0.17.19: + resolution: {integrity: sha512-yYx+8jwowUstVdorcMdNlzklLYhPxjniHWFKgRqH7IFlUEa0Umu3KuYplf1HUZZ422e3NU9F4LGb+4O0Kdcaag==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + requiresBuild: true + optional: true + + /@esbuild/win32-arm64@0.18.20: + resolution: {integrity: sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-ia32@0.17.19: + resolution: {integrity: sha512-eggDKanJszUtCdlVs0RB+h35wNlb5v4TWEkq4vZcmVt5u/HiDZrTXe2bWFQUez3RgNHwx/x4sk5++4NSSicKkw==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + requiresBuild: true + optional: true + + /@esbuild/win32-ia32@0.18.20: + resolution: {integrity: sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-x64@0.17.19: + resolution: {integrity: sha512-lAhycmKnVOuRYNtRtatQR1LPQf2oYCkRGkSFnseDAKPl8lu5SOsK/e1sXe5a0Pc5kHIHe6P2I/ilntNv2xf3cA==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + requiresBuild: true + optional: true + + /@esbuild/win32-x64@0.18.20: + resolution: {integrity: sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@eslint/eslintrc@1.4.1: + resolution: {integrity: sha512-XXrH9Uarn0stsyldqDYq8r++mROmWRI1xKMXa640Bb//SY1+ECYX6VzT6Lcx5frD0V30XieqJ0oX9I2Xj5aoMA==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + ajv: 6.12.6 + debug: 4.3.4 + espree: 9.6.1 + globals: 13.24.0 + ignore: 5.3.0 + import-fresh: 3.3.0 + js-yaml: 4.1.0 + minimatch: 3.1.2 + strip-json-comments: 3.1.1 + transitivePeerDependencies: + - supports-color + dev: true + + /@ethersproject/abi@5.7.0: + resolution: {integrity: sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==} + dependencies: + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + + /@ethersproject/abstract-provider@5.7.0: + resolution: {integrity: sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/networks': 5.7.1 + '@ethersproject/properties': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/web': 5.7.1 + + /@ethersproject/abstract-signer@5.7.0: + resolution: {integrity: sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==} + dependencies: + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + + /@ethersproject/address@5.7.0: + resolution: {integrity: sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/rlp': 5.7.0 + + /@ethersproject/base64@5.7.0: + resolution: {integrity: sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==} + dependencies: + '@ethersproject/bytes': 5.7.0 + + /@ethersproject/basex@5.7.0: + resolution: {integrity: sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/properties': 5.7.0 + + /@ethersproject/bignumber@5.7.0: + resolution: {integrity: sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + bn.js: 5.2.1 + + /@ethersproject/bytes@5.7.0: + resolution: {integrity: sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==} + dependencies: + '@ethersproject/logger': 5.7.0 + + /@ethersproject/constants@5.7.0: + resolution: {integrity: sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + + /@ethersproject/contracts@5.7.0: + resolution: {integrity: sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==} + dependencies: + '@ethersproject/abi': 5.7.0 + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/transactions': 5.7.0 + + /@ethersproject/hash@5.7.0: + resolution: {integrity: sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==} + dependencies: + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/base64': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + + /@ethersproject/hdnode@5.7.0: + resolution: {integrity: sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==} + dependencies: + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/basex': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/pbkdf2': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/wordlists': 5.7.0 + + /@ethersproject/json-wallets@5.7.0: + resolution: {integrity: sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==} + dependencies: + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/hdnode': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/pbkdf2': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/random': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + aes-js: 3.0.0 + scrypt-js: 3.0.1 + + /@ethersproject/keccak256@5.7.0: + resolution: {integrity: sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==} + dependencies: + '@ethersproject/bytes': 5.7.0 + js-sha3: 0.8.0 + + /@ethersproject/logger@5.7.0: + resolution: {integrity: sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==} + + /@ethersproject/networks@5.7.1: + resolution: {integrity: sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==} + dependencies: + '@ethersproject/logger': 5.7.0 + + /@ethersproject/pbkdf2@5.7.0: + resolution: {integrity: sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/sha2': 5.7.0 + + /@ethersproject/properties@5.7.0: + resolution: {integrity: sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==} + dependencies: + '@ethersproject/logger': 5.7.0 + + /@ethersproject/providers@5.7.2: + resolution: {integrity: sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==} + dependencies: + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/base64': 5.7.0 + '@ethersproject/basex': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/networks': 5.7.1 + '@ethersproject/properties': 5.7.0 + '@ethersproject/random': 5.7.0 + '@ethersproject/rlp': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/web': 5.7.1 + bech32: 1.1.4 + ws: 7.4.6 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + /@ethersproject/random@5.7.0: + resolution: {integrity: sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + + /@ethersproject/rlp@5.7.0: + resolution: {integrity: sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + + /@ethersproject/sha2@5.7.0: + resolution: {integrity: sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + hash.js: 1.1.7 + + /@ethersproject/signing-key@5.7.0: + resolution: {integrity: sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + bn.js: 5.2.1 + elliptic: 6.5.4 + hash.js: 1.1.7 + + /@ethersproject/solidity@5.7.0: + resolution: {integrity: sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/strings': 5.7.0 + + /@ethersproject/strings@5.7.0: + resolution: {integrity: sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/logger': 5.7.0 + + /@ethersproject/transactions@5.7.0: + resolution: {integrity: sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==} + dependencies: + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/rlp': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + + /@ethersproject/units@5.7.0: + resolution: {integrity: sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/logger': 5.7.0 + + /@ethersproject/wallet@5.7.0: + resolution: {integrity: sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==} + dependencies: + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/hdnode': 5.7.0 + '@ethersproject/json-wallets': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/random': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/wordlists': 5.7.0 + + /@ethersproject/web@5.7.1: + resolution: {integrity: sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==} + dependencies: + '@ethersproject/base64': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + + /@ethersproject/wordlists@5.7.0: + resolution: {integrity: sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + + /@humanwhocodes/config-array@0.11.13: + resolution: {integrity: sha512-JSBDMiDKSzQVngfRjOdFXgFfklaXI4K9nLF49Auh21lmBWRLIK3+xTErTWD4KU54pb6coM6ESE7Awz/FNU3zgQ==} + engines: {node: '>=10.10.0'} + dependencies: + '@humanwhocodes/object-schema': 2.0.1 + debug: 4.3.4 + minimatch: 3.1.2 + transitivePeerDependencies: + - supports-color + dev: true + + /@humanwhocodes/module-importer@1.0.1: + resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} + engines: {node: '>=12.22'} + dev: true + + /@humanwhocodes/object-schema@2.0.1: + resolution: {integrity: sha512-dvuCeX5fC9dXgJn9t+X5atfmgQAzUOWqS1254Gh0m6i8wKd10ebXkfNKiRK+1GWi/yTvvLDHpoxLr0xxxeslWw==} + dev: true + + /@improbable-eng/grpc-web-node-http-transport@0.15.0(@improbable-eng/grpc-web@0.15.0): + resolution: {integrity: sha512-HLgJfVolGGpjc9DWPhmMmXJx8YGzkek7jcCFO1YYkSOoO81MWRZentPOd/JiKiZuU08wtc4BG+WNuGzsQB5jZA==} + peerDependencies: + '@improbable-eng/grpc-web': '>=0.13.0' + dependencies: + '@improbable-eng/grpc-web': 0.15.0(google-protobuf@3.21.2) + + /@improbable-eng/grpc-web@0.15.0(google-protobuf@3.21.2): + resolution: {integrity: sha512-ERft9/0/8CmYalqOVnJnpdDry28q+j+nAlFFARdjyxXDJ+Mhgv9+F600QC8BR9ygOfrXRlAk6CvST2j+JCpQPg==} + peerDependencies: + google-protobuf: ^3.14.0 + dependencies: + browser-headers: 0.4.1 + google-protobuf: 3.21.2 + + /@jridgewell/gen-mapping@0.3.3: + resolution: {integrity: sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ==} + engines: {node: '>=6.0.0'} + dependencies: + '@jridgewell/set-array': 1.1.2 + '@jridgewell/sourcemap-codec': 1.4.15 + '@jridgewell/trace-mapping': 0.3.20 + dev: true + + /@jridgewell/resolve-uri@3.1.1: + resolution: {integrity: sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==} + engines: {node: '>=6.0.0'} + dev: true + + /@jridgewell/set-array@1.1.2: + resolution: {integrity: sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==} + engines: {node: '>=6.0.0'} + dev: true + + /@jridgewell/sourcemap-codec@1.4.15: + resolution: {integrity: sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==} + dev: true + + /@jridgewell/trace-mapping@0.3.20: + resolution: {integrity: sha512-R8LcPeWZol2zR8mmH3JeKQ6QRCFb7XgUhV9ZlGhHLGyg4wpPiPZNQOOWhFZhxKw8u//yTbNGI42Bx/3paXEQ+Q==} + dependencies: + '@jridgewell/resolve-uri': 3.1.1 + '@jridgewell/sourcemap-codec': 1.4.15 + dev: true + + /@latticexyz/abi-ts@2.0.0-next.15: + resolution: {integrity: sha512-/YJi9Xjstn24APqHCgUZj5QZhFeuD6TNR0fEnfKfr4GQ90Clv4Lr63Rhk847NdA/eI3c1rKfNWbFc6pJYFYwYQ==} + hasBin: true + dependencies: + chalk: 5.3.0 + debug: 4.3.4 + execa: 7.2.0 + glob: 8.1.0 + yargs: 17.7.2 + transitivePeerDependencies: + - supports-color + + /@latticexyz/block-logs-stream@2.0.0-next.15(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-OcvbBT3oOQVNJHfIX89a6KFUn6X3WFisKoG2Q0yYANdyFWPfoNYopep1h4RMzvKGebuN+XQ/ClHIsrupcAuxZA==} + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + abitype: 0.9.8(typescript@5.1.6)(zod@3.22.4) + debug: 4.3.4 + rxjs: 7.5.5 + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + - zod + dev: false + + /@latticexyz/cli@2.0.0-next.15(google-protobuf@3.21.2): + resolution: {integrity: sha512-q10NkBxDwnTK+6vIbHgwpeYlo/aD/3h6E8ZyzdAl8GgYLtDmZ+XJyOJwj0Bss6BdzjD9gThfMQ/wFVfGGvgung==} + hasBin: true + dependencies: + '@ethersproject/abi': 5.7.0 + '@ethersproject/providers': 5.7.2 + '@improbable-eng/grpc-web': 0.15.0(google-protobuf@3.21.2) + '@improbable-eng/grpc-web-node-http-transport': 0.15.0(@improbable-eng/grpc-web@0.15.0) + '@latticexyz/abi-ts': 2.0.0-next.15 + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/config': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/gas-report': 2.0.0-next.15 + '@latticexyz/protocol-parser': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/services': 2.0.0-next.15(google-protobuf@3.21.2) + '@latticexyz/store': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/utils': 2.0.0-next.15 + '@latticexyz/world': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/world-modules': 2.0.0-next.15(typescript@5.1.6) + chalk: 5.3.0 + chokidar: 3.5.3 + debug: 4.3.4 + dotenv: 16.3.1 + ejs: 3.1.9 + ethers: 5.7.2 + execa: 7.2.0 + glob: 8.1.0 + nice-grpc-web: 2.0.2(google-protobuf@3.21.2) + openurl: 1.1.1 + p-retry: 5.1.2 + path: 0.12.7 + rxjs: 7.5.5 + throttle-debounce: 5.0.0 + typescript: 5.1.6 + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + yargs: 17.7.2 + zod: 3.22.4 + zod-validation-error: 1.5.0(zod@3.22.4) + transitivePeerDependencies: + - bufferutil + - google-protobuf + - supports-color + - utf-8-validate + + /@latticexyz/common@2.0.0-next.15(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-BncRMgjVqd0jbWtNBFUDk/RMWV+/LRcQAZAY2yWWuoVg7SZfX2ZrSSMOuf/V04TNJK2gKd6GVdRFF6Hjdi5C0A==} + dependencies: + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@solidity-parser/parser': 0.16.2 + debug: 4.3.4 + execa: 7.2.0 + p-queue: 7.4.1 + p-retry: 5.1.2 + prettier: 2.8.8 + prettier-plugin-solidity: 1.1.3(prettier@2.8.8) + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + - zod + + /@latticexyz/config@2.0.0-next.15(typescript@5.1.6): + resolution: {integrity: sha512-dHcOJydgIBo4F6eK0bpo8JspE1kZN7TpXU0DceANlKTDUBg+KtSsSeGNDJBT3Ch0nJGeW0RXdA7LuiN6UIEG/Q==} + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + esbuild: 0.17.19 + find-up: 6.3.0 + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + zod: 3.22.4 + zod-validation-error: 1.5.0(zod@3.22.4) + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + + /@latticexyz/dev-tools@2.0.0-next.15(@latticexyz/common@2.0.0-next.15)(@latticexyz/recs@2.0.0-next.15)(@latticexyz/store-sync@2.0.0-next.15)(@latticexyz/store@2.0.0-next.15)(@latticexyz/utils@2.0.0-next.15)(@latticexyz/world@2.0.0-next.15)(@types/react@18.2.22)(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-BPzxOaNb5ia/5IksfgO96XiJo+cvJYbL77haWFkVMIJtyb+ToWWtHm8vjHOor3OeicpffXEPEJUKIdbwKHaAwg==} + peerDependencies: + '@latticexyz/common': 2.0.0-next.15 + '@latticexyz/recs': 2.0.0-next.15 + '@latticexyz/store': 2.0.0-next.15 + '@latticexyz/store-sync': 2.0.0-next.15 + '@latticexyz/utils': 2.0.0-next.15 + '@latticexyz/world': 2.0.0-next.15 + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/react': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/recs': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/store': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/store-sync': 2.0.0-next.15(@types/react@18.2.22)(react@18.2.0)(typescript@5.1.6) + '@latticexyz/utils': 2.0.0-next.15 + '@latticexyz/world': 2.0.0-next.15(typescript@5.1.6) + react: 18.2.0 + react-dom: 18.2.0(react@18.2.0) + react-router-dom: 6.21.1(react-dom@18.2.0)(react@18.2.0) + rxjs: 7.5.5 + tailwind-merge: 1.14.0 + use-local-storage-state: 18.3.3(react-dom@18.2.0)(react@18.2.0) + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + zustand: 4.4.7(@types/react@18.2.22)(react@18.2.0) + transitivePeerDependencies: + - '@types/react' + - bufferutil + - immer + - supports-color + - typescript + - utf-8-validate + - zod + dev: false + + /@latticexyz/gas-report@2.0.0-next.15: + resolution: {integrity: sha512-/n4EKkkr4GUjopbPnnyNtLnu8EqOUnoT1QKX+6R5BW7uu1zdkvE+WpTiPpDHgV1lP1b/qIX9+z5WGbb2nknPtw==} + hasBin: true + dependencies: + chalk: 5.3.0 + dotenv: 16.3.1 + execa: 7.2.0 + stream-to-array: 2.3.0 + strip-ansi: 7.1.0 + table: 6.8.1 + yargs: 17.7.2 + + /@latticexyz/protocol-parser@2.0.0-next.15(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-J6hMEhW2XnBGs5o1dccTO7Qbpg+waN3EdYVyp7H2c2fmHZs0LByZuzbYY4Qqx6CCsqWqS8N4CPROIWqH5vrl+Q==} + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + abitype: 0.9.8(typescript@5.1.6)(zod@3.22.4) + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + - zod + + /@latticexyz/react@2.0.0-next.15(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-/pCGph3j4qXsB5q9CcsT5rMJyOeF7DIzX3+rAk2YUROihS3TgQSieNCdw9Bc9KZi+X73S6T/+MEy5+Mtnat2+A==} + dependencies: + '@latticexyz/recs': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/store': 2.0.0-next.15(typescript@5.1.6) + fast-deep-equal: 3.1.3 + mobx: 6.12.0 + react: 18.2.0 + rxjs: 7.5.5 + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + - zod + dev: false + + /@latticexyz/recs@2.0.0-next.15(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-ECqWQ5t3ZkLCFgF/Nrh+S6JsbmJ8egVOmnW+WO+7LTShODU2ZCSW+cpRvFaWJQGd/BSprxBK7nExF3bD/vhKHQ==} + dependencies: + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/utils': 2.0.0-next.15 + mobx: 6.12.0 + rxjs: 7.5.5 + transitivePeerDependencies: + - bufferutil + - typescript + - utf-8-validate + - zod + dev: false + + /@latticexyz/schema-type@2.0.0-next.15(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-Tx23kYFhF10cGmapfV8/Bc/rqfOh0FWvVyEJ6oVYks7cuaSHj9Zw0my0lZy4Zi9RleVYArNnsgcOyLLBsHtaKw==} + dependencies: + abitype: 0.9.8(typescript@5.1.6)(zod@3.22.4) + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + transitivePeerDependencies: + - bufferutil + - typescript + - utf-8-validate + - zod + + /@latticexyz/services@2.0.0-next.15(google-protobuf@3.21.2): + resolution: {integrity: sha512-stV0B/5rNA+YvGKfzULCrqLI72vN4LzEnRRdYoL3GWrAUHX/iI9v6DWyTiLcZ9+GgJTtrFrCrmy8UtKUM4EMug==} + dependencies: + long: 5.2.3 + nice-grpc-common: 2.0.2 + nice-grpc-web: 2.0.2(google-protobuf@3.21.2) + protobufjs: 7.2.5 + transitivePeerDependencies: + - google-protobuf + + /@latticexyz/store-sync@2.0.0-next.15(@types/react@18.2.22)(react@18.2.0)(typescript@5.1.6): + resolution: {integrity: sha512-E8OpSPCLCQ41kG58PPs5SDRhYQPUZrE/fzm7FObPYfV5YGMdNRMfp9QvvSjFRoBXDoC+gQiLsX1mgLTzkHTDZQ==} + dependencies: + '@latticexyz/block-logs-stream': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/protocol-parser': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/recs': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/store': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/world': 2.0.0-next.15(typescript@5.1.6) + '@trpc/client': 10.34.0(@trpc/server@10.34.0) + '@trpc/server': 10.34.0 + change-case: 5.4.0 + debug: 4.3.4 + drizzle-orm: 0.28.6(kysely@0.26.3)(postgres@3.4.3)(sql.js@1.9.0) + kysely: 0.26.3 + postgres: 3.4.3 + rxjs: 7.5.5 + sql.js: 1.9.0 + superjson: 1.13.3 + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + zod: 3.22.4 + zustand: 4.4.7(@types/react@18.2.22)(react@18.2.0) + transitivePeerDependencies: + - '@aws-sdk/client-rds-data' + - '@cloudflare/workers-types' + - '@libsql/client' + - '@neondatabase/serverless' + - '@opentelemetry/api' + - '@planetscale/database' + - '@types/better-sqlite3' + - '@types/pg' + - '@types/react' + - '@types/sql.js' + - '@vercel/postgres' + - better-sqlite3 + - bufferutil + - bun-types + - immer + - knex + - mysql2 + - pg + - react + - sqlite3 + - supports-color + - typescript + - utf-8-validate + dev: false + + /@latticexyz/store@2.0.0-next.15(typescript@5.1.6): + resolution: {integrity: sha512-ZU5O8hT9BqXPEFgEkpBJgcJeqVumYEAILhc7SxayKtbsLsjVuxdflH67Snvx4U+EHUXiO8BflCVrbBT5b57Vmw==} + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/config': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + abitype: 0.9.8(typescript@5.1.6)(zod@3.22.4) + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + zod: 3.22.4 + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + + /@latticexyz/utils@2.0.0-next.15: + resolution: {integrity: sha512-kal4N+VPnvxkkWdPrWKZeFwWDle7JAw9JQq54Ur6/VmH82tbvcrkeOgESIz3RFuzworFK2wyHuaxF9wq3aWxEw==} + dependencies: + mobx: 6.12.0 + proxy-deep: 3.1.1 + rxjs: 7.5.5 + + /@latticexyz/world-modules@2.0.0-next.15(typescript@5.1.6): + resolution: {integrity: sha512-cXT5nWSA2/0gFt3W5OwQhAKtd3546O2lwC1Cpbk+uGKgmsjF7fw6jriN8oVuWjba2OVIQvXqqTc8drrRjTZ4zQ==} + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/config': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/store': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/world': 2.0.0-next.15(typescript@5.1.6) + zod: 3.22.4 + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + + /@latticexyz/world@2.0.0-next.15(typescript@5.1.6): + resolution: {integrity: sha512-Ypydf7sCckEqWRzkKWQleNAIenj0e5Lz5GggTJ98B3t5rsdNmBAjTJedAZpFcs1VSvwXdVanZ3+Rk7KY11vEqA==} + dependencies: + '@latticexyz/common': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/config': 2.0.0-next.15(typescript@5.1.6) + '@latticexyz/schema-type': 2.0.0-next.15(typescript@5.1.6)(zod@3.22.4) + '@latticexyz/store': 2.0.0-next.15(typescript@5.1.6) + abitype: 0.9.8(typescript@5.1.6)(zod@3.22.4) + viem: 1.14.0(typescript@5.1.6)(zod@3.22.4) + zod: 3.22.4 + transitivePeerDependencies: + - bufferutil + - supports-color + - typescript + - utf-8-validate + + /@noble/curves@1.2.0: + resolution: {integrity: sha512-oYclrNgRaM9SsBUBVbb8M6DTV7ZHRTKugureoYEncY5c65HOmRzvSiTE3y5CYaPYJA/GVkrhXEoF0M3Ya9PMnw==} + dependencies: + '@noble/hashes': 1.3.2 + + /@noble/hashes@1.3.2: + resolution: {integrity: sha512-MVC8EAQp7MvEcm30KWENFjgR+Mkmf+D189XJTkFIlwohU5hcBbn1ZkKq7KVTi2Hme3PMGF390DaL52beVrIihQ==} + engines: {node: '>= 16'} + + /@nodelib/fs.scandir@2.1.5: + resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} + engines: {node: '>= 8'} + dependencies: + '@nodelib/fs.stat': 2.0.5 + run-parallel: 1.2.0 + dev: true + + /@nodelib/fs.stat@2.0.5: + resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} + engines: {node: '>= 8'} + dev: true + + /@nodelib/fs.walk@1.2.8: + resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} + engines: {node: '>= 8'} + dependencies: + '@nodelib/fs.scandir': 2.1.5 + fastq: 1.16.0 + dev: true + + /@protobufjs/aspromise@1.1.2: + resolution: {integrity: sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ==} + + /@protobufjs/base64@1.1.2: + resolution: {integrity: sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==} + + /@protobufjs/codegen@2.0.4: + resolution: {integrity: sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==} + + /@protobufjs/eventemitter@1.1.0: + resolution: {integrity: sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q==} + + /@protobufjs/fetch@1.1.0: + resolution: {integrity: sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ==} + dependencies: + '@protobufjs/aspromise': 1.1.2 + '@protobufjs/inquire': 1.1.0 + + /@protobufjs/float@1.0.2: + resolution: {integrity: sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ==} + + /@protobufjs/inquire@1.1.0: + resolution: {integrity: sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q==} + + /@protobufjs/path@1.1.2: + resolution: {integrity: sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA==} + + /@protobufjs/pool@1.1.0: + resolution: {integrity: sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw==} + + /@protobufjs/utf8@1.1.0: + resolution: {integrity: sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw==} + + /@remix-run/router@1.14.1: + resolution: {integrity: sha512-Qg4DMQsfPNAs88rb2xkdk03N3bjK4jgX5fR24eHCTR9q6PrhZQZ4UJBPzCHJkIpTRN1UKxx2DzjZmnC+7Lj0Ow==} + engines: {node: '>=14.0.0'} + dev: false + + /@scure/base@1.1.5: + resolution: {integrity: sha512-Brj9FiG2W1MRQSTB212YVPRrcbjkv48FoZi/u4l/zds/ieRrqsh7aUf6CLwkAq61oKXr/ZlTzlY66gLIj3TFTQ==} + + /@scure/bip32@1.3.2: + resolution: {integrity: sha512-N1ZhksgwD3OBlwTv3R6KFEcPojl/W4ElJOeCZdi+vuI5QmTFwLq3OFf2zd2ROpKvxFdgZ6hUpb0dx9bVNEwYCA==} + dependencies: + '@noble/curves': 1.2.0 + '@noble/hashes': 1.3.2 + '@scure/base': 1.1.5 + + /@scure/bip39@1.2.1: + resolution: {integrity: sha512-Z3/Fsz1yr904dduJD0NpiyRHhRYHdcnyh73FZWiV+/qhWi83wNJ3NWolYqCEN+ZWsUz2TWwajJggcRE9r1zUYg==} + dependencies: + '@noble/hashes': 1.3.2 + '@scure/base': 1.1.5 + + /@solidity-parser/parser@0.16.2: + resolution: {integrity: sha512-PI9NfoA3P8XK2VBkK5oIfRgKDsicwDZfkVq9ZTBCQYGOP1N2owgY2dyLGyU5/J/hQs8KRk55kdmvTLjy3Mu3vg==} + dependencies: + antlr4ts: 0.5.0-alpha.4 + + /@trpc/client@10.34.0(@trpc/server@10.34.0): + resolution: {integrity: sha512-nqtDTIqSY/9syo2EjSy4WWWXPU9GsamEh9Tsg698gLAh1nhgFc5+/YYeb+Ne1pbvWGZ5/3t9Dcz3h4wMyyJ9gQ==} + peerDependencies: + '@trpc/server': 10.34.0 + dependencies: + '@trpc/server': 10.34.0 + dev: false + + /@trpc/server@10.34.0: + resolution: {integrity: sha512-2VMW44Fpaoyqb50dBtzdSWMhqt8lmoJiocEyBBeDb03R0W+XrzbVD5kU/wqKPlcp1DWeNCkOEIMtetMZCfo1hA==} + dev: false + + /@types/debug@4.1.7: + resolution: {integrity: sha512-9AonUzyTjXXhEOa0DnqpzZi6VHlqKMswga9EXjpXnnqxwLtdvPPtlO8evrI5D9S6asFRCQ6v+wpiUKbw+vKqyg==} + dependencies: + '@types/ms': 0.7.34 + dev: true + + /@types/json-schema@7.0.15: + resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} + dev: true + + /@types/ms@0.7.34: + resolution: {integrity: sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==} + dev: true + + /@types/node@18.19.6: + resolution: {integrity: sha512-X36s5CXMrrJOs2lQCdDF68apW4Rfx9ixYMawlepwmE4Anezv/AV2LSpKD1Ub8DAc+urp5bk0BGZ6NtmBitfnsg==} + dependencies: + undici-types: 5.26.5 + + /@types/prettier@2.7.2: + resolution: {integrity: sha512-KufADq8uQqo1pYKVIYzfKbJfBAc0sOeXqGbFaSpv8MRmC/zXgowNZmFcbngndGk922QDmOASEXUZCaY48gs4cg==} + dev: true + + /@types/prop-types@15.7.11: + resolution: {integrity: sha512-ga8y9v9uyeiLdpKddhxYQkxNDrfvuPrlFb0N1qnZZByvcElJaXthF1UhvCh9TLWJBEHeNtdnbysW7Y6Uq8CVng==} + + /@types/react-dom@18.2.7: + resolution: {integrity: sha512-GRaAEriuT4zp9N4p1i8BDBYmEyfo+xQ3yHjJU4eiK5NDa1RmUZG+unZABUTK4/Ox/M+GaHwb6Ow8rUITrtjszA==} + dependencies: + '@types/react': 18.2.22 + dev: true + + /@types/react@18.2.22: + resolution: {integrity: sha512-60fLTOLqzarLED2O3UQImc/lsNRgG0jE/a1mPW9KjMemY0LMITWEsbS4VvZ4p6rorEHd5YKxxmMKSDK505GHpA==} + dependencies: + '@types/prop-types': 15.7.11 + '@types/scheduler': 0.16.8 + csstype: 3.1.3 + + /@types/retry@0.12.1: + resolution: {integrity: sha512-xoDlM2S4ortawSWORYqsdU+2rxdh4LRW9ytc3zmT37RIKQh6IHyKwwtKhKis9ah8ol07DCkZxPt8BBvPjC6v4g==} + + /@types/scheduler@0.16.8: + resolution: {integrity: sha512-WZLiwShhwLRmeV6zH+GkbOFT6Z6VklCItrDioxUnv+u4Ll+8vKeFySoFyK/0ctcRpOmwAicELfmys1sDc/Rw+A==} + + /@types/semver@7.5.6: + resolution: {integrity: sha512-dn1l8LaMea/IjDoHNd9J52uBbInB796CDffS6VdIxvqYCPSG0V0DzHp76GpaWnlhg88uYyPbXCDIowa86ybd5A==} + dev: true + + /@types/ws@8.5.10: + resolution: {integrity: sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==} + dependencies: + '@types/node': 18.19.6 + + /@typescript-eslint/eslint-plugin@5.46.1(@typescript-eslint/parser@5.46.1)(eslint@8.29.0)(typescript@5.1.6): + resolution: {integrity: sha512-YpzNv3aayRBwjs4J3oz65eVLXc9xx0PDbIRisHj+dYhvBn02MjYOD96P8YGiWEIFBrojaUjxvkaUpakD82phsA==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + '@typescript-eslint/parser': ^5.0.0 + eslint: ^6.0.0 || ^7.0.0 || ^8.0.0 + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/parser': 5.46.1(eslint@8.29.0)(typescript@5.1.6) + '@typescript-eslint/scope-manager': 5.46.1 + '@typescript-eslint/type-utils': 5.46.1(eslint@8.29.0)(typescript@5.1.6) + '@typescript-eslint/utils': 5.46.1(eslint@8.29.0)(typescript@5.1.6) + debug: 4.3.4 + eslint: 8.29.0 + ignore: 5.3.0 + natural-compare-lite: 1.4.0 + regexpp: 3.2.0 + semver: 7.5.4 + tsutils: 3.21.0(typescript@5.1.6) + typescript: 5.1.6 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/parser@5.46.1(eslint@8.29.0)(typescript@5.1.6): + resolution: {integrity: sha512-RelQ5cGypPh4ySAtfIMBzBGyrNerQcmfA1oJvPj5f+H4jI59rl9xxpn4bonC0tQvUKOEN7eGBFWxFLK3Xepneg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: ^6.0.0 || ^7.0.0 || ^8.0.0 + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/scope-manager': 5.46.1 + '@typescript-eslint/types': 5.46.1 + '@typescript-eslint/typescript-estree': 5.46.1(typescript@5.1.6) + debug: 4.3.4 + eslint: 8.29.0 + typescript: 5.1.6 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/scope-manager@5.46.1: + resolution: {integrity: sha512-iOChVivo4jpwUdrJZyXSMrEIM/PvsbbDOX1y3UCKjSgWn+W89skxWaYXACQfxmIGhPVpRWK/VWPYc+bad6smIA==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + '@typescript-eslint/types': 5.46.1 + '@typescript-eslint/visitor-keys': 5.46.1 + dev: true + + /@typescript-eslint/type-utils@5.46.1(eslint@8.29.0)(typescript@5.1.6): + resolution: {integrity: sha512-V/zMyfI+jDmL1ADxfDxjZ0EMbtiVqj8LUGPAGyBkXXStWmCUErMpW873zEHsyguWCuq2iN4BrlWUkmuVj84yng==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: '*' + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/typescript-estree': 5.46.1(typescript@5.1.6) + '@typescript-eslint/utils': 5.46.1(eslint@8.29.0)(typescript@5.1.6) + debug: 4.3.4 + eslint: 8.29.0 + tsutils: 3.21.0(typescript@5.1.6) + typescript: 5.1.6 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/types@5.46.1: + resolution: {integrity: sha512-Z5pvlCaZgU+93ryiYUwGwLl9AQVB/PQ1TsJ9NZ/gHzZjN7g9IAn6RSDkpCV8hqTwAiaj6fmCcKSQeBPlIpW28w==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dev: true + + /@typescript-eslint/typescript-estree@5.46.1(typescript@5.1.6): + resolution: {integrity: sha512-j9W4t67QiNp90kh5Nbr1w92wzt+toiIsaVPnEblB2Ih2U9fqBTyqV9T3pYWZBRt6QoMh/zVWP59EpuCjc4VRBg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/types': 5.46.1 + '@typescript-eslint/visitor-keys': 5.46.1 + debug: 4.3.4 + globby: 11.1.0 + is-glob: 4.0.3 + semver: 7.5.4 + tsutils: 3.21.0(typescript@5.1.6) + typescript: 5.1.6 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/utils@5.46.1(eslint@8.29.0)(typescript@5.1.6): + resolution: {integrity: sha512-RBdBAGv3oEpFojaCYT4Ghn4775pdjvwfDOfQ2P6qzNVgQOVrnSPe5/Pb88kv7xzYQjoio0eKHKB9GJ16ieSxvA==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: ^6.0.0 || ^7.0.0 || ^8.0.0 + dependencies: + '@types/json-schema': 7.0.15 + '@types/semver': 7.5.6 + '@typescript-eslint/scope-manager': 5.46.1 + '@typescript-eslint/types': 5.46.1 + '@typescript-eslint/typescript-estree': 5.46.1(typescript@5.1.6) + eslint: 8.29.0 + eslint-scope: 5.1.1 + eslint-utils: 3.0.0(eslint@8.29.0) + semver: 7.5.4 + transitivePeerDependencies: + - supports-color + - typescript + dev: true + + /@typescript-eslint/visitor-keys@5.46.1: + resolution: {integrity: sha512-jczZ9noovXwy59KjRTk1OftT78pwygdcmCuBf8yMoWt/8O8l+6x2LSEze0E4TeepXK4MezW3zGSyoDRZK7Y9cg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + '@typescript-eslint/types': 5.46.1 + eslint-visitor-keys: 3.4.3 + dev: true + + /@vitejs/plugin-react@3.1.0(vite@4.5.1): + resolution: {integrity: sha512-AfgcRL8ZBhAlc3BFdigClmTUMISmmzHn7sB2h9U1odvc5U/MjWXsAaz18b/WoppUTDBzxOJwo2VdClfUcItu9g==} + engines: {node: ^14.18.0 || >=16.0.0} + peerDependencies: + vite: ^4.1.0-beta.0 + dependencies: + '@babel/core': 7.23.7 + '@babel/plugin-transform-react-jsx-self': 7.23.3(@babel/core@7.23.7) + '@babel/plugin-transform-react-jsx-source': 7.23.3(@babel/core@7.23.7) + magic-string: 0.27.0 + react-refresh: 0.14.0 + vite: 4.5.1 + transitivePeerDependencies: + - supports-color + dev: true + + /abitype@0.9.8(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-puLifILdm+8sjyss4S+fsUN09obiT1g2YW6CtcQF+QDzxR0euzgEB29MZujC6zMk2a6SVmtttq1fc6+YFA7WYQ==} + peerDependencies: + typescript: '>=5.0.4' + zod: ^3 >=3.19.1 + peerDependenciesMeta: + typescript: + optional: true + zod: + optional: true + dependencies: + typescript: 5.1.6 + zod: 3.22.4 + + /abort-controller-x@0.4.3: + resolution: {integrity: sha512-VtUwTNU8fpMwvWGn4xE93ywbogTYsuT+AUxAXOeelbXuQVIwNmC5YLeho9sH4vZ4ITW8414TTAOG1nW6uIVHCA==} + + /acorn-jsx@5.3.2(acorn@8.11.3): + resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==} + peerDependencies: + acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 + dependencies: + acorn: 8.11.3 + dev: true + + /acorn@8.11.3: + resolution: {integrity: sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==} + engines: {node: '>=0.4.0'} + hasBin: true + dev: true + + /aes-js@3.0.0: + resolution: {integrity: sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==} + + /ajv@6.12.6: + resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} + dependencies: + fast-deep-equal: 3.1.3 + fast-json-stable-stringify: 2.1.0 + json-schema-traverse: 0.4.1 + uri-js: 4.4.1 + dev: true + + /ajv@8.12.0: + resolution: {integrity: sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==} + dependencies: + fast-deep-equal: 3.1.3 + json-schema-traverse: 1.0.0 + require-from-string: 2.0.2 + uri-js: 4.4.1 + + /ansi-regex@5.0.1: + resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} + engines: {node: '>=8'} + + /ansi-regex@6.0.1: + resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} + engines: {node: '>=12'} + + /ansi-styles@3.2.1: + resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} + engines: {node: '>=4'} + dependencies: + color-convert: 1.9.3 + dev: true + + /ansi-styles@4.3.0: + resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} + engines: {node: '>=8'} + dependencies: + color-convert: 2.0.1 + + /antlr4@4.13.1: + resolution: {integrity: sha512-kiXTspaRYvnIArgE97z5YVVf/cDVQABr3abFRR6mE7yesLMkgu4ujuyV/sgxafQ8wgve0DJQUJ38Z8tkgA2izA==} + engines: {node: '>=16'} + dev: true + + /antlr4ts@0.5.0-alpha.4: + resolution: {integrity: sha512-WPQDt1B74OfPv/IMS2ekXAKkTZIHl88uMetg6q3OTqgFxZ/dxDXI0EWLyZid/1Pe6hTftyg5N7gel5wNAGxXyQ==} + + /any-promise@1.3.0: + resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} + + /anymatch@3.1.3: + resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} + engines: {node: '>= 8'} + dependencies: + normalize-path: 3.0.0 + picomatch: 2.3.1 + + /argparse@2.0.1: + resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} + dev: true + + /array-buffer-byte-length@1.0.0: + resolution: {integrity: sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A==} + dependencies: + call-bind: 1.0.5 + is-array-buffer: 3.0.2 + dev: true + + /array-includes@3.1.7: + resolution: {integrity: sha512-dlcsNBIiWhPkHdOEEKnehA+RNUWDc4UqFtnIXU4uuYDPtA4LDkr7qip2p0VvFAEXNDr0yWZ9PJyIRiGjRLQzwQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + get-intrinsic: 1.2.2 + is-string: 1.0.7 + dev: true + + /array-union@2.1.0: + resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} + engines: {node: '>=8'} + dev: true + + /array.prototype.flat@1.3.2: + resolution: {integrity: sha512-djYB+Zx2vLewY8RWlNCUdHjDXs2XOgm602S9E7P/UpHgfeHL00cRiIF+IN/G/aUJ7kGPb6yO/ErDI5V2s8iycA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + es-shim-unscopables: 1.0.2 + dev: true + + /array.prototype.flatmap@1.3.2: + resolution: {integrity: sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + es-shim-unscopables: 1.0.2 + dev: true + + /array.prototype.tosorted@1.1.2: + resolution: {integrity: sha512-HuQCHOlk1Weat5jzStICBCd83NxiIMwqDg/dHEsoefabn/hJRj5pVdWcPUSpRrwhwxZOsQassMpgN/xRYFBMIg==} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + es-shim-unscopables: 1.0.2 + get-intrinsic: 1.2.2 + dev: true + + /arraybuffer.prototype.slice@1.0.2: + resolution: {integrity: sha512-yMBKppFur/fbHu9/6USUe03bZ4knMYiwFBcyiaXB8Go0qNehwX6inYPzK9U0NeQvGxKthcmHcaR8P5MStSRBAw==} + engines: {node: '>= 0.4'} + dependencies: + array-buffer-byte-length: 1.0.0 + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + get-intrinsic: 1.2.2 + is-array-buffer: 3.0.2 + is-shared-array-buffer: 1.0.2 + dev: true + + /ast-parents@0.0.1: + resolution: {integrity: sha512-XHusKxKz3zoYk1ic8Un640joHbFMhbqneyoZfoKnEGtf2ey9Uh/IdpcQplODdO/kENaMIWsD0nJm4+wX3UNLHA==} + dev: true + + /astral-regex@2.0.0: + resolution: {integrity: sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==} + engines: {node: '>=8'} + + /async@3.2.5: + resolution: {integrity: sha512-baNZyqaaLhyLVKm/DlvdW051MSgO6b8eVfIezl9E5PqWxFgzLm/wQntEW4zOytVburDEr0JlALEpdOFwvErLsg==} + + /available-typed-arrays@1.0.5: + resolution: {integrity: sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==} + engines: {node: '>= 0.4'} + dev: true + + /balanced-match@1.0.2: + resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + + /bech32@1.1.4: + resolution: {integrity: sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==} + + /binary-extensions@2.2.0: + resolution: {integrity: sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==} + engines: {node: '>=8'} + + /bn.js@4.12.0: + resolution: {integrity: sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==} + + /bn.js@5.2.1: + resolution: {integrity: sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==} + + /brace-expansion@1.1.11: + resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} + dependencies: + balanced-match: 1.0.2 + concat-map: 0.0.1 + + /brace-expansion@2.0.1: + resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} + dependencies: + balanced-match: 1.0.2 + + /braces@3.0.2: + resolution: {integrity: sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==} + engines: {node: '>=8'} + dependencies: + fill-range: 7.0.1 + + /brorand@1.1.0: + resolution: {integrity: sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==} + + /browser-headers@0.4.1: + resolution: {integrity: sha512-CA9hsySZVo9371qEHjHZtYxV2cFtVj5Wj/ZHi8ooEsrtm4vOnl9Y9HmyYWk9q+05d7K3rdoAE0j3MVEFVvtQtg==} + + /browserslist@4.22.2: + resolution: {integrity: sha512-0UgcrvQmBDvZHFGdYUehrCNIazki7/lUP3kkoi/r3YB2amZbFM9J43ZRkJTXBUZK4gmx56+Sqk9+Vs9mwZx9+A==} + engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7} + hasBin: true + dependencies: + caniuse-lite: 1.0.30001576 + electron-to-chromium: 1.4.626 + node-releases: 2.0.14 + update-browserslist-db: 1.0.13(browserslist@4.22.2) + dev: true + + /call-bind@1.0.5: + resolution: {integrity: sha512-C3nQxfFZxFRVoJoGKKI8y3MOEo129NQ+FgQ08iye+Mk4zNZZGdjfs06bVTr+DBSlA66Q2VEcMki/cUCP4SercQ==} + dependencies: + function-bind: 1.1.2 + get-intrinsic: 1.2.2 + set-function-length: 1.1.1 + dev: true + + /callsites@3.1.0: + resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} + engines: {node: '>=6'} + dev: true + + /caniuse-lite@1.0.30001576: + resolution: {integrity: sha512-ff5BdakGe2P3SQsMsiqmt1Lc8221NR1VzHj5jXN5vBny9A6fpze94HiVV/n7XRosOlsShJcvMv5mdnpjOGCEgg==} + dev: true + + /chalk@2.4.2: + resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} + engines: {node: '>=4'} + dependencies: + ansi-styles: 3.2.1 + escape-string-regexp: 1.0.5 + supports-color: 5.5.0 + dev: true + + /chalk@4.1.2: + resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + supports-color: 7.2.0 + + /chalk@5.3.0: + resolution: {integrity: sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==} + engines: {node: ^12.17.0 || ^14.13 || >=16.0.0} + + /change-case@5.4.0: + resolution: {integrity: sha512-11YRFf0f4pI+ROHUfq64WivyrcNSrZjdDt2qgVxvAObtj/Pwnu5uAKObHRbN9uAhaDFkvkqcKVEl8Dxnmx+N7w==} + dev: false + + /chokidar@3.5.3: + resolution: {integrity: sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==} + engines: {node: '>= 8.10.0'} + dependencies: + anymatch: 3.1.3 + braces: 3.0.2 + glob-parent: 5.1.2 + is-binary-path: 2.1.0 + is-glob: 4.0.3 + normalize-path: 3.0.0 + readdirp: 3.6.0 + optionalDependencies: + fsevents: 2.3.3 + + /cliui@8.0.1: + resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} + engines: {node: '>=12'} + dependencies: + string-width: 4.2.3 + strip-ansi: 6.0.1 + wrap-ansi: 7.0.0 + + /color-convert@1.9.3: + resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} + dependencies: + color-name: 1.1.3 + dev: true + + /color-convert@2.0.1: + resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} + engines: {node: '>=7.0.0'} + dependencies: + color-name: 1.1.4 + + /color-name@1.1.3: + resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} + dev: true + + /color-name@1.1.4: + resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + + /commander@10.0.1: + resolution: {integrity: sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==} + engines: {node: '>=14'} + dev: true + + /commander@9.5.0: + resolution: {integrity: sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==} + engines: {node: ^12.20.0 || >=14} + dev: true + + /concat-map@0.0.1: + resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + + /convert-source-map@2.0.0: + resolution: {integrity: sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==} + dev: true + + /copy-anything@3.0.5: + resolution: {integrity: sha512-yCEafptTtb4bk7GLEQoM8KVJpxAfdBJYaXyzQEgQQQgYrZiDp8SJmGKlYza6CYjEDNstAdNdKA3UuoULlEbS6w==} + engines: {node: '>=12.13'} + dependencies: + is-what: 4.1.16 + dev: false + + /cosmiconfig@8.3.6(typescript@5.1.6): + resolution: {integrity: sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==} + engines: {node: '>=14'} + peerDependencies: + typescript: '>=4.9.5' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + import-fresh: 3.3.0 + js-yaml: 4.1.0 + parse-json: 5.2.0 + path-type: 4.0.0 + typescript: 5.1.6 + dev: true + + /cross-spawn@7.0.3: + resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} + engines: {node: '>= 8'} + dependencies: + path-key: 3.1.1 + shebang-command: 2.0.0 + which: 2.0.2 + + /csstype@3.1.3: + resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==} + + /debug@4.3.4: + resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + dependencies: + ms: 2.1.2 + + /deep-is@0.1.4: + resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} + dev: true + + /define-data-property@1.1.1: + resolution: {integrity: sha512-E7uGkTzkk1d0ByLeSc6ZsFS79Axg+m1P/VsgYsxHgiuc3tFSj+MjMIwe90FC4lOAZzNBdY7kkO2P2wKdsQ1vgQ==} + engines: {node: '>= 0.4'} + dependencies: + get-intrinsic: 1.2.2 + gopd: 1.0.1 + has-property-descriptors: 1.0.1 + dev: true + + /define-properties@1.2.1: + resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} + engines: {node: '>= 0.4'} + dependencies: + define-data-property: 1.1.1 + has-property-descriptors: 1.0.1 + object-keys: 1.1.1 + dev: true + + /dir-glob@3.0.1: + resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} + engines: {node: '>=8'} + dependencies: + path-type: 4.0.0 + dev: true + + /doctrine@2.1.0: + resolution: {integrity: sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==} + engines: {node: '>=0.10.0'} + dependencies: + esutils: 2.0.3 + dev: true + + /doctrine@3.0.0: + resolution: {integrity: sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==} + engines: {node: '>=6.0.0'} + dependencies: + esutils: 2.0.3 + dev: true + + /dotenv@16.3.1: + resolution: {integrity: sha512-IPzF4w4/Rd94bA9imS68tZBaYyBWSCE47V1RGuMrB94iyTOIEwRmVL2x/4An+6mETpLrKJ5hQkB8W4kFAadeIQ==} + engines: {node: '>=12'} + + /drizzle-orm@0.28.6(kysely@0.26.3)(postgres@3.4.3)(sql.js@1.9.0): + resolution: {integrity: sha512-yBe+F9htrlYER7uXgDJUQsTHFoIrI5yMm5A0bg0GiZ/kY5jNXTWoEy4KQtg35cE27sw1VbgzoMWHAgCckUUUww==} + peerDependencies: + '@aws-sdk/client-rds-data': '>=3' + '@cloudflare/workers-types': '>=3' + '@libsql/client': '*' + '@neondatabase/serverless': '>=0.1' + '@opentelemetry/api': ^1.4.1 + '@planetscale/database': '>=1' + '@types/better-sqlite3': '*' + '@types/pg': '*' + '@types/sql.js': '*' + '@vercel/postgres': '*' + better-sqlite3: '>=7' + bun-types: '*' + knex: '*' + kysely: '*' + mysql2: '>=2' + pg: '>=8' + postgres: '>=3' + sql.js: '>=1' + sqlite3: '>=5' + peerDependenciesMeta: + '@aws-sdk/client-rds-data': + optional: true + '@cloudflare/workers-types': + optional: true + '@libsql/client': + optional: true + '@neondatabase/serverless': + optional: true + '@opentelemetry/api': + optional: true + '@planetscale/database': + optional: true + '@types/better-sqlite3': + optional: true + '@types/pg': + optional: true + '@types/sql.js': + optional: true + '@vercel/postgres': + optional: true + better-sqlite3: + optional: true + bun-types: + optional: true + knex: + optional: true + kysely: + optional: true + mysql2: + optional: true + pg: + optional: true + postgres: + optional: true + sql.js: + optional: true + sqlite3: + optional: true + dependencies: + kysely: 0.26.3 + postgres: 3.4.3 + sql.js: 1.9.0 + dev: false + + /ejs@3.1.9: + resolution: {integrity: sha512-rC+QVNMJWv+MtPgkt0y+0rVEIdbtxVADApW9JXrUVlzHetgcyczP/E7DJmWJ4fJCZF2cPcBk0laWO9ZHMG3DmQ==} + engines: {node: '>=0.10.0'} + hasBin: true + dependencies: + jake: 10.8.7 + + /electron-to-chromium@1.4.626: + resolution: {integrity: sha512-f7/be56VjRRQk+Ric6PmIrEtPcIqsn3tElyAu9Sh6egha2VLJ82qwkcOdcnT06W+Pb6RUulV1ckzrGbKzVcTHg==} + dev: true + + /elliptic@6.5.4: + resolution: {integrity: sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==} + dependencies: + bn.js: 4.12.0 + brorand: 1.1.0 + hash.js: 1.1.7 + hmac-drbg: 1.0.1 + inherits: 2.0.4 + minimalistic-assert: 1.0.1 + minimalistic-crypto-utils: 1.0.1 + + /emoji-regex@8.0.0: + resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} + + /error-ex@1.3.2: + resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} + dependencies: + is-arrayish: 0.2.1 + dev: true + + /es-abstract@1.22.3: + resolution: {integrity: sha512-eiiY8HQeYfYH2Con2berK+To6GrK2RxbPawDkGq4UiCQQfZHb6wX9qQqkbpPqaxQFcl8d9QzZqo0tGE0VcrdwA==} + engines: {node: '>= 0.4'} + dependencies: + array-buffer-byte-length: 1.0.0 + arraybuffer.prototype.slice: 1.0.2 + available-typed-arrays: 1.0.5 + call-bind: 1.0.5 + es-set-tostringtag: 2.0.2 + es-to-primitive: 1.2.1 + function.prototype.name: 1.1.6 + get-intrinsic: 1.2.2 + get-symbol-description: 1.0.0 + globalthis: 1.0.3 + gopd: 1.0.1 + has-property-descriptors: 1.0.1 + has-proto: 1.0.1 + has-symbols: 1.0.3 + hasown: 2.0.0 + internal-slot: 1.0.6 + is-array-buffer: 3.0.2 + is-callable: 1.2.7 + is-negative-zero: 2.0.2 + is-regex: 1.1.4 + is-shared-array-buffer: 1.0.2 + is-string: 1.0.7 + is-typed-array: 1.1.12 + is-weakref: 1.0.2 + object-inspect: 1.13.1 + object-keys: 1.1.1 + object.assign: 4.1.5 + regexp.prototype.flags: 1.5.1 + safe-array-concat: 1.0.1 + safe-regex-test: 1.0.1 + string.prototype.trim: 1.2.8 + string.prototype.trimend: 1.0.7 + string.prototype.trimstart: 1.0.7 + typed-array-buffer: 1.0.0 + typed-array-byte-length: 1.0.0 + typed-array-byte-offset: 1.0.0 + typed-array-length: 1.0.4 + unbox-primitive: 1.0.2 + which-typed-array: 1.1.13 + dev: true + + /es-set-tostringtag@2.0.2: + resolution: {integrity: sha512-BuDyupZt65P9D2D2vA/zqcI3G5xRsklm5N3xCwuiy+/vKy8i0ifdsQP1sLgO4tZDSCaQUSnmC48khknGMV3D2Q==} + engines: {node: '>= 0.4'} + dependencies: + get-intrinsic: 1.2.2 + has-tostringtag: 1.0.0 + hasown: 2.0.0 + dev: true + + /es-shim-unscopables@1.0.2: + resolution: {integrity: sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==} + dependencies: + hasown: 2.0.0 + dev: true + + /es-to-primitive@1.2.1: + resolution: {integrity: sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==} + engines: {node: '>= 0.4'} + dependencies: + is-callable: 1.2.7 + is-date-object: 1.0.5 + is-symbol: 1.0.4 + dev: true + + /esbuild@0.17.19: + resolution: {integrity: sha512-XQ0jAPFkK/u3LcVRcvVHQcTIqD6E2H1fvZMA5dQPSOWb3suUbWbfbRf94pjc0bNzRYLfIrDRQXr7X+LHIm5oHw==} + engines: {node: '>=12'} + hasBin: true + requiresBuild: true + optionalDependencies: + '@esbuild/android-arm': 0.17.19 + '@esbuild/android-arm64': 0.17.19 + '@esbuild/android-x64': 0.17.19 + '@esbuild/darwin-arm64': 0.17.19 + '@esbuild/darwin-x64': 0.17.19 + '@esbuild/freebsd-arm64': 0.17.19 + '@esbuild/freebsd-x64': 0.17.19 + '@esbuild/linux-arm': 0.17.19 + '@esbuild/linux-arm64': 0.17.19 + '@esbuild/linux-ia32': 0.17.19 + '@esbuild/linux-loong64': 0.17.19 + '@esbuild/linux-mips64el': 0.17.19 + '@esbuild/linux-ppc64': 0.17.19 + '@esbuild/linux-riscv64': 0.17.19 + '@esbuild/linux-s390x': 0.17.19 + '@esbuild/linux-x64': 0.17.19 + '@esbuild/netbsd-x64': 0.17.19 + '@esbuild/openbsd-x64': 0.17.19 + '@esbuild/sunos-x64': 0.17.19 + '@esbuild/win32-arm64': 0.17.19 + '@esbuild/win32-ia32': 0.17.19 + '@esbuild/win32-x64': 0.17.19 + + /esbuild@0.18.20: + resolution: {integrity: sha512-ceqxoedUrcayh7Y7ZX6NdbbDzGROiyVBgC4PriJThBKSVPWnnFHZAkfI1lJT8QFkOwH4qOS2SJkS4wvpGl8BpA==} + engines: {node: '>=12'} + hasBin: true + requiresBuild: true + optionalDependencies: + '@esbuild/android-arm': 0.18.20 + '@esbuild/android-arm64': 0.18.20 + '@esbuild/android-x64': 0.18.20 + '@esbuild/darwin-arm64': 0.18.20 + '@esbuild/darwin-x64': 0.18.20 + '@esbuild/freebsd-arm64': 0.18.20 + '@esbuild/freebsd-x64': 0.18.20 + '@esbuild/linux-arm': 0.18.20 + '@esbuild/linux-arm64': 0.18.20 + '@esbuild/linux-ia32': 0.18.20 + '@esbuild/linux-loong64': 0.18.20 + '@esbuild/linux-mips64el': 0.18.20 + '@esbuild/linux-ppc64': 0.18.20 + '@esbuild/linux-riscv64': 0.18.20 + '@esbuild/linux-s390x': 0.18.20 + '@esbuild/linux-x64': 0.18.20 + '@esbuild/netbsd-x64': 0.18.20 + '@esbuild/openbsd-x64': 0.18.20 + '@esbuild/sunos-x64': 0.18.20 + '@esbuild/win32-arm64': 0.18.20 + '@esbuild/win32-ia32': 0.18.20 + '@esbuild/win32-x64': 0.18.20 + dev: true + + /escalade@3.1.1: + resolution: {integrity: sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==} + engines: {node: '>=6'} + + /escape-string-regexp@1.0.5: + resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} + engines: {node: '>=0.8.0'} + dev: true + + /escape-string-regexp@4.0.0: + resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} + engines: {node: '>=10'} + dev: true + + /eslint-plugin-react-hooks@4.6.0(eslint@8.29.0): + resolution: {integrity: sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==} + engines: {node: '>=10'} + peerDependencies: + eslint: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 + dependencies: + eslint: 8.29.0 + dev: true + + /eslint-plugin-react@7.31.11(eslint@8.29.0): + resolution: {integrity: sha512-TTvq5JsT5v56wPa9OYHzsrOlHzKZKjV+aLgS+55NJP/cuzdiQPC7PfYoUjMoxlffKtvijpk7vA/jmuqRb9nohw==} + engines: {node: '>=4'} + peerDependencies: + eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 + dependencies: + array-includes: 3.1.7 + array.prototype.flatmap: 1.3.2 + array.prototype.tosorted: 1.1.2 + doctrine: 2.1.0 + eslint: 8.29.0 + estraverse: 5.3.0 + jsx-ast-utils: 3.3.5 + minimatch: 3.1.2 + object.entries: 1.1.7 + object.fromentries: 2.0.7 + object.hasown: 1.1.3 + object.values: 1.1.7 + prop-types: 15.8.1 + resolve: 2.0.0-next.5 + semver: 6.3.1 + string.prototype.matchall: 4.0.10 + dev: true + + /eslint-scope@5.1.1: + resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} + engines: {node: '>=8.0.0'} + dependencies: + esrecurse: 4.3.0 + estraverse: 4.3.0 + dev: true + + /eslint-scope@7.2.2: + resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + esrecurse: 4.3.0 + estraverse: 5.3.0 + dev: true + + /eslint-utils@3.0.0(eslint@8.29.0): + resolution: {integrity: sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==} + engines: {node: ^10.0.0 || ^12.0.0 || >= 14.0.0} + peerDependencies: + eslint: '>=5' + dependencies: + eslint: 8.29.0 + eslint-visitor-keys: 2.1.0 + dev: true + + /eslint-visitor-keys@2.1.0: + resolution: {integrity: sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==} + engines: {node: '>=10'} + dev: true + + /eslint-visitor-keys@3.4.3: + resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dev: true + + /eslint@8.29.0: + resolution: {integrity: sha512-isQ4EEiyUjZFbEKvEGJKKGBwXtvXX+zJbkVKCgTuB9t/+jUBcy8avhkEwWJecI15BkRkOYmvIM5ynbhRjEkoeg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + hasBin: true + dependencies: + '@eslint/eslintrc': 1.4.1 + '@humanwhocodes/config-array': 0.11.13 + '@humanwhocodes/module-importer': 1.0.1 + '@nodelib/fs.walk': 1.2.8 + ajv: 6.12.6 + chalk: 4.1.2 + cross-spawn: 7.0.3 + debug: 4.3.4 + doctrine: 3.0.0 + escape-string-regexp: 4.0.0 + eslint-scope: 7.2.2 + eslint-utils: 3.0.0(eslint@8.29.0) + eslint-visitor-keys: 3.4.3 + espree: 9.6.1 + esquery: 1.5.0 + esutils: 2.0.3 + fast-deep-equal: 3.1.3 + file-entry-cache: 6.0.1 + find-up: 5.0.0 + glob-parent: 6.0.2 + globals: 13.24.0 + grapheme-splitter: 1.0.4 + ignore: 5.3.0 + import-fresh: 3.3.0 + imurmurhash: 0.1.4 + is-glob: 4.0.3 + is-path-inside: 3.0.3 + js-sdsl: 4.4.2 + js-yaml: 4.1.0 + json-stable-stringify-without-jsonify: 1.0.1 + levn: 0.4.1 + lodash.merge: 4.6.2 + minimatch: 3.1.2 + natural-compare: 1.4.0 + optionator: 0.9.3 + regexpp: 3.2.0 + strip-ansi: 6.0.1 + strip-json-comments: 3.1.1 + text-table: 0.2.0 + transitivePeerDependencies: + - supports-color + dev: true + + /espree@9.6.1: + resolution: {integrity: sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + acorn: 8.11.3 + acorn-jsx: 5.3.2(acorn@8.11.3) + eslint-visitor-keys: 3.4.3 + dev: true + + /esquery@1.5.0: + resolution: {integrity: sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==} + engines: {node: '>=0.10'} + dependencies: + estraverse: 5.3.0 + dev: true + + /esrecurse@4.3.0: + resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} + engines: {node: '>=4.0'} + dependencies: + estraverse: 5.3.0 + dev: true + + /estraverse@4.3.0: + resolution: {integrity: sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==} + engines: {node: '>=4.0'} + dev: true + + /estraverse@5.3.0: + resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} + engines: {node: '>=4.0'} + dev: true + + /esutils@2.0.3: + resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} + engines: {node: '>=0.10.0'} + dev: true + + /ethers@5.7.2: + resolution: {integrity: sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==} + dependencies: + '@ethersproject/abi': 5.7.0 + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/base64': 5.7.0 + '@ethersproject/basex': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/contracts': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/hdnode': 5.7.0 + '@ethersproject/json-wallets': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/networks': 5.7.1 + '@ethersproject/pbkdf2': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/providers': 5.7.2 + '@ethersproject/random': 5.7.0 + '@ethersproject/rlp': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + '@ethersproject/solidity': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/units': 5.7.0 + '@ethersproject/wallet': 5.7.0 + '@ethersproject/web': 5.7.1 + '@ethersproject/wordlists': 5.7.0 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + /eventemitter3@5.0.1: + resolution: {integrity: sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==} + + /execa@7.2.0: + resolution: {integrity: sha512-UduyVP7TLB5IcAQl+OzLyLcS/l32W/GLg+AhHJ+ow40FOk2U3SAllPwR44v4vmdFwIWqpdwxxpQbF1n5ta9seA==} + engines: {node: ^14.18.0 || ^16.14.0 || >=18.0.0} + dependencies: + cross-spawn: 7.0.3 + get-stream: 6.0.1 + human-signals: 4.3.1 + is-stream: 3.0.0 + merge-stream: 2.0.0 + npm-run-path: 5.2.0 + onetime: 6.0.0 + signal-exit: 3.0.7 + strip-final-newline: 3.0.0 + + /fast-deep-equal@3.1.3: + resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} + + /fast-diff@1.3.0: + resolution: {integrity: sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==} + dev: true + + /fast-glob@3.3.2: + resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} + engines: {node: '>=8.6.0'} + dependencies: + '@nodelib/fs.stat': 2.0.5 + '@nodelib/fs.walk': 1.2.8 + glob-parent: 5.1.2 + merge2: 1.4.1 + micromatch: 4.0.5 + dev: true + + /fast-json-stable-stringify@2.1.0: + resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} + dev: true + + /fast-levenshtein@2.0.6: + resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==} + dev: true + + /fastq@1.16.0: + resolution: {integrity: sha512-ifCoaXsDrsdkWTtiNJX5uzHDsrck5TzfKKDcuFFTIrrc/BS076qgEIfoIy1VeZqViznfKiysPYTh/QeHtnIsYA==} + dependencies: + reusify: 1.0.4 + dev: true + + /file-entry-cache@6.0.1: + resolution: {integrity: sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==} + engines: {node: ^10.12.0 || >=12.0.0} + dependencies: + flat-cache: 3.2.0 + dev: true + + /filelist@1.0.4: + resolution: {integrity: sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==} + dependencies: + minimatch: 5.1.6 + + /fill-range@7.0.1: + resolution: {integrity: sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==} + engines: {node: '>=8'} + dependencies: + to-regex-range: 5.0.1 + + /find-up@5.0.0: + resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} + engines: {node: '>=10'} + dependencies: + locate-path: 6.0.0 + path-exists: 4.0.0 + dev: true + + /find-up@6.3.0: + resolution: {integrity: sha512-v2ZsoEuVHYy8ZIlYqwPe/39Cy+cFDzp4dXPaxNvkEuouymu+2Jbz0PxpKarJHYJTmv2HWT3O382qY8l4jMWthw==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + locate-path: 7.2.0 + path-exists: 5.0.0 + + /flat-cache@3.2.0: + resolution: {integrity: sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==} + engines: {node: ^10.12.0 || >=12.0.0} + dependencies: + flatted: 3.2.9 + keyv: 4.5.4 + rimraf: 3.0.2 + dev: true + + /flatted@3.2.9: + resolution: {integrity: sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ==} + dev: true + + /for-each@0.3.3: + resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==} + dependencies: + is-callable: 1.2.7 + dev: true + + /fs.realpath@1.0.0: + resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} + + /fsevents@2.3.3: + resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} + engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + os: [darwin] + requiresBuild: true + optional: true + + /function-bind@1.1.2: + resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} + dev: true + + /function.prototype.name@1.1.6: + resolution: {integrity: sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + functions-have-names: 1.2.3 + dev: true + + /functions-have-names@1.2.3: + resolution: {integrity: sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==} + dev: true + + /gensync@1.0.0-beta.2: + resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==} + engines: {node: '>=6.9.0'} + dev: true + + /get-caller-file@2.0.5: + resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} + engines: {node: 6.* || 8.* || >= 10.*} + + /get-intrinsic@1.2.2: + resolution: {integrity: sha512-0gSo4ml/0j98Y3lngkFEot/zhiCeWsbYIlZ+uZOVgzLyLaUw7wxUL+nCTP0XJvJg1AXulJRI3UJi8GsbDuxdGA==} + dependencies: + function-bind: 1.1.2 + has-proto: 1.0.1 + has-symbols: 1.0.3 + hasown: 2.0.0 + dev: true + + /get-stream@6.0.1: + resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} + engines: {node: '>=10'} + + /get-symbol-description@1.0.0: + resolution: {integrity: sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + get-intrinsic: 1.2.2 + dev: true + + /glob-parent@5.1.2: + resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} + engines: {node: '>= 6'} + dependencies: + is-glob: 4.0.3 + + /glob-parent@6.0.2: + resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} + engines: {node: '>=10.13.0'} + dependencies: + is-glob: 4.0.3 + dev: true + + /glob@7.2.3: + resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 3.1.2 + once: 1.4.0 + path-is-absolute: 1.0.1 + dev: true + + /glob@8.1.0: + resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} + engines: {node: '>=12'} + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 5.1.6 + once: 1.4.0 + + /globals@11.12.0: + resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==} + engines: {node: '>=4'} + dev: true + + /globals@13.24.0: + resolution: {integrity: sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==} + engines: {node: '>=8'} + dependencies: + type-fest: 0.20.2 + dev: true + + /globalthis@1.0.3: + resolution: {integrity: sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==} + engines: {node: '>= 0.4'} + dependencies: + define-properties: 1.2.1 + dev: true + + /globby@11.1.0: + resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} + engines: {node: '>=10'} + dependencies: + array-union: 2.1.0 + dir-glob: 3.0.1 + fast-glob: 3.3.2 + ignore: 5.3.0 + merge2: 1.4.1 + slash: 3.0.0 + dev: true + + /google-protobuf@3.21.2: + resolution: {integrity: sha512-3MSOYFO5U9mPGikIYCzK0SaThypfGgS6bHqrUGXG3DPHCrb+txNqeEcns1W0lkGfk0rCyNXm7xB9rMxnCiZOoA==} + + /gopd@1.0.1: + resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} + dependencies: + get-intrinsic: 1.2.2 + dev: true + + /grapheme-splitter@1.0.4: + resolution: {integrity: sha512-bzh50DW9kTPM00T8y4o8vQg89Di9oLJVLW/KaOGIXJWP/iqCN6WKYkbNOF04vFLJhwcpYUh9ydh/+5vpOqV4YQ==} + dev: true + + /has-bigints@1.0.2: + resolution: {integrity: sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==} + dev: true + + /has-flag@3.0.0: + resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} + engines: {node: '>=4'} + dev: true + + /has-flag@4.0.0: + resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} + engines: {node: '>=8'} + + /has-property-descriptors@1.0.1: + resolution: {integrity: sha512-VsX8eaIewvas0xnvinAe9bw4WfIeODpGYikiWYLH+dma0Jw6KHYqWiWfhQlgOVK8D6PvjubK5Uc4P0iIhIcNVg==} + dependencies: + get-intrinsic: 1.2.2 + dev: true + + /has-proto@1.0.1: + resolution: {integrity: sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==} + engines: {node: '>= 0.4'} + dev: true + + /has-symbols@1.0.3: + resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} + engines: {node: '>= 0.4'} + dev: true + + /has-tostringtag@1.0.0: + resolution: {integrity: sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==} + engines: {node: '>= 0.4'} + dependencies: + has-symbols: 1.0.3 + dev: true + + /hash.js@1.1.7: + resolution: {integrity: sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==} + dependencies: + inherits: 2.0.4 + minimalistic-assert: 1.0.1 + + /hasown@2.0.0: + resolution: {integrity: sha512-vUptKVTpIJhcczKBbgnS+RtcuYMB8+oNzPK2/Hp3hanz8JmpATdmmgLgSaadVREkDm+e2giHwY3ZRkyjSIDDFA==} + engines: {node: '>= 0.4'} + dependencies: + function-bind: 1.1.2 + dev: true + + /hmac-drbg@1.0.1: + resolution: {integrity: sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==} + dependencies: + hash.js: 1.1.7 + minimalistic-assert: 1.0.1 + minimalistic-crypto-utils: 1.0.1 + + /human-signals@4.3.1: + resolution: {integrity: sha512-nZXjEF2nbo7lIw3mgYjItAfgQXog3OjJogSbKa2CQIIvSGWcKgeJnQlNXip6NglNzYH45nSRiEVimMvYL8DDqQ==} + engines: {node: '>=14.18.0'} + + /ignore@5.3.0: + resolution: {integrity: sha512-g7dmpshy+gD7mh88OC9NwSGTKoc3kyLAZQRU1mt53Aw/vnvfXnbC+F/7F7QoYVKbV+KNvJx8wArewKy1vXMtlg==} + engines: {node: '>= 4'} + dev: true + + /import-fresh@3.3.0: + resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} + engines: {node: '>=6'} + dependencies: + parent-module: 1.0.1 + resolve-from: 4.0.0 + dev: true + + /imurmurhash@0.1.4: + resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} + engines: {node: '>=0.8.19'} + dev: true + + /inflight@1.0.6: + resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} + dependencies: + once: 1.4.0 + wrappy: 1.0.2 + + /inherits@2.0.3: + resolution: {integrity: sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==} + + /inherits@2.0.4: + resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} + + /internal-slot@1.0.6: + resolution: {integrity: sha512-Xj6dv+PsbtwyPpEflsejS+oIZxmMlV44zAhG479uYu89MsjcYOhCFnNyKrkJrihbsiasQyY0afoCl/9BLR65bg==} + engines: {node: '>= 0.4'} + dependencies: + get-intrinsic: 1.2.2 + hasown: 2.0.0 + side-channel: 1.0.4 + dev: true + + /is-array-buffer@3.0.2: + resolution: {integrity: sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==} + dependencies: + call-bind: 1.0.5 + get-intrinsic: 1.2.2 + is-typed-array: 1.1.12 + dev: true + + /is-arrayish@0.2.1: + resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} + dev: true + + /is-bigint@1.0.4: + resolution: {integrity: sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==} + dependencies: + has-bigints: 1.0.2 + dev: true + + /is-binary-path@2.1.0: + resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} + engines: {node: '>=8'} + dependencies: + binary-extensions: 2.2.0 + + /is-boolean-object@1.1.2: + resolution: {integrity: sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + has-tostringtag: 1.0.0 + dev: true + + /is-callable@1.2.7: + resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} + engines: {node: '>= 0.4'} + dev: true + + /is-core-module@2.13.1: + resolution: {integrity: sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==} + dependencies: + hasown: 2.0.0 + dev: true + + /is-date-object@1.0.5: + resolution: {integrity: sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==} + engines: {node: '>= 0.4'} + dependencies: + has-tostringtag: 1.0.0 + dev: true + + /is-extglob@2.1.1: + resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} + engines: {node: '>=0.10.0'} + + /is-fullwidth-code-point@3.0.0: + resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} + engines: {node: '>=8'} + + /is-glob@4.0.3: + resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} + engines: {node: '>=0.10.0'} + dependencies: + is-extglob: 2.1.1 + + /is-negative-zero@2.0.2: + resolution: {integrity: sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==} + engines: {node: '>= 0.4'} + dev: true + + /is-number-object@1.0.7: + resolution: {integrity: sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==} + engines: {node: '>= 0.4'} + dependencies: + has-tostringtag: 1.0.0 + dev: true + + /is-number@7.0.0: + resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} + engines: {node: '>=0.12.0'} + + /is-path-inside@3.0.3: + resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==} + engines: {node: '>=8'} + dev: true + + /is-regex@1.1.4: + resolution: {integrity: sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + has-tostringtag: 1.0.0 + dev: true + + /is-shared-array-buffer@1.0.2: + resolution: {integrity: sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==} + dependencies: + call-bind: 1.0.5 + dev: true + + /is-stream@3.0.0: + resolution: {integrity: sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + + /is-string@1.0.7: + resolution: {integrity: sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==} + engines: {node: '>= 0.4'} + dependencies: + has-tostringtag: 1.0.0 + dev: true + + /is-symbol@1.0.4: + resolution: {integrity: sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==} + engines: {node: '>= 0.4'} + dependencies: + has-symbols: 1.0.3 + dev: true + + /is-typed-array@1.1.12: + resolution: {integrity: sha512-Z14TF2JNG8Lss5/HMqt0//T9JeHXttXy5pH/DBU4vi98ozO2btxzq9MwYDZYnKwU8nRsz/+GVFVRDq3DkVuSPg==} + engines: {node: '>= 0.4'} + dependencies: + which-typed-array: 1.1.13 + dev: true + + /is-weakref@1.0.2: + resolution: {integrity: sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==} + dependencies: + call-bind: 1.0.5 + dev: true + + /is-what@4.1.16: + resolution: {integrity: sha512-ZhMwEosbFJkA0YhFnNDgTM4ZxDRsS6HqTo7qsZM08fehyRYIYa0yHu5R6mgo1n/8MgaPBXiPimPD77baVFYg+A==} + engines: {node: '>=12.13'} + dev: false + + /isarray@2.0.5: + resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} + dev: true + + /isexe@2.0.0: + resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} + + /isomorphic-ws@5.0.0(ws@8.13.0): + resolution: {integrity: sha512-muId7Zzn9ywDsyXgTIafTry2sV3nySZeUDe6YedVd1Hvuuep5AsIlqK+XefWpYTyJG5e503F2xIuT2lcU6rCSw==} + peerDependencies: + ws: '*' + dependencies: + ws: 8.13.0 + + /jake@10.8.7: + resolution: {integrity: sha512-ZDi3aP+fG/LchyBzUM804VjddnwfSfsdeYkwt8NcbKRvo4rFkjhs456iLFn3k2ZUWvNe4i48WACDbza8fhq2+w==} + engines: {node: '>=10'} + hasBin: true + dependencies: + async: 3.2.5 + chalk: 4.1.2 + filelist: 1.0.4 + minimatch: 3.1.2 + + /js-base64@3.7.5: + resolution: {integrity: sha512-3MEt5DTINKqfScXKfJFrRbxkrnk2AxPWGBL/ycjz4dK8iqiSJ06UxD8jh8xuh6p10TX4t2+7FsBYVxxQbMg+qA==} + + /js-sdsl@4.4.2: + resolution: {integrity: sha512-dwXFwByc/ajSV6m5bcKAPwe4yDDF6D614pxmIi5odytzxRlwqF6nwoiCek80Ixc7Cvma5awClxrzFtxCQvcM8w==} + dev: true + + /js-sha3@0.8.0: + resolution: {integrity: sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==} + + /js-tokens@4.0.0: + resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} + + /js-yaml@4.1.0: + resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} + hasBin: true + dependencies: + argparse: 2.0.1 + dev: true + + /jsesc@2.5.2: + resolution: {integrity: sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==} + engines: {node: '>=4'} + hasBin: true + dev: true + + /json-buffer@3.0.1: + resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} + dev: true + + /json-parse-even-better-errors@2.3.1: + resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} + dev: true + + /json-schema-traverse@0.4.1: + resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} + dev: true + + /json-schema-traverse@1.0.0: + resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} + + /json-stable-stringify-without-jsonify@1.0.1: + resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} + dev: true + + /json5@2.2.3: + resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} + engines: {node: '>=6'} + hasBin: true + dev: true + + /jsx-ast-utils@3.3.5: + resolution: {integrity: sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==} + engines: {node: '>=4.0'} + dependencies: + array-includes: 3.1.7 + array.prototype.flat: 1.3.2 + object.assign: 4.1.5 + object.values: 1.1.7 + dev: true + + /keyv@4.5.4: + resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} + dependencies: + json-buffer: 3.0.1 + dev: true + + /kysely@0.26.3: + resolution: {integrity: sha512-yWSgGi9bY13b/W06DD2OCDDHQmq1kwTGYlQ4wpZkMOJqMGCstVCFIvxCCVG4KfY1/3G0MhDAcZsip/Lw8/vJWw==} + engines: {node: '>=14.0.0'} + dev: false + + /levn@0.4.1: + resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} + engines: {node: '>= 0.8.0'} + dependencies: + prelude-ls: 1.2.1 + type-check: 0.4.0 + dev: true + + /lines-and-columns@1.2.4: + resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} + dev: true + + /locate-path@6.0.0: + resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} + engines: {node: '>=10'} + dependencies: + p-locate: 5.0.0 + dev: true + + /locate-path@7.2.0: + resolution: {integrity: sha512-gvVijfZvn7R+2qyPX8mAuKcFGDf6Nc61GdvGafQsHL0sBIxfKzA+usWn4GFC/bk+QdwPUD4kWFJLhElipq+0VA==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + p-locate: 6.0.0 + + /lodash.merge@4.6.2: + resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} + dev: true + + /lodash.truncate@4.4.2: + resolution: {integrity: sha512-jttmRe7bRse52OsWIMDLaXxWqRAmtIUccAQ3garviCqJjafXOfNMO0yMfNpdD6zbGaTU0P5Nz7e7gAT6cKmJRw==} + + /lodash@4.17.21: + resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} + dev: true + + /long@5.2.3: + resolution: {integrity: sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q==} + + /loose-envify@1.4.0: + resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} + hasBin: true + dependencies: + js-tokens: 4.0.0 + + /lru-cache@5.1.1: + resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} + dependencies: + yallist: 3.1.1 + dev: true + + /lru-cache@6.0.0: + resolution: {integrity: sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==} + engines: {node: '>=10'} + dependencies: + yallist: 4.0.0 + + /magic-string@0.27.0: + resolution: {integrity: sha512-8UnnX2PeRAPZuN12svgR9j7M1uWMovg/CEnIwIG0LFkXSJJe4PdfUGiTGl8V9bsBHFUtfVINcSyYxd7q+kx9fA==} + engines: {node: '>=12'} + dependencies: + '@jridgewell/sourcemap-codec': 1.4.15 + dev: true + + /merge-stream@2.0.0: + resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==} + + /merge2@1.4.1: + resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} + engines: {node: '>= 8'} + dev: true + + /micromatch@4.0.5: + resolution: {integrity: sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==} + engines: {node: '>=8.6'} + dependencies: + braces: 3.0.2 + picomatch: 2.3.1 + dev: true + + /mimic-fn@4.0.0: + resolution: {integrity: sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw==} + engines: {node: '>=12'} + + /minimalistic-assert@1.0.1: + resolution: {integrity: sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==} + + /minimalistic-crypto-utils@1.0.1: + resolution: {integrity: sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==} + + /minimatch@3.1.2: + resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} + dependencies: + brace-expansion: 1.1.11 + + /minimatch@5.1.6: + resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} + engines: {node: '>=10'} + dependencies: + brace-expansion: 2.0.1 + + /mobx@6.12.0: + resolution: {integrity: sha512-Mn6CN6meXEnMa0a5u6a5+RKrqRedHBhZGd15AWLk9O6uFY4KYHzImdt8JI8WODo1bjTSRnwXhJox+FCUZhCKCQ==} + + /mprocs@0.6.4: + resolution: {integrity: sha512-Y4eqnAjp3mjy0eT+zPoMQ+P/ISOzjgRG/4kh4I5cRA4Tv0rPxTCBRadn3+j+boMF5id7IoLhrVq9NFWFPuzD9A==} + engines: {node: '>=0.10.0'} + hasBin: true + dev: true + + /ms@2.1.2: + resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} + + /nanoid@3.3.7: + resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==} + engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} + hasBin: true + dev: true + + /natural-compare-lite@1.4.0: + resolution: {integrity: sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==} + dev: true + + /natural-compare@1.4.0: + resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} + dev: true + + /nice-grpc-common@2.0.2: + resolution: {integrity: sha512-7RNWbls5kAL1QVUOXvBsv1uO0wPQK3lHv+cY1gwkTzirnG1Nop4cBJZubpgziNbaVc/bl9QJcyvsf/NQxa3rjQ==} + dependencies: + ts-error: 1.0.6 + + /nice-grpc-web@2.0.2(google-protobuf@3.21.2): + resolution: {integrity: sha512-hIgvd0ufjOR1FqrakqQ1HdO6A8M1rps9cTCnO+8aPml2NXyLOsuZCdzCp/f31Y1ngBGvDkoalW5H+CBd11x+GQ==} + dependencies: + '@improbable-eng/grpc-web': 0.15.0(google-protobuf@3.21.2) + abort-controller-x: 0.4.3 + js-base64: 3.7.5 + nice-grpc-common: 2.0.2 + transitivePeerDependencies: + - google-protobuf + + /node-releases@2.0.14: + resolution: {integrity: sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==} + dev: true + + /normalize-path@3.0.0: + resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} + engines: {node: '>=0.10.0'} + + /npm-run-path@5.2.0: + resolution: {integrity: sha512-W4/tgAXFqFA0iL7fk0+uQ3g7wkL8xJmx3XdK0VGb4cHW//eZTtKGvFBBoRKVTpY7n6ze4NL9ly7rgXcHufqXKg==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + path-key: 4.0.0 + + /object-assign@4.1.1: + resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} + engines: {node: '>=0.10.0'} + dev: true + + /object-inspect@1.13.1: + resolution: {integrity: sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==} + dev: true + + /object-keys@1.1.1: + resolution: {integrity: sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==} + engines: {node: '>= 0.4'} + dev: true + + /object.assign@4.1.5: + resolution: {integrity: sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + has-symbols: 1.0.3 + object-keys: 1.1.1 + dev: true + + /object.entries@1.1.7: + resolution: {integrity: sha512-jCBs/0plmPsOnrKAfFQXRG2NFjlhZgjjcBLSmTnEhU8U6vVTsVe8ANeQJCHTl3gSsI4J+0emOoCgoKlmQPMgmA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /object.fromentries@2.0.7: + resolution: {integrity: sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /object.hasown@1.1.3: + resolution: {integrity: sha512-fFI4VcYpRHvSLXxP7yiZOMAd331cPfd2p7PFDVbgUsYOfCT3tICVqXWngbjr4m49OvsBwUBQ6O2uQoJvy3RexA==} + dependencies: + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /object.values@1.1.7: + resolution: {integrity: sha512-aU6xnDFYT3x17e/f0IiiwlGPTy2jzMySGfUB4fq6z7CV8l85CWHDk5ErhyhpfDHhrOMwGFhSQkhMGHaIotA6Ng==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /once@1.4.0: + resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} + dependencies: + wrappy: 1.0.2 + + /onetime@6.0.0: + resolution: {integrity: sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ==} + engines: {node: '>=12'} + dependencies: + mimic-fn: 4.0.0 + + /openurl@1.1.1: + resolution: {integrity: sha512-d/gTkTb1i1GKz5k3XE3XFV/PxQ1k45zDqGP2OA7YhgsaLoqm6qRvARAZOFer1fcXritWlGBRCu/UgeS4HAnXAA==} + + /optionator@0.9.3: + resolution: {integrity: sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==} + engines: {node: '>= 0.8.0'} + dependencies: + '@aashutoshrathi/word-wrap': 1.2.6 + deep-is: 0.1.4 + fast-levenshtein: 2.0.6 + levn: 0.4.1 + prelude-ls: 1.2.1 + type-check: 0.4.0 + dev: true + + /p-limit@3.1.0: + resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} + engines: {node: '>=10'} + dependencies: + yocto-queue: 0.1.0 + dev: true + + /p-limit@4.0.0: + resolution: {integrity: sha512-5b0R4txpzjPWVw/cXXUResoD4hb6U/x9BH08L7nw+GN1sezDzPdxeRvpc9c433fZhBan/wusjbCsqwqm4EIBIQ==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + yocto-queue: 1.0.0 + + /p-locate@5.0.0: + resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} + engines: {node: '>=10'} + dependencies: + p-limit: 3.1.0 + dev: true + + /p-locate@6.0.0: + resolution: {integrity: sha512-wPrq66Llhl7/4AGC6I+cqxT07LhXvWL08LNXz1fENOw0Ap4sRZZ/gZpTTJ5jpurzzzfS2W/Ge9BY3LgLjCShcw==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + p-limit: 4.0.0 + + /p-queue@7.4.1: + resolution: {integrity: sha512-vRpMXmIkYF2/1hLBKisKeVYJZ8S2tZ0zEAmIJgdVKP2nq0nh4qCdf8bgw+ZgKrkh71AOCaqzwbJJk1WtdcF3VA==} + engines: {node: '>=12'} + dependencies: + eventemitter3: 5.0.1 + p-timeout: 5.1.0 + + /p-retry@5.1.2: + resolution: {integrity: sha512-couX95waDu98NfNZV+i/iLt+fdVxmI7CbrrdC2uDWfPdUAApyxT4wmDlyOtR5KtTDmkDO0zDScDjDou9YHhd9g==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + '@types/retry': 0.12.1 + retry: 0.13.1 + + /p-timeout@5.1.0: + resolution: {integrity: sha512-auFDyzzzGZZZdHz3BtET9VEz0SE/uMEAx7uWfGPucfzEwwe/xH0iVeZibQmANYE/hp9T2+UUZT5m+BKyrDp3Ew==} + engines: {node: '>=12'} + + /parent-module@1.0.1: + resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} + engines: {node: '>=6'} + dependencies: + callsites: 3.1.0 + dev: true + + /parse-json@5.2.0: + resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} + engines: {node: '>=8'} + dependencies: + '@babel/code-frame': 7.23.5 + error-ex: 1.3.2 + json-parse-even-better-errors: 2.3.1 + lines-and-columns: 1.2.4 + dev: true + + /path-exists@4.0.0: + resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} + engines: {node: '>=8'} + dev: true + + /path-exists@5.0.0: + resolution: {integrity: sha512-RjhtfwJOxzcFmNOi6ltcbcu4Iu+FL3zEj83dk4kAS+fVpTxXLO1b38RvJgT/0QwvV/L3aY9TAnyv0EOqW4GoMQ==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + + /path-is-absolute@1.0.1: + resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==} + engines: {node: '>=0.10.0'} + dev: true + + /path-key@3.1.1: + resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} + engines: {node: '>=8'} + + /path-key@4.0.0: + resolution: {integrity: sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==} + engines: {node: '>=12'} + + /path-parse@1.0.7: + resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} + dev: true + + /path-type@4.0.0: + resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} + engines: {node: '>=8'} + dev: true + + /path@0.12.7: + resolution: {integrity: sha512-aXXC6s+1w7otVF9UletFkFcDsJeO7lSZBPUQhtb5O0xJe8LtYhj/GxldoL09bBj9+ZmE2hNoHqQSFMN5fikh4Q==} + dependencies: + process: 0.11.10 + util: 0.10.4 + + /picocolors@1.0.0: + resolution: {integrity: sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==} + dev: true + + /picomatch@2.3.1: + resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} + engines: {node: '>=8.6'} + + /pluralize@8.0.0: + resolution: {integrity: sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==} + engines: {node: '>=4'} + dev: true + + /postcss@8.4.33: + resolution: {integrity: sha512-Kkpbhhdjw2qQs2O2DGX+8m5OVqEcbB9HRBvuYM9pgrjEFUg30A9LmXNlTAUj4S9kgtGyrMbTzVjH7E+s5Re2yg==} + engines: {node: ^10 || ^12 || >=14} + dependencies: + nanoid: 3.3.7 + picocolors: 1.0.0 + source-map-js: 1.0.2 + dev: true + + /postgres@3.4.3: + resolution: {integrity: sha512-iHJn4+M9vbTdHSdDzNkC0crHq+1CUdFhx+YqCE+SqWxPjm+Zu63jq7yZborOBF64c8pc58O5uMudyL1FQcHacA==} + engines: {node: '>=12'} + dev: false + + /prelude-ls@1.2.1: + resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==} + engines: {node: '>= 0.8.0'} + dev: true + + /prettier-plugin-solidity@1.1.3(prettier@2.8.8): + resolution: {integrity: sha512-fQ9yucPi2sBbA2U2Xjh6m4isUTJ7S7QLc/XDDsktqqxYfTwdYKJ0EnnywXHwCGAaYbQNK+HIYPL1OemxuMsgeg==} + engines: {node: '>=12'} + peerDependencies: + prettier: '>=2.3.0 || >=3.0.0-alpha.0' + dependencies: + '@solidity-parser/parser': 0.16.2 + prettier: 2.8.8 + semver: 7.5.4 + solidity-comments-extractor: 0.0.7 + + /prettier@2.8.8: + resolution: {integrity: sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==} + engines: {node: '>=10.13.0'} + hasBin: true + + /process@0.11.10: + resolution: {integrity: sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==} + engines: {node: '>= 0.6.0'} + + /prop-types@15.8.1: + resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} + dependencies: + loose-envify: 1.4.0 + object-assign: 4.1.1 + react-is: 16.13.1 + dev: true + + /protobufjs@7.2.5: + resolution: {integrity: sha512-gGXRSXvxQ7UiPgfw8gevrfRWcTlSbOFg+p/N+JVJEK5VhueL2miT6qTymqAmjr1Q5WbOCyJbyrk6JfWKwlFn6A==} + engines: {node: '>=12.0.0'} + requiresBuild: true + dependencies: + '@protobufjs/aspromise': 1.1.2 + '@protobufjs/base64': 1.1.2 + '@protobufjs/codegen': 2.0.4 + '@protobufjs/eventemitter': 1.1.0 + '@protobufjs/fetch': 1.1.0 + '@protobufjs/float': 1.0.2 + '@protobufjs/inquire': 1.1.0 + '@protobufjs/path': 1.1.2 + '@protobufjs/pool': 1.1.0 + '@protobufjs/utf8': 1.1.0 + '@types/node': 18.19.6 + long: 5.2.3 + + /proxy-deep@3.1.1: + resolution: {integrity: sha512-kppbvLUNJ4IOMZds9/4gz/rtT5OFiesy3XosLsgMKlF3vb6GA5Y3ptyDlzKLcOcUBW+zaY+RiMINTsgE+O6e+Q==} + + /punycode@2.3.1: + resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} + engines: {node: '>=6'} + + /queue-microtask@1.2.3: + resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} + dev: true + + /react-dom@18.2.0(react@18.2.0): + resolution: {integrity: sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==} + peerDependencies: + react: ^18.2.0 + dependencies: + loose-envify: 1.4.0 + react: 18.2.0 + scheduler: 0.23.0 + dev: false + + /react-is@16.13.1: + resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==} + dev: true + + /react-refresh@0.14.0: + resolution: {integrity: sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==} + engines: {node: '>=0.10.0'} + dev: true + + /react-router-dom@6.21.1(react-dom@18.2.0)(react@18.2.0): + resolution: {integrity: sha512-QCNrtjtDPwHDO+AO21MJd7yIcr41UetYt5jzaB9Y1UYaPTCnVuJq6S748g1dE11OQlCFIQg+RtAA1SEZIyiBeA==} + engines: {node: '>=14.0.0'} + peerDependencies: + react: '>=16.8' + react-dom: '>=16.8' + dependencies: + '@remix-run/router': 1.14.1 + react: 18.2.0 + react-dom: 18.2.0(react@18.2.0) + react-router: 6.21.1(react@18.2.0) + dev: false + + /react-router@6.21.1(react@18.2.0): + resolution: {integrity: sha512-W0l13YlMTm1YrpVIOpjCADJqEUpz1vm+CMo47RuFX4Ftegwm6KOYsL5G3eiE52jnJpKvzm6uB/vTKTPKM8dmkA==} + engines: {node: '>=14.0.0'} + peerDependencies: + react: '>=16.8' + dependencies: + '@remix-run/router': 1.14.1 + react: 18.2.0 + dev: false + + /react@18.2.0: + resolution: {integrity: sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==} + engines: {node: '>=0.10.0'} + dependencies: + loose-envify: 1.4.0 + dev: false + + /readdirp@3.6.0: + resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} + engines: {node: '>=8.10.0'} + dependencies: + picomatch: 2.3.1 + + /regexp.prototype.flags@1.5.1: + resolution: {integrity: sha512-sy6TXMN+hnP/wMy+ISxg3krXx7BAtWVO4UouuCN/ziM9UEne0euamVNafDfvC83bRNr95y0V5iijeDQFUNpvrg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + set-function-name: 2.0.1 + dev: true + + /regexpp@3.2.0: + resolution: {integrity: sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==} + engines: {node: '>=8'} + dev: true + + /require-directory@2.1.1: + resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} + engines: {node: '>=0.10.0'} + + /require-from-string@2.0.2: + resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} + engines: {node: '>=0.10.0'} + + /resolve-from@4.0.0: + resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} + engines: {node: '>=4'} + dev: true + + /resolve@2.0.0-next.5: + resolution: {integrity: sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==} + hasBin: true + dependencies: + is-core-module: 2.13.1 + path-parse: 1.0.7 + supports-preserve-symlinks-flag: 1.0.0 + dev: true + + /retry@0.13.1: + resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} + engines: {node: '>= 4'} + + /reusify@1.0.4: + resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} + engines: {iojs: '>=1.0.0', node: '>=0.10.0'} + dev: true + + /rimraf@3.0.2: + resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==} + hasBin: true + dependencies: + glob: 7.2.3 + dev: true + + /rollup@3.29.4: + resolution: {integrity: sha512-oWzmBZwvYrU0iJHtDmhsm662rC15FRXmcjCk1xD771dFDx5jJ02ufAQQTn0etB2emNk4J9EZg/yWKpsn9BWGRw==} + engines: {node: '>=14.18.0', npm: '>=8.0.0'} + hasBin: true + optionalDependencies: + fsevents: 2.3.3 + dev: true + + /run-parallel@1.2.0: + resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} + dependencies: + queue-microtask: 1.2.3 + dev: true + + /rxjs@7.5.5: + resolution: {integrity: sha512-sy+H0pQofO95VDmFLzyaw9xNJU4KTRSwQIGM6+iG3SypAtCiLDzpeG8sJrNCWn2Up9km+KhkvTdbkrdy+yzZdw==} + dependencies: + tslib: 2.6.2 + + /safe-array-concat@1.0.1: + resolution: {integrity: sha512-6XbUAseYE2KtOuGueyeobCySj9L4+66Tn6KQMOPQJrAJEowYKW/YR/MGJZl7FdydUdaFu4LYyDZjxf4/Nmo23Q==} + engines: {node: '>=0.4'} + dependencies: + call-bind: 1.0.5 + get-intrinsic: 1.2.2 + has-symbols: 1.0.3 + isarray: 2.0.5 + dev: true + + /safe-regex-test@1.0.1: + resolution: {integrity: sha512-Y5NejJTTliTyY4H7sipGqY+RX5P87i3F7c4Rcepy72nq+mNLhIsD0W4c7kEmduMDQCSqtPsXPlSTsFhh2LQv+g==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + get-intrinsic: 1.2.2 + is-regex: 1.1.4 + dev: true + + /scheduler@0.23.0: + resolution: {integrity: sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==} + dependencies: + loose-envify: 1.4.0 + dev: false + + /scrypt-js@3.0.1: + resolution: {integrity: sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==} + + /semver@6.3.1: + resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} + hasBin: true + dev: true + + /semver@7.5.4: + resolution: {integrity: sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==} + engines: {node: '>=10'} + hasBin: true + dependencies: + lru-cache: 6.0.0 + + /set-function-length@1.1.1: + resolution: {integrity: sha512-VoaqjbBJKiWtg4yRcKBQ7g7wnGnLV3M8oLvVWwOk2PdYY6PEFegR1vezXR0tw6fZGF9csVakIRjrJiy2veSBFQ==} + engines: {node: '>= 0.4'} + dependencies: + define-data-property: 1.1.1 + get-intrinsic: 1.2.2 + gopd: 1.0.1 + has-property-descriptors: 1.0.1 + dev: true + + /set-function-name@2.0.1: + resolution: {integrity: sha512-tMNCiqYVkXIZgc2Hnoy2IvC/f8ezc5koaRFkCjrpWzGpCd3qbZXPzVy9MAZzK1ch/X0jvSkojys3oqJN0qCmdA==} + engines: {node: '>= 0.4'} + dependencies: + define-data-property: 1.1.1 + functions-have-names: 1.2.3 + has-property-descriptors: 1.0.1 + dev: true + + /shebang-command@2.0.0: + resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} + engines: {node: '>=8'} + dependencies: + shebang-regex: 3.0.0 + + /shebang-regex@3.0.0: + resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} + engines: {node: '>=8'} + + /side-channel@1.0.4: + resolution: {integrity: sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==} + dependencies: + call-bind: 1.0.5 + get-intrinsic: 1.2.2 + object-inspect: 1.13.1 + dev: true + + /signal-exit@3.0.7: + resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} + + /slash@3.0.0: + resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} + engines: {node: '>=8'} + dev: true + + /slice-ansi@4.0.0: + resolution: {integrity: sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + astral-regex: 2.0.0 + is-fullwidth-code-point: 3.0.0 + + /solhint-config-mud@2.0.0-next.15: + resolution: {integrity: sha512-QvNcDFBNXcW63jqDN9oVUSn68B/UhSbaqiwn9S40jrGKpr7m7hIpJvsB/IzvcLAjryFWANVPWzc36iSNAOn2KA==} + dev: true + + /solhint-plugin-mud@2.0.0-next.15: + resolution: {integrity: sha512-3EZ1sc3x8sr/A64r2Sa1bx+l0Lec308l0gZQ3qYJTN8GpG3bjG3fr6SugBjWvyPjZm00SvfJbDcP/y+u2ewb2Q==} + dependencies: + '@solidity-parser/parser': 0.16.2 + dev: true + + /solhint@3.6.2(typescript@5.1.6): + resolution: {integrity: sha512-85EeLbmkcPwD+3JR7aEMKsVC9YrRSxd4qkXuMzrlf7+z2Eqdfm1wHWq1ffTuo5aDhoZxp2I9yF3QkxZOxOL7aQ==} + hasBin: true + dependencies: + '@solidity-parser/parser': 0.16.2 + ajv: 6.12.6 + antlr4: 4.13.1 + ast-parents: 0.0.1 + chalk: 4.1.2 + commander: 10.0.1 + cosmiconfig: 8.3.6(typescript@5.1.6) + fast-diff: 1.3.0 + glob: 8.1.0 + ignore: 5.3.0 + js-yaml: 4.1.0 + lodash: 4.17.21 + pluralize: 8.0.0 + semver: 7.5.4 + strip-ansi: 6.0.1 + table: 6.8.1 + text-table: 0.2.0 + optionalDependencies: + prettier: 2.8.8 + transitivePeerDependencies: + - typescript + dev: true + + /solidity-comments-extractor@0.0.7: + resolution: {integrity: sha512-wciNMLg/Irp8OKGrh3S2tfvZiZ0NEyILfcRCXCD4mp7SgK/i9gzLfhY2hY7VMCQJ3kH9UB9BzNdibIVMchzyYw==} + + /source-map-js@1.0.2: + resolution: {integrity: sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==} + engines: {node: '>=0.10.0'} + dev: true + + /sql.js@1.9.0: + resolution: {integrity: sha512-+QMN8NU5KJxofT+lEaSLYdhh+Pdq7ZYS6X5bSbpmD+4SKFf+qBmr+coKT07LZ+keUNh1sf3Nz9dQwD8WNI2i/Q==} + dev: false + + /stream-to-array@2.3.0: + resolution: {integrity: sha512-UsZtOYEn4tWU2RGLOXr/o/xjRBftZRlG3dEWoaHr8j4GuypJ3isitGbVyjQKAuMu+xbiop8q224TjiZWc4XTZA==} + dependencies: + any-promise: 1.3.0 + + /string-width@4.2.3: + resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} + engines: {node: '>=8'} + dependencies: + emoji-regex: 8.0.0 + is-fullwidth-code-point: 3.0.0 + strip-ansi: 6.0.1 + + /string.prototype.matchall@4.0.10: + resolution: {integrity: sha512-rGXbGmOEosIQi6Qva94HUjgPs9vKW+dkG7Y8Q5O2OYkWL6wFaTRZO8zM4mhP94uX55wgyrXzfS2aGtGzUL7EJQ==} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + get-intrinsic: 1.2.2 + has-symbols: 1.0.3 + internal-slot: 1.0.6 + regexp.prototype.flags: 1.5.1 + set-function-name: 2.0.1 + side-channel: 1.0.4 + dev: true + + /string.prototype.trim@1.2.8: + resolution: {integrity: sha512-lfjY4HcixfQXOfaqCvcBuOIapyaroTXhbkfJN3gcB1OtyupngWK4sEET9Knd0cXd28kTUqu/kHoV4HKSJdnjiQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /string.prototype.trimend@1.0.7: + resolution: {integrity: sha512-Ni79DqeB72ZFq1uH/L6zJ+DKZTkOtPIHovb3YZHQViE+HDouuU4mBrLOLDn5Dde3RF8qw5qVETEjhu9locMLvA==} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /string.prototype.trimstart@1.0.7: + resolution: {integrity: sha512-NGhtDFu3jCEm7B4Fy0DpLewdJQOZcQ0rGbwQ/+stjnrp2i+rlKeCvos9hOIeCmqwratM47OBxY7uFZzjxHXmrg==} + dependencies: + call-bind: 1.0.5 + define-properties: 1.2.1 + es-abstract: 1.22.3 + dev: true + + /strip-ansi@6.0.1: + resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} + engines: {node: '>=8'} + dependencies: + ansi-regex: 5.0.1 + + /strip-ansi@7.1.0: + resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} + engines: {node: '>=12'} + dependencies: + ansi-regex: 6.0.1 + + /strip-final-newline@3.0.0: + resolution: {integrity: sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw==} + engines: {node: '>=12'} + + /strip-json-comments@3.1.1: + resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} + engines: {node: '>=8'} + dev: true + + /superjson@1.13.3: + resolution: {integrity: sha512-mJiVjfd2vokfDxsQPOwJ/PtanO87LhpYY88ubI5dUB1Ab58Txbyje3+jpm+/83R/fevaq/107NNhtYBLuoTrFg==} + engines: {node: '>=10'} + dependencies: + copy-anything: 3.0.5 + dev: false + + /supports-color@5.5.0: + resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} + engines: {node: '>=4'} + dependencies: + has-flag: 3.0.0 + dev: true + + /supports-color@7.2.0: + resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} + engines: {node: '>=8'} + dependencies: + has-flag: 4.0.0 + + /supports-preserve-symlinks-flag@1.0.0: + resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} + engines: {node: '>= 0.4'} + dev: true + + /table@6.8.1: + resolution: {integrity: sha512-Y4X9zqrCftUhMeH2EptSSERdVKt/nEdijTOacGD/97EKjhQ/Qs8RTlEGABSJNNN8lac9kheH+af7yAkEWlgneA==} + engines: {node: '>=10.0.0'} + dependencies: + ajv: 8.12.0 + lodash.truncate: 4.4.2 + slice-ansi: 4.0.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + + /tailwind-merge@1.14.0: + resolution: {integrity: sha512-3mFKyCo/MBcgyOTlrY8T7odzZFx+w+qKSMAmdFzRvqBfLlSigU6TZnlFHK0lkMwj9Bj8OYU+9yW9lmGuS0QEnQ==} + dev: false + + /text-table@0.2.0: + resolution: {integrity: sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==} + dev: true + + /throttle-debounce@5.0.0: + resolution: {integrity: sha512-2iQTSgkkc1Zyk0MeVrt/3BvuOXYPl/R8Z0U2xxo9rjwNciaHDG3R+Lm6dh4EeUci49DanvBnuqI6jshoQQRGEg==} + engines: {node: '>=12.22'} + + /to-fast-properties@2.0.0: + resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==} + engines: {node: '>=4'} + dev: true + + /to-regex-range@5.0.1: + resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} + engines: {node: '>=8.0'} + dependencies: + is-number: 7.0.0 + + /ts-error@1.0.6: + resolution: {integrity: sha512-tLJxacIQUM82IR7JO1UUkKlYuUTmoY9HBJAmNWFzheSlDS5SPMcNIepejHJa4BpPQLAcbRhRf3GDJzyj6rbKvA==} + + /tslib@1.14.1: + resolution: {integrity: sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==} + dev: true + + /tslib@2.6.2: + resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==} + + /tsutils@3.21.0(typescript@5.1.6): + resolution: {integrity: sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==} + engines: {node: '>= 6'} + peerDependencies: + typescript: '>=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta' + dependencies: + tslib: 1.14.1 + typescript: 5.1.6 + dev: true + + /type-check@0.4.0: + resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} + engines: {node: '>= 0.8.0'} + dependencies: + prelude-ls: 1.2.1 + dev: true + + /type-fest@0.20.2: + resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==} + engines: {node: '>=10'} + dev: true + + /typed-array-buffer@1.0.0: + resolution: {integrity: sha512-Y8KTSIglk9OZEr8zywiIHG/kmQ7KWyjseXs1CbSo8vC42w7hg2HgYTxSWwP0+is7bWDc1H+Fo026CpHFwm8tkw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + get-intrinsic: 1.2.2 + is-typed-array: 1.1.12 + dev: true + + /typed-array-byte-length@1.0.0: + resolution: {integrity: sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.5 + for-each: 0.3.3 + has-proto: 1.0.1 + is-typed-array: 1.1.12 + dev: true + + /typed-array-byte-offset@1.0.0: + resolution: {integrity: sha512-RD97prjEt9EL8YgAgpOkf3O4IF9lhJFr9g0htQkm0rchFp/Vx7LW5Q8fSXXub7BXAODyUQohRMyOc3faCPd0hg==} + engines: {node: '>= 0.4'} + dependencies: + available-typed-arrays: 1.0.5 + call-bind: 1.0.5 + for-each: 0.3.3 + has-proto: 1.0.1 + is-typed-array: 1.1.12 + dev: true + + /typed-array-length@1.0.4: + resolution: {integrity: sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==} + dependencies: + call-bind: 1.0.5 + for-each: 0.3.3 + is-typed-array: 1.1.12 + dev: true + + /typescript@5.1.6: + resolution: {integrity: sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA==} + engines: {node: '>=14.17'} + hasBin: true + + /unbox-primitive@1.0.2: + resolution: {integrity: sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==} + dependencies: + call-bind: 1.0.5 + has-bigints: 1.0.2 + has-symbols: 1.0.3 + which-boxed-primitive: 1.0.2 + dev: true + + /undici-types@5.26.5: + resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} + + /update-browserslist-db@1.0.13(browserslist@4.22.2): + resolution: {integrity: sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg==} + hasBin: true + peerDependencies: + browserslist: '>= 4.21.0' + dependencies: + browserslist: 4.22.2 + escalade: 3.1.1 + picocolors: 1.0.0 + dev: true + + /uri-js@4.4.1: + resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} + dependencies: + punycode: 2.3.1 + + /use-local-storage-state@18.3.3(react-dom@18.2.0)(react@18.2.0): + resolution: {integrity: sha512-SwwW6LPbxf3q5XimJyYE2jBefpvEJTjAgBO47wCs0+ZkL/Hx8heF/0wtBJ7Df0SiSwyfNDIPHo+8Z3q569jlow==} + engines: {node: '>=12'} + peerDependencies: + react: '>=18' + react-dom: '>=18' + dependencies: + react: 18.2.0 + react-dom: 18.2.0(react@18.2.0) + dev: false + + /use-sync-external-store@1.2.0(react@18.2.0): + resolution: {integrity: sha512-eEgnFxGQ1Ife9bzYs6VLi8/4X6CObHMw9Qr9tPY43iKwsPw8xE8+EFsf/2cFZ5S3esXgpWgtSCtLNS41F+sKPA==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + dependencies: + react: 18.2.0 + dev: false + + /util@0.10.4: + resolution: {integrity: sha512-0Pm9hTQ3se5ll1XihRic3FDIku70C+iHUdT/W926rSgHV5QgXsYbKZN8MSC3tJtSkhuROzvsQjAaFENRXr+19A==} + dependencies: + inherits: 2.0.3 + + /viem@1.14.0(typescript@5.1.6)(zod@3.22.4): + resolution: {integrity: sha512-4d+4/H3lnbkSAbrpQ15i1nBA7hne06joLFy3L3m0ZpMc+g+Zr3D4nuSTyeiqbHAYs9m2P9Kjap0HlyGkehasgg==} + peerDependencies: + typescript: '>=5.0.4' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@adraffy/ens-normalize': 1.9.4 + '@noble/curves': 1.2.0 + '@noble/hashes': 1.3.2 + '@scure/bip32': 1.3.2 + '@scure/bip39': 1.2.1 + '@types/ws': 8.5.10 + abitype: 0.9.8(typescript@5.1.6)(zod@3.22.4) + isomorphic-ws: 5.0.0(ws@8.13.0) + typescript: 5.1.6 + ws: 8.13.0 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + - zod + + /vite@4.5.1: + resolution: {integrity: sha512-AXXFaAJ8yebyqzoNB9fu2pHoo/nWX+xZlaRwoeYUxEqBO+Zj4msE5G+BhGBll9lYEKv9Hfks52PAF2X7qDYXQA==} + engines: {node: ^14.18.0 || >=16.0.0} + hasBin: true + peerDependencies: + '@types/node': '>= 14' + less: '*' + lightningcss: ^1.21.0 + sass: '*' + stylus: '*' + sugarss: '*' + terser: ^5.4.0 + peerDependenciesMeta: + '@types/node': + optional: true + less: + optional: true + lightningcss: + optional: true + sass: + optional: true + stylus: + optional: true + sugarss: + optional: true + terser: + optional: true + dependencies: + esbuild: 0.18.20 + postcss: 8.4.33 + rollup: 3.29.4 + optionalDependencies: + fsevents: 2.3.3 + dev: true + + /wait-port@1.1.0: + resolution: {integrity: sha512-3e04qkoN3LxTMLakdqeWth8nih8usyg+sf1Bgdf9wwUkp05iuK1eSY/QpLvscT/+F/gA89+LpUmmgBtesbqI2Q==} + engines: {node: '>=10'} + hasBin: true + dependencies: + chalk: 4.1.2 + commander: 9.5.0 + debug: 4.3.4 + transitivePeerDependencies: + - supports-color + dev: true + + /which-boxed-primitive@1.0.2: + resolution: {integrity: sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==} + dependencies: + is-bigint: 1.0.4 + is-boolean-object: 1.1.2 + is-number-object: 1.0.7 + is-string: 1.0.7 + is-symbol: 1.0.4 + dev: true + + /which-typed-array@1.1.13: + resolution: {integrity: sha512-P5Nra0qjSncduVPEAr7xhoF5guty49ArDTwzJ/yNuPIbZppyRxFQsRCWrocxIY+CnMVG+qfbU2FmDKyvSGClow==} + engines: {node: '>= 0.4'} + dependencies: + available-typed-arrays: 1.0.5 + call-bind: 1.0.5 + for-each: 0.3.3 + gopd: 1.0.1 + has-tostringtag: 1.0.0 + dev: true + + /which@2.0.2: + resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} + engines: {node: '>= 8'} + hasBin: true + dependencies: + isexe: 2.0.0 + + /wrap-ansi@7.0.0: + resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + + /wrappy@1.0.2: + resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} + + /ws@7.4.6: + resolution: {integrity: sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==} + engines: {node: '>=8.3.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: ^5.0.2 + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + + /ws@8.13.0: + resolution: {integrity: sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA==} + engines: {node: '>=10.0.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: '>=5.0.2' + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + + /y18n@5.0.8: + resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} + engines: {node: '>=10'} + + /yallist@3.1.1: + resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==} + dev: true + + /yallist@4.0.0: + resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==} + + /yargs-parser@21.1.1: + resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} + engines: {node: '>=12'} + + /yargs@17.7.2: + resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} + engines: {node: '>=12'} + dependencies: + cliui: 8.0.1 + escalade: 3.1.1 + get-caller-file: 2.0.5 + require-directory: 2.1.1 + string-width: 4.2.3 + y18n: 5.0.8 + yargs-parser: 21.1.1 + + /yocto-queue@0.1.0: + resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} + engines: {node: '>=10'} + dev: true + + /yocto-queue@1.0.0: + resolution: {integrity: sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==} + engines: {node: '>=12.20'} + + /zod-validation-error@1.5.0(zod@3.22.4): + resolution: {integrity: sha512-/7eFkAI4qV0tcxMBB/3+d2c1P6jzzZYdYSlBuAklzMuCrJu5bzJfHS0yVAS87dRHVlhftd6RFJDIvv03JgkSbw==} + engines: {node: '>=16.0.0'} + peerDependencies: + zod: ^3.18.0 + dependencies: + zod: 3.22.4 + + /zod@3.22.4: + resolution: {integrity: sha512-iC+8Io04lddc+mVqQ9AZ7OQ2MrUKGN+oIQyq1vemgt46jwCwLfhq7/pwnBnNXXXZb8VTVLKwp9EDkx+ryxIWmg==} + + /zustand@4.4.7(@types/react@18.2.22)(react@18.2.0): + resolution: {integrity: sha512-QFJWJMdlETcI69paJwhSMJz7PPWjVP8Sjhclxmxmxv/RYI7ZOvR5BHX+ktH0we9gTWQMxcne8q1OY8xxz604gw==} + engines: {node: '>=12.7.0'} + peerDependencies: + '@types/react': '>=16.8' + immer: '>=9.0' + react: '>=16.8' + peerDependenciesMeta: + '@types/react': + optional: true + immer: + optional: true + react: + optional: true + dependencies: + '@types/react': 18.2.22 + react: 18.2.0 + use-sync-external-store: 1.2.0(react@18.2.0) + dev: false + + github.com/dapphub/ds-test/e282159d5170298eb2455a6c05280ab5a73a4ef0: + resolution: {tarball: https://codeload.github.com/dapphub/ds-test/tar.gz/e282159d5170298eb2455a6c05280ab5a73a4ef0} + name: ds-test + version: 1.0.0 + dev: true + + github.com/foundry-rs/forge-std/74cfb77e308dd188d2f58864aaf44963ae6b88b1: + resolution: {tarball: https://codeload.github.com/foundry-rs/forge-std/tar.gz/74cfb77e308dd188d2f58864aaf44963ae6b88b1} + name: forge-std + version: 1.6.0 + dev: true diff --git a/examples/multiple-accounts/pnpm-workspace.yaml b/examples/multiple-accounts/pnpm-workspace.yaml new file mode 100644 index 0000000000..924b55f42e --- /dev/null +++ b/examples/multiple-accounts/pnpm-workspace.yaml @@ -0,0 +1,2 @@ +packages: + - packages/* From 42f85dcb9e7fcd709b7b59bf9884216db9c41e31 Mon Sep 17 00:00:00 2001 From: Ori Pomerantz Date: Thu, 11 Jan 2024 11:52:01 +0200 Subject: [PATCH 11/23] docs(services): add `indexerUrl` to the indexer doc (#2009) Co-authored-by: alvarius Co-authored-by: Kevin Ingersoll --- docs/pages/services/indexer.mdx | 26 +++++++++++++++++++++++++- 1 file changed, 25 insertions(+), 1 deletion(-) diff --git a/docs/pages/services/indexer.mdx b/docs/pages/services/indexer.mdx index 6fcb437f9f..1b3831be45 100644 --- a/docs/pages/services/indexer.mdx +++ b/docs/pages/services/indexer.mdx @@ -1,3 +1,5 @@ +import { CollapseCode } from "../../components/CollapseCode"; + # Indexer The MUD Indexer is an offchain indexer for onchain applications built with MUD. @@ -180,7 +182,7 @@ The easiest way to test the indexer is to [run the template as a world](/templat #### SQLite The command to start the indexer in SQLite mode is `pnpm start:sqlite`. -To index an `anvil` instance running to the host, use: +To index an `anvil` instance running to the host you can use this command. ```sh copy docker run \ @@ -336,6 +338,28 @@ Meaning that query 0 is for everything in the `World` at address `0x6e9474e9c836 +## Using the indexer + +The source code of a MUD client has a call either to [`syncToRecs`](https://github.com/latticexyz/mud/blob/main/packages/store-sync/src/recs/syncToRecs.ts#L21-L82) or to [`syncToZustand`](https://github.com/latticexyz/mud/blob/main/packages/store-sync/src/zustand/syncToZustand.ts#L33-L74), typically in [`setupNetwork.ts`](https://github.com/latticexyz/mud/blob/main/templates/react/packages/client/src/mud/setupNetwork.ts#L74-L79). +This call initializes the synchronization between the data source (RPC or indexer) and the client's copy. + +To use the indexer, specify an `indexerUrl` parameter with the URL. + + + +```ts {7} +const { components, latestBlock$, storedBlockLogs$, waitForTransaction } = await syncToRecs({ + world, + config: mudConfig, + address: networkConfig.worldAddress as Hex, + publicClient, + startBlock: BigInt(networkConfig.initialBlockNumber), + indexerUrl: "http://127.0.0.1:3001", +}); +``` + + + ## Usage examples ### SQLite to view the tasks from the React template From aadd7dbfd64c3b7965945c7a7fb474bcea557cbc Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Thu, 11 Jan 2024 11:34:43 +0000 Subject: [PATCH 12/23] refactor(common,store): codegen clean up [N-02] (#2110) --- packages/common/src/codegen/render-solidity/common.ts | 8 ++++---- packages/common/src/codegen/utils/loadUserTypesFile.ts | 2 +- packages/store/ts/codegen/field.ts | 2 +- packages/store/ts/codegen/renderTableIndex.ts | 2 +- packages/store/ts/codegen/tableOptions.ts | 10 +++------- packages/store/ts/codegen/userType.ts | 2 +- 6 files changed, 11 insertions(+), 15 deletions(-) diff --git a/packages/common/src/codegen/render-solidity/common.ts b/packages/common/src/codegen/render-solidity/common.ts index 9ce0ec120e..c18fb2e756 100644 --- a/packages/common/src/codegen/render-solidity/common.ts +++ b/packages/common/src/codegen/render-solidity/common.ts @@ -194,11 +194,11 @@ export function renderValueTypeToBytes32(name: string, { typeUnwrap, internalTyp if (internalTypeId === "bytes32") { return innerText; - } else if (internalTypeId.match(/^bytes\d{1,2}$/)) { + } else if (/^bytes\d{1,2}$/.test(internalTypeId)) { return `bytes32(${innerText})`; - } else if (internalTypeId.match(/^uint\d{1,3}$/)) { + } else if (/^uint\d{1,3}$/.test(internalTypeId)) { return `bytes32(uint256(${innerText}))`; - } else if (internalTypeId.match(/^int\d{1,3}$/)) { + } else if (/^int\d{1,3}$/.test(internalTypeId)) { return `bytes32(uint256(int256(${innerText})))`; } else if (internalTypeId === "address") { return `bytes32(uint256(uint160(${innerText})))`; @@ -210,7 +210,7 @@ export function renderValueTypeToBytes32(name: string, { typeUnwrap, internalTyp } export function isLeftAligned(field: Pick): boolean { - return field.internalTypeId.match(/^bytes\d{1,2}$/) !== null; + return /^bytes\d{1,2}$/.test(field.internalTypeId); } export function getLeftPaddingBits(field: Pick): number { diff --git a/packages/common/src/codegen/utils/loadUserTypesFile.ts b/packages/common/src/codegen/utils/loadUserTypesFile.ts index fe69596b44..edceddf23a 100644 --- a/packages/common/src/codegen/utils/loadUserTypesFile.ts +++ b/packages/common/src/codegen/utils/loadUserTypesFile.ts @@ -35,7 +35,7 @@ export function loadAndExtractUserTypes( } } - extractedUserTypes = Object.assign(extractedUserTypes, userTypesInFile); + extractedUserTypes = { ...extractedUserTypes, ...userTypesInFile }; } return extractedUserTypes; } diff --git a/packages/store/ts/codegen/field.ts b/packages/store/ts/codegen/field.ts index 4ccc255669..4db8e38dc3 100644 --- a/packages/store/ts/codegen/field.ts +++ b/packages/store/ts/codegen/field.ts @@ -259,7 +259,7 @@ function fieldPortionData(field: RenderField) { const elementFieldData = { ...field.arrayElement, arrayElement: undefined, name, methodNameSuffix }; return { typeWithLocation: field.arrayElement.typeWithLocation, - name: "_element", + name, encoded: renderEncodeFieldSingle(elementFieldData), decoded: renderDecodeFieldSingle(elementFieldData), title: "an element", diff --git a/packages/store/ts/codegen/renderTableIndex.ts b/packages/store/ts/codegen/renderTableIndex.ts index 3ad6a51cb0..d1eeaa5a23 100644 --- a/packages/store/ts/codegen/renderTableIndex.ts +++ b/packages/store/ts/codegen/renderTableIndex.ts @@ -8,7 +8,7 @@ export function renderTableIndex(options: TableOptions[]) { ${renderList(options, ({ outputPath, tableName, renderOptions: { structName, staticResourceData } }) => { const imports = [tableName]; if (structName) imports.push(structName); - if (staticResourceData) imports.push(`${tableName}TableId`); + if (staticResourceData) imports.push(staticResourceData.tableIdName); return `import { ${imports.join(", ")} } from "./${posixPath(outputPath)}";`; })} diff --git a/packages/store/ts/codegen/tableOptions.ts b/packages/store/ts/codegen/tableOptions.ts index 9e3cff168e..95ae6ffbb4 100644 --- a/packages/store/ts/codegen/tableOptions.ts +++ b/packages/store/ts/codegen/tableOptions.ts @@ -75,18 +75,14 @@ export function getTableOptions( // With tableIdArgument: tableId is a dynamic argument for each method // Without tableIdArgument: tableId is a file-level constant generated from `staticResourceData` - const staticResourceData = (() => { - if (tableData.tableIdArgument) { - return; - } else { - return { + const staticResourceData = tableData.tableIdArgument + ? undefined + : { tableIdName: tableName + "TableId", namespace: config.namespace, name: tableData.name, offchainOnly: tableData.offchainOnly, }; - } - })(); options.push({ outputPath: path.join(tableData.directory, `${tableName}.sol`), diff --git a/packages/store/ts/codegen/userType.ts b/packages/store/ts/codegen/userType.ts index 75de3826c6..62b6a5eb92 100644 --- a/packages/store/ts/codegen/userType.ts +++ b/packages/store/ts/codegen/userType.ts @@ -148,7 +148,7 @@ export function getUserTypeInfo( isDynamic: false, typeWrap: `${typeId}.wrap`, typeUnwrap: `${typeId}.unwrap`, - internalTypeId: `${solidityUserType.internalTypeId}`, + internalTypeId: solidityUserType.internalTypeId, }, }; } From 3bebc4455ea528ef39cea3dcb26621e34d576075 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Thu, 11 Jan 2024 11:45:21 +0000 Subject: [PATCH 13/23] docs: fix codegen typos [N-07] (#2101) --- e2e/packages/contracts/src/codegen/tables/Multi.sol | 2 +- e2e/packages/contracts/src/codegen/tables/Number.sol | 2 +- e2e/packages/contracts/src/codegen/tables/NumberList.sol | 2 +- e2e/packages/contracts/src/codegen/tables/Position.sol | 2 +- e2e/packages/contracts/src/codegen/tables/StaticArray.sol | 2 +- e2e/packages/contracts/src/codegen/tables/Vector.sol | 2 +- .../packages/contracts/src/codegen/tables/CounterTable.sol | 2 +- .../minimal/packages/contracts/src/codegen/tables/Inventory.sol | 2 +- .../packages/contracts/src/codegen/tables/MessageTable.sol | 2 +- packages/cli/contracts/src/codegen/tables/Dynamics1.sol | 2 +- packages/cli/contracts/src/codegen/tables/Dynamics2.sol | 2 +- packages/cli/contracts/src/codegen/tables/Offchain.sol | 2 +- packages/cli/contracts/src/codegen/tables/Singleton.sol | 2 +- packages/cli/contracts/src/codegen/tables/Statics.sol | 2 +- packages/cli/contracts/src/codegen/tables/UserTyped.sol | 2 +- packages/store/src/codegen/tables/Hooks.sol | 2 +- packages/store/src/codegen/tables/ResourceIds.sol | 2 +- packages/store/src/codegen/tables/StoreHooks.sol | 2 +- packages/store/src/codegen/tables/Tables.sol | 2 +- packages/store/test/codegen/tables/Callbacks.sol | 2 +- packages/store/test/codegen/tables/KeyEncoding.sol | 2 +- packages/store/test/codegen/tables/Mixed.sol | 2 +- packages/store/test/codegen/tables/Vector2.sol | 2 +- packages/store/ts/codegen/renderTable.ts | 2 +- packages/store/ts/codegen/types.ts | 2 +- .../src/modules/erc20-puppet/tables/Allowances.sol | 2 +- .../src/modules/erc20-puppet/tables/ERC20Metadata.sol | 2 +- .../src/modules/erc20-puppet/tables/ERC20Registry.sol | 2 +- .../src/modules/erc20-puppet/tables/TotalSupply.sol | 2 +- .../src/modules/erc721-puppet/tables/ERC721Metadata.sol | 2 +- .../src/modules/erc721-puppet/tables/ERC721Registry.sol | 2 +- .../src/modules/erc721-puppet/tables/OperatorApproval.sol | 2 +- .../world-modules/src/modules/erc721-puppet/tables/Owners.sol | 2 +- .../src/modules/erc721-puppet/tables/TokenApproval.sol | 2 +- .../world-modules/src/modules/erc721-puppet/tables/TokenURI.sol | 2 +- .../src/modules/keysintable/tables/KeysInTable.sol | 2 +- .../src/modules/keysintable/tables/UsedKeysIndex.sol | 2 +- .../src/modules/keyswithvalue/tables/KeysWithValue.sol | 2 +- .../world-modules/src/modules/puppet/tables/PuppetRegistry.sol | 2 +- .../src/modules/std-delegations/tables/CallboundDelegations.sol | 2 +- .../modules/std-delegations/tables/SystemboundDelegations.sol | 2 +- .../src/modules/std-delegations/tables/TimeboundDelegations.sol | 2 +- packages/world-modules/src/modules/tokens/tables/Balances.sol | 2 +- .../src/modules/uniqueentity/tables/UniqueEntity.sol | 2 +- packages/world/src/codegen/tables/Balances.sol | 2 +- packages/world/src/codegen/tables/FunctionSelectors.sol | 2 +- packages/world/src/codegen/tables/FunctionSignatures.sol | 2 +- packages/world/src/codegen/tables/InstalledModules.sol | 2 +- .../world/src/codegen/tables/NamespaceDelegationControl.sol | 2 +- packages/world/src/codegen/tables/NamespaceOwner.sol | 2 +- packages/world/src/codegen/tables/ResourceAccess.sol | 2 +- packages/world/src/codegen/tables/SystemHooks.sol | 2 +- packages/world/src/codegen/tables/SystemRegistry.sol | 2 +- packages/world/src/codegen/tables/Systems.sol | 2 +- packages/world/src/codegen/tables/UserDelegationControl.sol | 2 +- packages/world/test/codegen/tables/AddressArray.sol | 2 +- packages/world/test/codegen/tables/Bool.sol | 2 +- packages/world/test/codegen/tables/TwoFields.sol | 2 +- packages/world/ts/node/render-solidity/worldgen.ts | 2 +- .../phaser/packages/contracts/src/codegen/tables/Counter.sol | 2 +- .../react-ecs/packages/contracts/src/codegen/tables/Counter.sol | 2 +- templates/react/packages/contracts/src/codegen/tables/Tasks.sol | 2 +- .../threejs/packages/contracts/src/codegen/tables/Position.sol | 2 +- .../vanilla/packages/contracts/src/codegen/tables/Counter.sol | 2 +- 64 files changed, 64 insertions(+), 64 deletions(-) diff --git a/e2e/packages/contracts/src/codegen/tables/Multi.sol b/e2e/packages/contracts/src/codegen/tables/Multi.sol index a881e1403d..81b06c8aaf 100644 --- a/e2e/packages/contracts/src/codegen/tables/Multi.sol +++ b/e2e/packages/contracts/src/codegen/tables/Multi.sol @@ -382,7 +382,7 @@ library Multi { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(int256 num, bool value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(num, value); diff --git a/e2e/packages/contracts/src/codegen/tables/Number.sol b/e2e/packages/contracts/src/codegen/tables/Number.sol index 5464204518..def46cb309 100644 --- a/e2e/packages/contracts/src/codegen/tables/Number.sol +++ b/e2e/packages/contracts/src/codegen/tables/Number.sol @@ -208,7 +208,7 @@ library Number { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/e2e/packages/contracts/src/codegen/tables/NumberList.sol b/e2e/packages/contracts/src/codegen/tables/NumberList.sol index af5daed541..a0b71641e8 100644 --- a/e2e/packages/contracts/src/codegen/tables/NumberList.sol +++ b/e2e/packages/contracts/src/codegen/tables/NumberList.sol @@ -427,7 +427,7 @@ library NumberList { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32[] memory value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/e2e/packages/contracts/src/codegen/tables/Position.sol b/e2e/packages/contracts/src/codegen/tables/Position.sol index 8c501c3d16..e34da534d7 100644 --- a/e2e/packages/contracts/src/codegen/tables/Position.sol +++ b/e2e/packages/contracts/src/codegen/tables/Position.sol @@ -232,7 +232,7 @@ library Position { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address player) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(player); diff --git a/e2e/packages/contracts/src/codegen/tables/StaticArray.sol b/e2e/packages/contracts/src/codegen/tables/StaticArray.sol index 7dd01137ff..8a2b55e49e 100644 --- a/e2e/packages/contracts/src/codegen/tables/StaticArray.sol +++ b/e2e/packages/contracts/src/codegen/tables/StaticArray.sol @@ -427,7 +427,7 @@ library StaticArray { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256[2] memory value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/e2e/packages/contracts/src/codegen/tables/Vector.sol b/e2e/packages/contracts/src/codegen/tables/Vector.sol index a084416923..9bcb795d5b 100644 --- a/e2e/packages/contracts/src/codegen/tables/Vector.sol +++ b/e2e/packages/contracts/src/codegen/tables/Vector.sol @@ -328,7 +328,7 @@ library Vector { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(int32 x, int32 y) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(x, y); diff --git a/examples/minimal/packages/contracts/src/codegen/tables/CounterTable.sol b/examples/minimal/packages/contracts/src/codegen/tables/CounterTable.sol index aaa471f58f..709f5330da 100644 --- a/examples/minimal/packages/contracts/src/codegen/tables/CounterTable.sol +++ b/examples/minimal/packages/contracts/src/codegen/tables/CounterTable.sol @@ -250,7 +250,7 @@ library CounterTable { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/examples/minimal/packages/contracts/src/codegen/tables/Inventory.sol b/examples/minimal/packages/contracts/src/codegen/tables/Inventory.sol index 0601a0afc3..2098e55a28 100644 --- a/examples/minimal/packages/contracts/src/codegen/tables/Inventory.sol +++ b/examples/minimal/packages/contracts/src/codegen/tables/Inventory.sol @@ -232,7 +232,7 @@ library Inventory { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 amount) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(amount); diff --git a/examples/minimal/packages/contracts/src/codegen/tables/MessageTable.sol b/examples/minimal/packages/contracts/src/codegen/tables/MessageTable.sol index c770e69eb2..157e866c76 100644 --- a/examples/minimal/packages/contracts/src/codegen/tables/MessageTable.sol +++ b/examples/minimal/packages/contracts/src/codegen/tables/MessageTable.sol @@ -186,7 +186,7 @@ library MessageTable { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(string memory value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/cli/contracts/src/codegen/tables/Dynamics1.sol b/packages/cli/contracts/src/codegen/tables/Dynamics1.sol index 0c656c4b95..a749ee0b67 100644 --- a/packages/cli/contracts/src/codegen/tables/Dynamics1.sol +++ b/packages/cli/contracts/src/codegen/tables/Dynamics1.sol @@ -1176,7 +1176,7 @@ library Dynamics1 { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( bytes32[1] memory staticB32, diff --git a/packages/cli/contracts/src/codegen/tables/Dynamics2.sol b/packages/cli/contracts/src/codegen/tables/Dynamics2.sol index f697ba36d3..8fa1b6c9de 100644 --- a/packages/cli/contracts/src/codegen/tables/Dynamics2.sol +++ b/packages/cli/contracts/src/codegen/tables/Dynamics2.sol @@ -762,7 +762,7 @@ library Dynamics2 { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( uint64[] memory u64, diff --git a/packages/cli/contracts/src/codegen/tables/Offchain.sol b/packages/cli/contracts/src/codegen/tables/Offchain.sol index 19d095c784..d110880813 100644 --- a/packages/cli/contracts/src/codegen/tables/Offchain.sol +++ b/packages/cli/contracts/src/codegen/tables/Offchain.sol @@ -191,7 +191,7 @@ library Offchain { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/packages/cli/contracts/src/codegen/tables/Singleton.sol b/packages/cli/contracts/src/codegen/tables/Singleton.sol index ea0ee6dcca..430df832f9 100644 --- a/packages/cli/contracts/src/codegen/tables/Singleton.sol +++ b/packages/cli/contracts/src/codegen/tables/Singleton.sol @@ -746,7 +746,7 @@ library Singleton { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( int256 v1, diff --git a/packages/cli/contracts/src/codegen/tables/Statics.sol b/packages/cli/contracts/src/codegen/tables/Statics.sol index d906ebeeda..80d3d8a95d 100644 --- a/packages/cli/contracts/src/codegen/tables/Statics.sol +++ b/packages/cli/contracts/src/codegen/tables/Statics.sol @@ -738,7 +738,7 @@ library Statics { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( uint256 v1, diff --git a/packages/cli/contracts/src/codegen/tables/UserTyped.sol b/packages/cli/contracts/src/codegen/tables/UserTyped.sol index 24a11dbb75..22c230bbdc 100644 --- a/packages/cli/contracts/src/codegen/tables/UserTyped.sol +++ b/packages/cli/contracts/src/codegen/tables/UserTyped.sol @@ -825,7 +825,7 @@ library UserTyped { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( TestTypeAddress v1, diff --git a/packages/store/src/codegen/tables/Hooks.sol b/packages/store/src/codegen/tables/Hooks.sol index ed1be30ddd..8f0f04d667 100644 --- a/packages/store/src/codegen/tables/Hooks.sol +++ b/packages/store/src/codegen/tables/Hooks.sol @@ -457,7 +457,7 @@ library Hooks { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bytes21[] memory hooks) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/store/src/codegen/tables/ResourceIds.sol b/packages/store/src/codegen/tables/ResourceIds.sol index 5b33c94693..81950d4941 100644 --- a/packages/store/src/codegen/tables/ResourceIds.sol +++ b/packages/store/src/codegen/tables/ResourceIds.sol @@ -211,7 +211,7 @@ library ResourceIds { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool exists) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(exists); diff --git a/packages/store/src/codegen/tables/StoreHooks.sol b/packages/store/src/codegen/tables/StoreHooks.sol index 40e49c7770..72f9aab99d 100644 --- a/packages/store/src/codegen/tables/StoreHooks.sol +++ b/packages/store/src/codegen/tables/StoreHooks.sol @@ -462,7 +462,7 @@ library StoreHooks { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bytes21[] memory hooks) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/store/src/codegen/tables/Tables.sol b/packages/store/src/codegen/tables/Tables.sol index e62e947299..c3cfe5fed6 100644 --- a/packages/store/src/codegen/tables/Tables.sol +++ b/packages/store/src/codegen/tables/Tables.sol @@ -778,7 +778,7 @@ library Tables { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( FieldLayout fieldLayout, diff --git a/packages/store/test/codegen/tables/Callbacks.sol b/packages/store/test/codegen/tables/Callbacks.sol index 5d3552e1b8..ec96589c0c 100644 --- a/packages/store/test/codegen/tables/Callbacks.sol +++ b/packages/store/test/codegen/tables/Callbacks.sol @@ -459,7 +459,7 @@ library Callbacks { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bytes24[] memory value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/store/test/codegen/tables/KeyEncoding.sol b/packages/store/test/codegen/tables/KeyEncoding.sol index 44ab473659..08c517307d 100644 --- a/packages/store/test/codegen/tables/KeyEncoding.sol +++ b/packages/store/test/codegen/tables/KeyEncoding.sol @@ -299,7 +299,7 @@ library KeyEncoding { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/packages/store/test/codegen/tables/Mixed.sol b/packages/store/test/codegen/tables/Mixed.sol index f772c1ea84..cffb5500b3 100644 --- a/packages/store/test/codegen/tables/Mixed.sol +++ b/packages/store/test/codegen/tables/Mixed.sol @@ -700,7 +700,7 @@ library Mixed { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( uint32 u32, diff --git a/packages/store/test/codegen/tables/Vector2.sol b/packages/store/test/codegen/tables/Vector2.sol index e39a5c9440..a22765840f 100644 --- a/packages/store/test/codegen/tables/Vector2.sol +++ b/packages/store/test/codegen/tables/Vector2.sol @@ -328,7 +328,7 @@ library Vector2 { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 x, uint32 y) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(x, y); diff --git a/packages/store/ts/codegen/renderTable.ts b/packages/store/ts/codegen/renderTable.ts index f6b06822a0..7131fbbde8 100644 --- a/packages/store/ts/codegen/renderTable.ts +++ b/packages/store/ts/codegen/renderTable.ts @@ -152,7 +152,7 @@ export function renderTable(options: RenderTableOptions) { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(${renderArguments( options.fields.map(({ name, typeWithLocation }) => `${typeWithLocation} ${name}`) diff --git a/packages/store/ts/codegen/types.ts b/packages/store/ts/codegen/types.ts index 5238965664..164b512afa 100644 --- a/packages/store/ts/codegen/types.ts +++ b/packages/store/ts/codegen/types.ts @@ -14,7 +14,7 @@ export interface RenderTableOptions { libraryName: string; /** Name of the struct to render. If undefined, struct and its methods aren't rendered. */ structName?: string; - /** Data used to statically registed the table. If undefined, all methods receive `_tableId` as an argument. */ + /** Data used to statically register the table. If undefined, all methods receive `_tableId` as an argument. */ staticResourceData?: StaticResourceData; /** Path for store package imports */ storeImportPath: string; diff --git a/packages/world-modules/src/modules/erc20-puppet/tables/Allowances.sol b/packages/world-modules/src/modules/erc20-puppet/tables/Allowances.sol index c1c3e083ed..8b649d6eb9 100644 --- a/packages/world-modules/src/modules/erc20-puppet/tables/Allowances.sol +++ b/packages/world-modules/src/modules/erc20-puppet/tables/Allowances.sol @@ -215,7 +215,7 @@ library Allowances { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Metadata.sol b/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Metadata.sol index c62353a8e4..8af6dec769 100644 --- a/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Metadata.sol +++ b/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Metadata.sol @@ -609,7 +609,7 @@ library ERC20Metadata { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( uint8 decimals, diff --git a/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Registry.sol b/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Registry.sol index e560cd0430..3d5e4a9a5f 100644 --- a/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Registry.sol +++ b/packages/world-modules/src/modules/erc20-puppet/tables/ERC20Registry.sol @@ -206,7 +206,7 @@ library ERC20Registry { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address tokenAddress) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(tokenAddress); diff --git a/packages/world-modules/src/modules/erc20-puppet/tables/TotalSupply.sol b/packages/world-modules/src/modules/erc20-puppet/tables/TotalSupply.sol index 876013c395..d8a0fdf02f 100644 --- a/packages/world-modules/src/modules/erc20-puppet/tables/TotalSupply.sol +++ b/packages/world-modules/src/modules/erc20-puppet/tables/TotalSupply.sol @@ -191,7 +191,7 @@ library TotalSupply { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 totalSupply) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(totalSupply); diff --git a/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Metadata.sol b/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Metadata.sol index 3586fa252f..a7f96ef1d4 100644 --- a/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Metadata.sol +++ b/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Metadata.sol @@ -709,7 +709,7 @@ library ERC721Metadata { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( string memory name, diff --git a/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Registry.sol b/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Registry.sol index e563db1aa8..6730fd023e 100644 --- a/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Registry.sol +++ b/packages/world-modules/src/modules/erc721-puppet/tables/ERC721Registry.sol @@ -206,7 +206,7 @@ library ERC721Registry { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address tokenAddress) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(tokenAddress); diff --git a/packages/world-modules/src/modules/erc721-puppet/tables/OperatorApproval.sol b/packages/world-modules/src/modules/erc721-puppet/tables/OperatorApproval.sol index 0cf1a0d775..49fa556e1d 100644 --- a/packages/world-modules/src/modules/erc721-puppet/tables/OperatorApproval.sol +++ b/packages/world-modules/src/modules/erc721-puppet/tables/OperatorApproval.sol @@ -215,7 +215,7 @@ library OperatorApproval { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool approved) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(approved); diff --git a/packages/world-modules/src/modules/erc721-puppet/tables/Owners.sol b/packages/world-modules/src/modules/erc721-puppet/tables/Owners.sol index 139d8e092a..2dd1521c68 100644 --- a/packages/world-modules/src/modules/erc721-puppet/tables/Owners.sol +++ b/packages/world-modules/src/modules/erc721-puppet/tables/Owners.sol @@ -203,7 +203,7 @@ library Owners { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address owner) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(owner); diff --git a/packages/world-modules/src/modules/erc721-puppet/tables/TokenApproval.sol b/packages/world-modules/src/modules/erc721-puppet/tables/TokenApproval.sol index 908ade8d32..ff748e11b7 100644 --- a/packages/world-modules/src/modules/erc721-puppet/tables/TokenApproval.sol +++ b/packages/world-modules/src/modules/erc721-puppet/tables/TokenApproval.sol @@ -203,7 +203,7 @@ library TokenApproval { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address account) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(account); diff --git a/packages/world-modules/src/modules/erc721-puppet/tables/TokenURI.sol b/packages/world-modules/src/modules/erc721-puppet/tables/TokenURI.sol index 15ef83a978..dce8311f33 100644 --- a/packages/world-modules/src/modules/erc721-puppet/tables/TokenURI.sol +++ b/packages/world-modules/src/modules/erc721-puppet/tables/TokenURI.sol @@ -458,7 +458,7 @@ library TokenURI { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(string memory tokenURI) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/world-modules/src/modules/keysintable/tables/KeysInTable.sol b/packages/world-modules/src/modules/keysintable/tables/KeysInTable.sol index 211c46d853..08e0443c72 100644 --- a/packages/world-modules/src/modules/keysintable/tables/KeysInTable.sol +++ b/packages/world-modules/src/modules/keysintable/tables/KeysInTable.sol @@ -1628,7 +1628,7 @@ library KeysInTable { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( bytes32[] memory keys0, diff --git a/packages/world-modules/src/modules/keysintable/tables/UsedKeysIndex.sol b/packages/world-modules/src/modules/keysintable/tables/UsedKeysIndex.sol index d6b469838e..e3b53fce16 100644 --- a/packages/world-modules/src/modules/keysintable/tables/UsedKeysIndex.sol +++ b/packages/world-modules/src/modules/keysintable/tables/UsedKeysIndex.sol @@ -412,7 +412,7 @@ library UsedKeysIndex { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool has, uint40 index) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(has, index); diff --git a/packages/world-modules/src/modules/keyswithvalue/tables/KeysWithValue.sol b/packages/world-modules/src/modules/keyswithvalue/tables/KeysWithValue.sol index c558072f90..8db28b8d01 100644 --- a/packages/world-modules/src/modules/keyswithvalue/tables/KeysWithValue.sol +++ b/packages/world-modules/src/modules/keyswithvalue/tables/KeysWithValue.sol @@ -676,7 +676,7 @@ library KeysWithValue { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bytes32[] memory keysWithValue) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/world-modules/src/modules/puppet/tables/PuppetRegistry.sol b/packages/world-modules/src/modules/puppet/tables/PuppetRegistry.sol index 054b093cbd..8e6b2dfa90 100644 --- a/packages/world-modules/src/modules/puppet/tables/PuppetRegistry.sol +++ b/packages/world-modules/src/modules/puppet/tables/PuppetRegistry.sol @@ -206,7 +206,7 @@ library PuppetRegistry { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address puppet) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(puppet); diff --git a/packages/world-modules/src/modules/std-delegations/tables/CallboundDelegations.sol b/packages/world-modules/src/modules/std-delegations/tables/CallboundDelegations.sol index 1e4b96c541..2801cba1a8 100644 --- a/packages/world-modules/src/modules/std-delegations/tables/CallboundDelegations.sol +++ b/packages/world-modules/src/modules/std-delegations/tables/CallboundDelegations.sol @@ -291,7 +291,7 @@ library CallboundDelegations { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 availableCalls) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(availableCalls); diff --git a/packages/world-modules/src/modules/std-delegations/tables/SystemboundDelegations.sol b/packages/world-modules/src/modules/std-delegations/tables/SystemboundDelegations.sol index 49fd116cc1..19d1c6f9c2 100644 --- a/packages/world-modules/src/modules/std-delegations/tables/SystemboundDelegations.sol +++ b/packages/world-modules/src/modules/std-delegations/tables/SystemboundDelegations.sol @@ -261,7 +261,7 @@ library SystemboundDelegations { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 availableCalls) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(availableCalls); diff --git a/packages/world-modules/src/modules/std-delegations/tables/TimeboundDelegations.sol b/packages/world-modules/src/modules/std-delegations/tables/TimeboundDelegations.sol index 48e4445f61..e75d018947 100644 --- a/packages/world-modules/src/modules/std-delegations/tables/TimeboundDelegations.sol +++ b/packages/world-modules/src/modules/std-delegations/tables/TimeboundDelegations.sol @@ -220,7 +220,7 @@ library TimeboundDelegations { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 maxTimestamp) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(maxTimestamp); diff --git a/packages/world-modules/src/modules/tokens/tables/Balances.sol b/packages/world-modules/src/modules/tokens/tables/Balances.sol index bad1af5c6d..6af8925cf8 100644 --- a/packages/world-modules/src/modules/tokens/tables/Balances.sol +++ b/packages/world-modules/src/modules/tokens/tables/Balances.sol @@ -203,7 +203,7 @@ library Balances { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/packages/world-modules/src/modules/uniqueentity/tables/UniqueEntity.sol b/packages/world-modules/src/modules/uniqueentity/tables/UniqueEntity.sol index fe577682d2..7350853225 100644 --- a/packages/world-modules/src/modules/uniqueentity/tables/UniqueEntity.sol +++ b/packages/world-modules/src/modules/uniqueentity/tables/UniqueEntity.sol @@ -245,7 +245,7 @@ library UniqueEntity { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/packages/world/src/codegen/tables/Balances.sol b/packages/world/src/codegen/tables/Balances.sol index 96d0fe791e..071750373f 100644 --- a/packages/world/src/codegen/tables/Balances.sol +++ b/packages/world/src/codegen/tables/Balances.sol @@ -211,7 +211,7 @@ library Balances { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint256 balance) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(balance); diff --git a/packages/world/src/codegen/tables/FunctionSelectors.sol b/packages/world/src/codegen/tables/FunctionSelectors.sol index e58c224c9a..cea95028bc 100644 --- a/packages/world/src/codegen/tables/FunctionSelectors.sol +++ b/packages/world/src/codegen/tables/FunctionSelectors.sol @@ -296,7 +296,7 @@ library FunctionSelectors { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( ResourceId systemId, diff --git a/packages/world/src/codegen/tables/FunctionSignatures.sol b/packages/world/src/codegen/tables/FunctionSignatures.sol index 45f8efd487..9f4e5b4345 100644 --- a/packages/world/src/codegen/tables/FunctionSignatures.sol +++ b/packages/world/src/codegen/tables/FunctionSignatures.sol @@ -192,7 +192,7 @@ library FunctionSignatures { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(string memory functionSignature) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/world/src/codegen/tables/InstalledModules.sol b/packages/world/src/codegen/tables/InstalledModules.sol index ff16296f5d..24a69c1386 100644 --- a/packages/world/src/codegen/tables/InstalledModules.sol +++ b/packages/world/src/codegen/tables/InstalledModules.sol @@ -220,7 +220,7 @@ library InstalledModules { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address moduleAddress) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(moduleAddress); diff --git a/packages/world/src/codegen/tables/NamespaceDelegationControl.sol b/packages/world/src/codegen/tables/NamespaceDelegationControl.sol index 5c13ef1dab..4e92b94f08 100644 --- a/packages/world/src/codegen/tables/NamespaceDelegationControl.sol +++ b/packages/world/src/codegen/tables/NamespaceDelegationControl.sol @@ -235,7 +235,7 @@ library NamespaceDelegationControl { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(ResourceId delegationControlId) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(delegationControlId); diff --git a/packages/world/src/codegen/tables/NamespaceOwner.sol b/packages/world/src/codegen/tables/NamespaceOwner.sol index 84c23e729f..2c29dff8f8 100644 --- a/packages/world/src/codegen/tables/NamespaceOwner.sol +++ b/packages/world/src/codegen/tables/NamespaceOwner.sol @@ -211,7 +211,7 @@ library NamespaceOwner { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address owner) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(owner); diff --git a/packages/world/src/codegen/tables/ResourceAccess.sol b/packages/world/src/codegen/tables/ResourceAccess.sol index c7a4064708..755fd757d6 100644 --- a/packages/world/src/codegen/tables/ResourceAccess.sol +++ b/packages/world/src/codegen/tables/ResourceAccess.sol @@ -223,7 +223,7 @@ library ResourceAccess { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool access) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(access); diff --git a/packages/world/src/codegen/tables/SystemHooks.sol b/packages/world/src/codegen/tables/SystemHooks.sol index 01d11b7bbe..fa13166bf8 100644 --- a/packages/world/src/codegen/tables/SystemHooks.sol +++ b/packages/world/src/codegen/tables/SystemHooks.sol @@ -462,7 +462,7 @@ library SystemHooks { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bytes21[] memory value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/world/src/codegen/tables/SystemRegistry.sol b/packages/world/src/codegen/tables/SystemRegistry.sol index 150d77a536..af6e6707c2 100644 --- a/packages/world/src/codegen/tables/SystemRegistry.sol +++ b/packages/world/src/codegen/tables/SystemRegistry.sol @@ -211,7 +211,7 @@ library SystemRegistry { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(ResourceId systemId) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(systemId); diff --git a/packages/world/src/codegen/tables/Systems.sol b/packages/world/src/codegen/tables/Systems.sol index 09dc0c4b3f..bc10a9a446 100644 --- a/packages/world/src/codegen/tables/Systems.sol +++ b/packages/world/src/codegen/tables/Systems.sol @@ -296,7 +296,7 @@ library Systems { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address system, bool publicAccess) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(system, publicAccess); diff --git a/packages/world/src/codegen/tables/UserDelegationControl.sol b/packages/world/src/codegen/tables/UserDelegationControl.sol index 6e075b4211..4c2de3caa6 100644 --- a/packages/world/src/codegen/tables/UserDelegationControl.sol +++ b/packages/world/src/codegen/tables/UserDelegationControl.sol @@ -253,7 +253,7 @@ library UserDelegationControl { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(ResourceId delegationControlId) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(delegationControlId); diff --git a/packages/world/test/codegen/tables/AddressArray.sol b/packages/world/test/codegen/tables/AddressArray.sol index 2fd41c8983..08aad51d8a 100644 --- a/packages/world/test/codegen/tables/AddressArray.sol +++ b/packages/world/test/codegen/tables/AddressArray.sol @@ -454,7 +454,7 @@ library AddressArray { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(address[] memory value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData; diff --git a/packages/world/test/codegen/tables/Bool.sol b/packages/world/test/codegen/tables/Bool.sol index ce8ed61ffa..83d6134ec1 100644 --- a/packages/world/test/codegen/tables/Bool.sol +++ b/packages/world/test/codegen/tables/Bool.sol @@ -191,7 +191,7 @@ library Bool { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/packages/world/test/codegen/tables/TwoFields.sol b/packages/world/test/codegen/tables/TwoFields.sol index 6bfccb3758..420573e30e 100644 --- a/packages/world/test/codegen/tables/TwoFields.sol +++ b/packages/world/test/codegen/tables/TwoFields.sol @@ -305,7 +305,7 @@ library TwoFields { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(bool value1, bool value2) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value1, value2); diff --git a/packages/world/ts/node/render-solidity/worldgen.ts b/packages/world/ts/node/render-solidity/worldgen.ts index 7c67686170..0bb8672393 100644 --- a/packages/world/ts/node/render-solidity/worldgen.ts +++ b/packages/world/ts/node/render-solidity/worldgen.ts @@ -72,5 +72,5 @@ export async function worldgen( }); // write to file const fullOutputPath = path.join(worldgenBaseDirectory, config.worldInterfaceName + ".sol"); - await formatAndWriteSolidity(output, fullOutputPath, "Generated system interface"); + await formatAndWriteSolidity(output, fullOutputPath, "Generated world interface"); } diff --git a/templates/phaser/packages/contracts/src/codegen/tables/Counter.sol b/templates/phaser/packages/contracts/src/codegen/tables/Counter.sol index 7239ee6e69..28ed85f3bb 100644 --- a/templates/phaser/packages/contracts/src/codegen/tables/Counter.sol +++ b/templates/phaser/packages/contracts/src/codegen/tables/Counter.sol @@ -196,7 +196,7 @@ library Counter { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/templates/react-ecs/packages/contracts/src/codegen/tables/Counter.sol b/templates/react-ecs/packages/contracts/src/codegen/tables/Counter.sol index 7239ee6e69..28ed85f3bb 100644 --- a/templates/react-ecs/packages/contracts/src/codegen/tables/Counter.sol +++ b/templates/react-ecs/packages/contracts/src/codegen/tables/Counter.sol @@ -196,7 +196,7 @@ library Counter { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); diff --git a/templates/react/packages/contracts/src/codegen/tables/Tasks.sol b/templates/react/packages/contracts/src/codegen/tables/Tasks.sol index ff0552ef06..ffe177c0dc 100644 --- a/templates/react/packages/contracts/src/codegen/tables/Tasks.sol +++ b/templates/react/packages/contracts/src/codegen/tables/Tasks.sol @@ -529,7 +529,7 @@ library Tasks { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode( uint256 createdAt, diff --git a/templates/threejs/packages/contracts/src/codegen/tables/Position.sol b/templates/threejs/packages/contracts/src/codegen/tables/Position.sol index 346fa619ee..156e794bcf 100644 --- a/templates/threejs/packages/contracts/src/codegen/tables/Position.sol +++ b/templates/threejs/packages/contracts/src/codegen/tables/Position.sol @@ -375,7 +375,7 @@ library Position { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(int32 x, int32 y, int32 z) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(x, y, z); diff --git a/templates/vanilla/packages/contracts/src/codegen/tables/Counter.sol b/templates/vanilla/packages/contracts/src/codegen/tables/Counter.sol index 7239ee6e69..28ed85f3bb 100644 --- a/templates/vanilla/packages/contracts/src/codegen/tables/Counter.sol +++ b/templates/vanilla/packages/contracts/src/codegen/tables/Counter.sol @@ -196,7 +196,7 @@ library Counter { * @notice Encode all of a record's fields. * @return The static (fixed length) data, encoded into a sequence of bytes. * @return The lengths of the dynamic fields (packed into a single bytes32 value). - * @return The dyanmic (variable length) data, encoded into a sequence of bytes. + * @return The dynamic (variable length) data, encoded into a sequence of bytes. */ function encode(uint32 value) internal pure returns (bytes memory, PackedCounter, bytes memory) { bytes memory _staticData = encodeStatic(value); From f3d1ac8d9aa3152e50182021fa564a505e214ad1 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Thu, 11 Jan 2024 12:15:00 +0000 Subject: [PATCH 14/23] chore(common,store): remove unused codegen vars [N-06] (#2103) --- .../src/codegen/render-solidity/common.ts | 41 ++++++++++-------- packages/store/ts/codegen/field.ts | 19 ++++----- packages/store/ts/codegen/record.ts | 42 ++++++++----------- packages/store/ts/codegen/renderTable.ts | 2 +- packages/store/ts/codegen/userType.ts | 2 - 5 files changed, 51 insertions(+), 55 deletions(-) diff --git a/packages/common/src/codegen/render-solidity/common.ts b/packages/common/src/codegen/render-solidity/common.ts index c18fb2e756..dae876ee4a 100644 --- a/packages/common/src/codegen/render-solidity/common.ts +++ b/packages/common/src/codegen/render-solidity/common.ts @@ -36,17 +36,12 @@ export function renderCommonData({ staticResourceData?: StaticResourceData; keyTuple: RenderKeyTuple[]; }): { - _tableId: string; _typedTableId: string; - _keyArgs: string; _typedKeyArgs: string; _keyTupleDefinition: string; } { // static resource means static tableId as well, and no tableId arguments - const _tableId = staticResourceData ? "" : "_tableId"; const _typedTableId = staticResourceData ? "" : "ResourceId _tableId"; - - const _keyArgs = renderArguments(keyTuple.map(({ name }) => name)); const _typedKeyArgs = renderArguments(keyTuple.map(({ name, typeWithLocation }) => `${typeWithLocation} ${name}`)); const _keyTupleDefinition = ` @@ -55,9 +50,7 @@ export function renderCommonData({ `; return { - _tableId, _typedTableId, - _keyArgs, _typedKeyArgs, _keyTupleDefinition, }; @@ -127,21 +120,33 @@ export function renderAbsoluteImports(imports: AbsoluteImportDatum[]): string { export function renderWithStore( storeArgument: boolean, - callback: ( - _typedStore: string | undefined, - _store: string, - _commentSuffix: string, - _untypedStore: string | undefined, - _methodPrefix: string, - _internal?: boolean - ) => string + callback: (data: { + _typedStore: string | undefined; + _store: string; + _commentSuffix: string; + _methodNamePrefix: string; + _internal?: boolean; + }) => string ): string { let result = ""; - result += callback(undefined, "StoreSwitch", "", undefined, ""); - result += callback(undefined, "StoreCore", "", undefined, "_", true); + result += callback({ _typedStore: undefined, _store: "StoreSwitch", _commentSuffix: "", _methodNamePrefix: "" }); + result += callback({ + _typedStore: undefined, + _store: "StoreCore", + _commentSuffix: "", + _methodNamePrefix: "_", + _internal: true, + }); if (storeArgument) { - result += "\n" + callback("IStore _store", "_store", " (using the specified store)", "_store", ""); + result += + "\n" + + callback({ + _typedStore: "IStore _store", + _store: "_store", + _commentSuffix: " (using the specified store)", + _methodNamePrefix: "", + }); } return result; diff --git a/packages/store/ts/codegen/field.ts b/packages/store/ts/codegen/field.ts index 4db8e38dc3..e24f5403a6 100644 --- a/packages/store/ts/codegen/field.ts +++ b/packages/store/ts/codegen/field.ts @@ -25,7 +25,7 @@ export function renderFieldMethods(options: RenderTableOptions) { result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Get ${field.name}${_commentSuffix}. */ @@ -57,7 +57,7 @@ export function renderFieldMethods(options: RenderTableOptions) { } result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => - renderWithStore(storeArgument, (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => { + renderWithStore(storeArgument, ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => { const externalArguments = renderArguments([_typedStore, _typedTableId, _typedKeyArgs, _typedFieldName]); const setFieldMethod = field.isDynamic ? "setDynamicField" : "setStaticField"; const encodeFieldSingle = renderEncodeFieldSingle(field); @@ -85,7 +85,7 @@ export function renderFieldMethods(options: RenderTableOptions) { result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Get the length of ${field.name}${_commentSuffix}. */ @@ -107,7 +107,7 @@ export function renderFieldMethods(options: RenderTableOptions) { result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Get an item of ${field.name}${_commentSuffix}. * @dev Reverts with Store_IndexOutOfBounds if \`_index\` is out of bounds for the array. @@ -138,7 +138,7 @@ export function renderFieldMethods(options: RenderTableOptions) { result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Push ${portionData.title} to ${field.name}${_commentSuffix}. */ @@ -158,7 +158,7 @@ export function renderFieldMethods(options: RenderTableOptions) { result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Pop ${portionData.title} from ${field.name}${_commentSuffix}. */ @@ -175,7 +175,7 @@ export function renderFieldMethods(options: RenderTableOptions) { ); result += renderWithFieldSuffix(options.withSuffixlessFieldMethods, field.name, (_methodNameSuffix) => - renderWithStore(storeArgument, (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => { + renderWithStore(storeArgument, ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => { const externalArguments = renderArguments([ _typedStore, _typedTableId, @@ -253,10 +253,9 @@ function renderCastStaticBytesToType(field: RenderType, staticBytes: string) { /** bytes/string are dynamic, but aren't really arrays */ function fieldPortionData(field: RenderField) { - const methodNameSuffix = ""; if (field.arrayElement) { const name = "_element"; - const elementFieldData = { ...field.arrayElement, arrayElement: undefined, name, methodNameSuffix }; + const elementFieldData = { ...field.arrayElement, arrayElement: undefined, name }; return { typeWithLocation: field.arrayElement.typeWithLocation, name, @@ -267,7 +266,7 @@ function fieldPortionData(field: RenderField) { }; } else { const name = "_slice"; - const elementFieldData = { ...field, name, methodNameSuffix }; + const elementFieldData = { ...field, name }; return { typeWithLocation: `${field.typeId} memory`, name, diff --git a/packages/store/ts/codegen/record.ts b/packages/store/ts/codegen/record.ts index c083d5f2ed..2b6d307f6e 100644 --- a/packages/store/ts/codegen/record.ts +++ b/packages/store/ts/codegen/record.ts @@ -17,7 +17,7 @@ export function renderRecordMethods(options: RenderTableOptions) { if (options.withGetters) { result += renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Get the full data${_commentSuffix}. */ @@ -39,20 +39,18 @@ export function renderRecordMethods(options: RenderTableOptions) { ); } - result += renderWithStore( - storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix, _internal) => { - const externalArguments = renderArguments([ - _typedStore, - _typedTableId, - _typedKeyArgs, - renderArguments(options.fields.map(({ name, typeWithLocation }) => `${typeWithLocation} ${name}`)), - ]); + result += renderWithStore(storeArgument, ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _internal }) => { + const externalArguments = renderArguments([ + _typedStore, + _typedTableId, + _typedKeyArgs, + renderArguments(options.fields.map(({ name, typeWithLocation }) => `${typeWithLocation} ${name}`)), + ]); - const internalArguments = - "_tableId, _keyTuple, _staticData, _encodedLengths, _dynamicData" + (_internal ? ", _fieldLayout" : ""); + const internalArguments = + "_tableId, _keyTuple, _staticData, _encodedLengths, _dynamicData" + (_internal ? ", _fieldLayout" : ""); - return ` + return ` /** * @notice Set the full data using individual values${_commentSuffix}. */ @@ -64,13 +62,12 @@ export function renderRecordMethods(options: RenderTableOptions) { ${_store}.setRecord(${internalArguments}); } `; - } - ); + }); if (structName !== undefined) { result += renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix, _internal) => { + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _internal }) => { const externalArguments = renderArguments([ _typedStore, _typedTableId, @@ -137,13 +134,11 @@ export function renderDeleteRecordMethods(options: RenderTableOptions) { const { storeArgument } = options; const { _typedTableId, _typedKeyArgs, _keyTupleDefinition } = renderCommonData(options); - return renderWithStore( - storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix, _internal) => { - const externalArguments = renderArguments([_typedStore, _typedTableId, _typedKeyArgs]); - const internalArguments = "_tableId, _keyTuple" + (_internal ? ", _fieldLayout" : ""); + return renderWithStore(storeArgument, ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _internal }) => { + const externalArguments = renderArguments([_typedStore, _typedTableId, _typedKeyArgs]); + const internalArguments = "_tableId, _keyTuple" + (_internal ? ", _fieldLayout" : ""); - return ` + return ` /** * @notice Delete all data for given keys${_commentSuffix}. */ @@ -152,8 +147,7 @@ export function renderDeleteRecordMethods(options: RenderTableOptions) { ${_store}.deleteRecord(${internalArguments}); } `; - } - ); + }); } // Renders the `decode` function that parses a bytes blob into the table data diff --git a/packages/store/ts/codegen/renderTable.ts b/packages/store/ts/codegen/renderTable.ts index 7131fbbde8..a0c03bade7 100644 --- a/packages/store/ts/codegen/renderTable.ts +++ b/packages/store/ts/codegen/renderTable.ts @@ -126,7 +126,7 @@ export function renderTable(options: RenderTableOptions) { ${renderWithStore( storeArgument, - (_typedStore, _store, _commentSuffix, _untypedStore, _methodNamePrefix) => ` + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix }) => ` /** * @notice Register the table with its config${_commentSuffix}. */ diff --git a/packages/store/ts/codegen/userType.ts b/packages/store/ts/codegen/userType.ts index 62b6a5eb92..72e613df37 100644 --- a/packages/store/ts/codegen/userType.ts +++ b/packages/store/ts/codegen/userType.ts @@ -8,8 +8,6 @@ import { parseStaticArray } from "@latticexyz/config"; import { ImportDatum, RenderType, SolidityUserDefinedType } from "@latticexyz/common/codegen"; import { StoreConfig } from "../config"; -export type UserTypeInfo = ReturnType; - /** * Resolve an abi or user type into a SchemaType and RenderType */ From 063daf80ef9aa9151903061fc7d80c170a96cb07 Mon Sep 17 00:00:00 2001 From: alvarius Date: Thu, 11 Jan 2024 15:41:53 +0100 Subject: [PATCH 15/23] feat(world): require namespace to exist before registering systems/tables in it [C-01] (#2007) Co-authored-by: Kevin Ingersoll --- .changeset/modern-brooms-rule.md | 25 +++++++ .../contracts/script/PostDeploy.s.sol | 6 +- .../cli/src/deploy/assertNamespaceOwner.ts | 42 ----------- packages/cli/src/deploy/deploy.ts | 11 ++- .../cli/src/deploy/ensureNamespaceOwner.ts | 71 +++++++++++++++++++ packages/world-modules/gas-report.json | 28 ++++---- .../src/modules/erc20-puppet/ERC20Module.sol | 13 +++- .../modules/erc721-puppet/ERC721Module.sol | 14 +++- .../keyswithvalue/KeysWithValueModule.sol | 19 +++-- .../src/modules/puppet/PuppetModule.sol | 27 +++++-- .../src/modules/puppet/constants.sol | 4 +- .../uniqueentity/UniqueEntityModule.sol | 13 +++- .../src/modules/uniqueentity/constants.sol | 3 +- .../world-modules/test/PuppetModule.t.sol | 35 +++++++++ .../test/StandardDelegationsModule.t.sol | 2 + .../world-modules/test/SystemSwitch.t.sol | 8 ++- .../test/UniqueEntityModule.t.sol | 1 + packages/world/gas-report.json | 10 +-- packages/world/src/AccessControl.sol | 13 ++++ .../StoreRegistrationSystem.sol | 26 +++---- .../WorldRegistrationSystem.sol | 16 ++--- packages/world/test/BatchCall.t.sol | 5 +- packages/world/test/Utils.t.sol | 7 ++ packages/world/test/World.t.sol | 55 +++++++++++--- packages/world/test/WorldBalance.t.sol | 2 + packages/world/test/WorldDynamicUpdate.t.sol | 1 + 26 files changed, 337 insertions(+), 120 deletions(-) create mode 100644 .changeset/modern-brooms-rule.md delete mode 100644 packages/cli/src/deploy/assertNamespaceOwner.ts create mode 100644 packages/cli/src/deploy/ensureNamespaceOwner.ts diff --git a/.changeset/modern-brooms-rule.md b/.changeset/modern-brooms-rule.md new file mode 100644 index 0000000000..ecc2abe9f9 --- /dev/null +++ b/.changeset/modern-brooms-rule.md @@ -0,0 +1,25 @@ +--- +"@latticexyz/cli": patch +"@latticexyz/world-modules": patch +"@latticexyz/world": major +--- + +Previously `registerSystem` and `registerTable` had a side effect of registering namespaces if the system or table's namespace didn't exist yet. +This caused a possible frontrunning issue, where an attacker could detect a `registerSystem`/`registerTable` transaction in the mempool, +insert a `registerNamespace` transaction before it, grant themselves access to the namespace, transfer ownership of the namespace to the victim, +so that the `registerSystem`/`registerTable` transactions still went through successfully. +To mitigate this issue, the side effect of registering a namespace in `registerSystem` and `registerTable` has been removed. +Calls to these functions now expect the respective namespace to exist and the caller to own the namespace, otherwise they revert. + +Changes in consuming projects are only necessary if tables or systems are registered manually. +If only the MUD deployer is used to register tables and systems, no changes are necessary, as the MUD deployer has been updated accordingly. + +```diff ++ world.registerNamespace(namespaceId); + world.registerSystem(systemId, system, true); +``` + +```diff ++ world.registerNamespace(namespaceId); + MyTable.register(); +``` diff --git a/examples/minimal/packages/contracts/script/PostDeploy.s.sol b/examples/minimal/packages/contracts/script/PostDeploy.s.sol index 22fd9d5f19..661a15ba47 100644 --- a/examples/minimal/packages/contracts/script/PostDeploy.s.sol +++ b/examples/minimal/packages/contracts/script/PostDeploy.s.sol @@ -4,7 +4,7 @@ pragma solidity >=0.8.21; import { Script } from "forge-std/Script.sol"; import { console } from "forge-std/console.sol"; import { StoreSwitch } from "@latticexyz/store/src/StoreSwitch.sol"; -import { ResourceId, WorldResourceIdLib } from "@latticexyz/world/src/WorldResourceId.sol"; +import { ResourceId, WorldResourceIdLib, WorldResourceIdInstance } from "@latticexyz/world/src/WorldResourceId.sol"; import { RESOURCE_SYSTEM } from "@latticexyz/world/src/worldResourceTypes.sol"; import { MessageTable, MessageTableTableId } from "../src/codegen/index.sol"; @@ -12,6 +12,8 @@ import { IWorld } from "../src/codegen/world/IWorld.sol"; import { ChatNamespacedSystem } from "../src/systems/ChatNamespacedSystem.sol"; contract PostDeploy is Script { + using WorldResourceIdInstance for ResourceId; + function run(address worldAddress) external { // Specify a store so that you can use tables directly in PostDeploy StoreSwitch.setStoreAddress(worldAddress); @@ -29,8 +31,10 @@ contract PostDeploy is Script { namespace: "namespace", name: "ChatNamespaced" }); + IWorld(worldAddress).registerNamespace(systemId.getNamespaceId()); IWorld(worldAddress).registerSystem(systemId, chatNamespacedSystem, true); IWorld(worldAddress).registerFunctionSelector(systemId, "sendMessage(string)"); + // Grant this system access to MessageTable IWorld(worldAddress).grantAccess(MessageTableTableId, address(chatNamespacedSystem)); diff --git a/packages/cli/src/deploy/assertNamespaceOwner.ts b/packages/cli/src/deploy/assertNamespaceOwner.ts deleted file mode 100644 index 4ed1e15c81..0000000000 --- a/packages/cli/src/deploy/assertNamespaceOwner.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { Account, Chain, Client, Hex, Transport, getAddress } from "viem"; -import { WorldDeploy, worldTables } from "./common"; -import { hexToResource, resourceToHex } from "@latticexyz/common"; -import { getResourceIds } from "./getResourceIds"; -import { getTableValue } from "./getTableValue"; - -export async function assertNamespaceOwner({ - client, - worldDeploy, - resourceIds, -}: { - readonly client: Client; - readonly worldDeploy: WorldDeploy; - readonly resourceIds: readonly Hex[]; -}): Promise { - const desiredNamespaces = Array.from(new Set(resourceIds.map((resourceId) => hexToResource(resourceId).namespace))); - const existingResourceIds = await getResourceIds({ client, worldDeploy }); - const existingNamespaces = Array.from( - new Set(existingResourceIds.map((resourceId) => hexToResource(resourceId).namespace)) - ); - - const namespaces = desiredNamespaces.filter((namespace) => existingNamespaces.includes(namespace)); - const namespaceOwners = await Promise.all( - namespaces.map(async (namespace) => { - const { owner } = await getTableValue({ - client, - worldDeploy, - table: worldTables.world_NamespaceOwner, - key: { namespaceId: resourceToHex({ type: "namespace", namespace, name: "" }) }, - }); - return [namespace, owner]; - }) - ); - - const unauthorizedNamespaces = namespaceOwners - .filter(([, owner]) => getAddress(owner) !== getAddress(client.account.address)) - .map(([namespace]) => namespace); - - if (unauthorizedNamespaces.length) { - throw new Error(`You are attempting to deploy to namespaces you do not own: ${unauthorizedNamespaces.join(", ")}`); - } -} diff --git a/packages/cli/src/deploy/deploy.ts b/packages/cli/src/deploy/deploy.ts index b75c189ed7..0973392c9b 100644 --- a/packages/cli/src/deploy/deploy.ts +++ b/packages/cli/src/deploy/deploy.ts @@ -9,12 +9,12 @@ import { getWorldDeploy } from "./getWorldDeploy"; import { ensureFunctions } from "./ensureFunctions"; import { ensureModules } from "./ensureModules"; import { Table } from "./configToTables"; -import { assertNamespaceOwner } from "./assertNamespaceOwner"; +import { ensureNamespaceOwner } from "./ensureNamespaceOwner"; import { debug } from "./debug"; import { resourceLabel } from "./resourceLabel"; import { uniqueBy } from "@latticexyz/common/utils"; import { ensureContractsDeployed } from "./ensureContractsDeployed"; -import { coreModuleBytecode, worldFactoryBytecode, worldFactoryContracts } from "./ensureWorldFactory"; +import { worldFactoryContracts } from "./ensureWorldFactory"; type DeployOptions = { client: Client; @@ -67,12 +67,17 @@ export async function deploy({ throw new Error(`Unsupported World version: ${worldDeploy.worldVersion}`); } - await assertNamespaceOwner({ + const namespaceTxs = await ensureNamespaceOwner({ client, worldDeploy, resourceIds: [...tables.map((table) => table.tableId), ...systems.map((system) => system.systemId)], }); + debug("waiting for all namespace registration transactions to confirm"); + for (const tx of namespaceTxs) { + await waitForTransactionReceipt(client, { hash: tx }); + } + const tableTxs = await ensureTables({ client, worldDeploy, diff --git a/packages/cli/src/deploy/ensureNamespaceOwner.ts b/packages/cli/src/deploy/ensureNamespaceOwner.ts new file mode 100644 index 0000000000..65384d2de3 --- /dev/null +++ b/packages/cli/src/deploy/ensureNamespaceOwner.ts @@ -0,0 +1,71 @@ +import { Account, Chain, Client, Hex, Transport, getAddress } from "viem"; +import { WorldDeploy, worldAbi, worldTables } from "./common"; +import { hexToResource, resourceToHex, writeContract } from "@latticexyz/common"; +import { getResourceIds } from "./getResourceIds"; +import { getTableValue } from "./getTableValue"; +import { debug } from "./debug"; + +export async function ensureNamespaceOwner({ + client, + worldDeploy, + resourceIds, +}: { + readonly client: Client; + readonly worldDeploy: WorldDeploy; + readonly resourceIds: readonly Hex[]; +}): Promise { + const desiredNamespaces = Array.from(new Set(resourceIds.map((resourceId) => hexToResource(resourceId).namespace))); + const existingResourceIds = await getResourceIds({ client, worldDeploy }); + const existingNamespaces = new Set(existingResourceIds.map((resourceId) => hexToResource(resourceId).namespace)); + if (existingNamespaces.size) { + debug( + "found", + existingNamespaces.size, + "existing namespaces:", + Array.from(existingNamespaces) + .map((namespace) => (namespace === "" ? "" : namespace)) + .join(", ") + ); + } + + // Assert ownership of existing namespaces + const existingDesiredNamespaces = desiredNamespaces.filter((namespace) => existingNamespaces.has(namespace)); + const namespaceOwners = await Promise.all( + existingDesiredNamespaces.map(async (namespace) => { + const { owner } = await getTableValue({ + client, + worldDeploy, + table: worldTables.world_NamespaceOwner, + key: { namespaceId: resourceToHex({ type: "namespace", namespace, name: "" }) }, + }); + return [namespace, owner]; + }) + ); + + const unauthorizedNamespaces = namespaceOwners + .filter(([, owner]) => getAddress(owner) !== getAddress(client.account.address)) + .map(([namespace]) => namespace); + + if (unauthorizedNamespaces.length) { + throw new Error(`You are attempting to deploy to namespaces you do not own: ${unauthorizedNamespaces.join(", ")}`); + } + + // Register missing namespaces + const missingNamespaces = desiredNamespaces.filter((namespace) => !existingNamespaces.has(namespace)); + if (missingNamespaces.length > 0) { + debug("registering namespaces", Array.from(missingNamespaces).join(", ")); + } + const registrationTxs = Promise.all( + missingNamespaces.map((namespace) => + writeContract(client, { + chain: client.chain ?? null, + address: worldDeploy.address, + abi: worldAbi, + functionName: "registerNamespace", + args: [resourceToHex({ namespace, type: "namespace", name: "" })], + }) + ) + ); + + return registrationTxs; +} diff --git a/packages/world-modules/gas-report.json b/packages/world-modules/gas-report.json index 2e39630ee8..87d1a9b1d4 100644 --- a/packages/world-modules/gas-report.json +++ b/packages/world-modules/gas-report.json @@ -75,13 +75,13 @@ "file": "test/KeysInTableModule.t.sol", "test": "testInstallComposite", "name": "install keys in table module", - "gasUsed": 1413179 + "gasUsed": 1413282 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallGas", "name": "install keys in table module", - "gasUsed": 1413179 + "gasUsed": 1413282 }, { "file": "test/KeysInTableModule.t.sol", @@ -93,13 +93,13 @@ "file": "test/KeysInTableModule.t.sol", "test": "testInstallSingleton", "name": "install keys in table module", - "gasUsed": 1413179 + "gasUsed": 1413282 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookCompositeGas", "name": "install keys in table module", - "gasUsed": 1413179 + "gasUsed": 1413282 }, { "file": "test/KeysInTableModule.t.sol", @@ -117,7 +117,7 @@ "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookGas", "name": "install keys in table module", - "gasUsed": 1413179 + "gasUsed": 1413282 }, { "file": "test/KeysInTableModule.t.sol", @@ -135,7 +135,7 @@ "file": "test/KeysWithValueModule.t.sol", "test": "testGetKeysWithValueGas", "name": "install keys with value module", - "gasUsed": 653541 + "gasUsed": 668083 }, { "file": "test/KeysWithValueModule.t.sol", @@ -153,7 +153,7 @@ "file": "test/KeysWithValueModule.t.sol", "test": "testInstall", "name": "install keys with value module", - "gasUsed": 653541 + "gasUsed": 668083 }, { "file": "test/KeysWithValueModule.t.sol", @@ -165,7 +165,7 @@ "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "install keys with value module", - "gasUsed": 653541 + "gasUsed": 668083 }, { "file": "test/KeysWithValueModule.t.sol", @@ -183,7 +183,7 @@ "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "install keys with value module", - "gasUsed": 653541 + "gasUsed": 668083 }, { "file": "test/KeysWithValueModule.t.sol", @@ -267,7 +267,7 @@ "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromCallboundDelegation", "name": "register a callbound delegation", - "gasUsed": 118325 + "gasUsed": 118319 }, { "file": "test/StandardDelegationsModule.t.sol", @@ -279,7 +279,7 @@ "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromSystemDelegation", "name": "register a systembound delegation", - "gasUsed": 115881 + "gasUsed": 115875 }, { "file": "test/StandardDelegationsModule.t.sol", @@ -291,7 +291,7 @@ "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromTimeboundDelegation", "name": "register a timebound delegation", - "gasUsed": 112819 + "gasUsed": 112813 }, { "file": "test/StandardDelegationsModule.t.sol", @@ -303,7 +303,7 @@ "file": "test/UniqueEntityModule.t.sol", "test": "testInstall", "name": "install unique entity module", - "gasUsed": 681852 + "gasUsed": 694780 }, { "file": "test/UniqueEntityModule.t.sol", @@ -315,7 +315,7 @@ "file": "test/UniqueEntityModule.t.sol", "test": "testInstallRoot", "name": "installRoot unique entity module", - "gasUsed": 648249 + "gasUsed": 663774 }, { "file": "test/UniqueEntityModule.t.sol", diff --git a/packages/world-modules/src/modules/erc20-puppet/ERC20Module.sol b/packages/world-modules/src/modules/erc20-puppet/ERC20Module.sol index 5a65cd5af5..7a5459f2cb 100644 --- a/packages/world-modules/src/modules/erc20-puppet/ERC20Module.sol +++ b/packages/world-modules/src/modules/erc20-puppet/ERC20Module.sol @@ -6,6 +6,7 @@ import { ResourceId } from "@latticexyz/store/src/ResourceId.sol"; import { Module } from "@latticexyz/world/src/Module.sol"; import { WorldResourceIdLib } from "@latticexyz/world/src/WorldResourceId.sol"; import { IBaseWorld } from "@latticexyz/world/src/codegen/interfaces/IBaseWorld.sol"; +import { revertWithBytes } from "@latticexyz/world/src/revertWithBytes.sol"; import { Puppet } from "../puppet/Puppet.sol"; import { createPuppet } from "../puppet/createPuppet.sol"; @@ -53,7 +54,10 @@ contract ERC20Module is Module { // Register the ERC20 tables and system IBaseWorld world = IBaseWorld(_world()); - registrationLibrary.delegatecall(abi.encodeCall(ERC20ModuleRegistrationLibrary.register, (world, namespace))); + (bool success, bytes memory returnData) = registrationLibrary.delegatecall( + abi.encodeCall(ERC20ModuleRegistrationLibrary.register, (world, namespace)) + ); + if (!success) revertWithBytes(returnData); // Initialize the Metadata ERC20Metadata.set(_metadataTableId(namespace), metadata); @@ -68,6 +72,7 @@ contract ERC20Module is Module { // Register the ERC20 in the ERC20Registry if (!ResourceIds.getExists(ERC20_REGISTRY_TABLE_ID)) { + world.registerNamespace(MODULE_NAMESPACE_ID); ERC20Registry.register(ERC20_REGISTRY_TABLE_ID); } ERC20Registry.set(ERC20_REGISTRY_TABLE_ID, namespaceId, puppet); @@ -83,6 +88,12 @@ contract ERC20ModuleRegistrationLibrary { * Register systems and tables for a new ERC20 token in a given namespace */ function register(IBaseWorld world, bytes14 namespace) public { + // Register the namespace if it doesn't exist yet + ResourceId tokenNamespace = WorldResourceIdLib.encodeNamespace(namespace); + if (!ResourceIds.getExists(tokenNamespace)) { + world.registerNamespace(tokenNamespace); + } + // Register the tables Allowances.register(_allowancesTableId(namespace)); Balances.register(_balancesTableId(namespace)); diff --git a/packages/world-modules/src/modules/erc721-puppet/ERC721Module.sol b/packages/world-modules/src/modules/erc721-puppet/ERC721Module.sol index 6a297e05cd..b96caea949 100644 --- a/packages/world-modules/src/modules/erc721-puppet/ERC721Module.sol +++ b/packages/world-modules/src/modules/erc721-puppet/ERC721Module.sol @@ -7,6 +7,7 @@ import { Module } from "@latticexyz/world/src/Module.sol"; import { WorldResourceIdLib } from "@latticexyz/world/src/WorldResourceId.sol"; import { IBaseWorld } from "@latticexyz/world/src/codegen/interfaces/IBaseWorld.sol"; import { InstalledModules } from "@latticexyz/world/src/codegen/tables/InstalledModules.sol"; +import { revertWithBytes } from "@latticexyz/world/src/revertWithBytes.sol"; import { Puppet } from "../puppet/Puppet.sol"; import { createPuppet } from "../puppet/createPuppet.sol"; @@ -57,7 +58,10 @@ contract ERC721Module is Module { // Register the ERC721 tables and system IBaseWorld world = IBaseWorld(_world()); - registrationLibrary.delegatecall(abi.encodeCall(ERC721ModuleRegistrationLibrary.register, (world, namespace))); + (bool success, bytes memory returnData) = registrationLibrary.delegatecall( + abi.encodeCall(ERC721ModuleRegistrationLibrary.register, (world, namespace)) + ); + if (!success) revertWithBytes(returnData); // Initialize the Metadata ERC721Metadata.set(_metadataTableId(namespace), metadata); @@ -72,6 +76,7 @@ contract ERC721Module is Module { // Register the ERC721 in the ERC20Registry if (!ResourceIds.getExists(ERC721_REGISTRY_TABLE_ID)) { + world.registerNamespace(MODULE_NAMESPACE_ID); ERC721Registry.register(ERC721_REGISTRY_TABLE_ID); } ERC721Registry.set(ERC721_REGISTRY_TABLE_ID, namespaceId, puppet); @@ -87,8 +92,13 @@ contract ERC721ModuleRegistrationLibrary { * Register systems and tables for a new ERC721 token in a given namespace */ function register(IBaseWorld world, bytes14 namespace) public { - // Register the tables + // Register the namespace if it doesn't exist yet + ResourceId tokenNamespace = WorldResourceIdLib.encodeNamespace(namespace); + if (!ResourceIds.getExists(tokenNamespace)) { + world.registerNamespace(tokenNamespace); + } + // Register the tables OperatorApproval.register(_operatorApprovalTableId(namespace)); Owners.register(_ownersTableId(namespace)); TokenApproval.register(_tokenApprovalTableId(namespace)); diff --git a/packages/world-modules/src/modules/keyswithvalue/KeysWithValueModule.sol b/packages/world-modules/src/modules/keyswithvalue/KeysWithValueModule.sol index 7ca3362bba..cdfdcab433 100644 --- a/packages/world-modules/src/modules/keyswithvalue/KeysWithValueModule.sol +++ b/packages/world-modules/src/modules/keyswithvalue/KeysWithValueModule.sol @@ -2,8 +2,9 @@ pragma solidity >=0.8.21; import { StoreSwitch } from "@latticexyz/store/src/StoreSwitch.sol"; import { BEFORE_SET_RECORD, BEFORE_SPLICE_STATIC_DATA, AFTER_SPLICE_STATIC_DATA, BEFORE_SPLICE_DYNAMIC_DATA, AFTER_SPLICE_DYNAMIC_DATA, BEFORE_DELETE_RECORD } from "@latticexyz/store/src/storeHookTypes.sol"; -import { Module } from "@latticexyz/world/src/Module.sol"; +import { ResourceIds } from "@latticexyz/store/src/codegen/tables/ResourceIds.sol"; +import { Module } from "@latticexyz/world/src/Module.sol"; import { IBaseWorld } from "@latticexyz/world/src/codegen/interfaces/IBaseWorld.sol"; import { InstalledModules } from "@latticexyz/world/src/codegen/index.sol"; @@ -45,16 +46,24 @@ contract KeysWithValueModule is Module { // Extract source table id from args ResourceId sourceTableId = ResourceId.wrap(abi.decode(args, (bytes32))); - ResourceId targetTableSelector = getTargetTableId(MODULE_NAMESPACE, sourceTableId); + ResourceId targetTableId = getTargetTableId(MODULE_NAMESPACE, sourceTableId); IBaseWorld world = IBaseWorld(_world()); + // Register the target namespace if it doesn't exist yet + if (!ResourceIds.getExists(targetTableId.getNamespaceId())) { + (bool registrationSuccess, bytes memory registrationReturnData) = address(world).delegatecall( + abi.encodeCall(world.registerNamespace, (targetTableId.getNamespaceId())) + ); + if (!registrationSuccess) revertWithBytes(registrationReturnData); + } + // Register the target table (bool success, bytes memory returnData) = address(world).delegatecall( abi.encodeCall( world.registerTable, ( - targetTableSelector, + targetTableId, KeysWithValue.getFieldLayout(), KeysWithValue.getKeySchema(), KeysWithValue.getValueSchema(), @@ -64,9 +73,11 @@ contract KeysWithValueModule is Module { ) ); + if (!success) revertWithBytes(returnData); + // Grant the hook access to the target table (success, returnData) = address(world).delegatecall( - abi.encodeCall(world.grantAccess, (targetTableSelector, address(hook))) + abi.encodeCall(world.grantAccess, (targetTableId, address(hook))) ); if (!success) revertWithBytes(returnData); diff --git a/packages/world-modules/src/modules/puppet/PuppetModule.sol b/packages/world-modules/src/modules/puppet/PuppetModule.sol index ecf4fcfad4..2b5ec19d9c 100644 --- a/packages/world-modules/src/modules/puppet/PuppetModule.sol +++ b/packages/world-modules/src/modules/puppet/PuppetModule.sol @@ -2,13 +2,15 @@ pragma solidity >=0.8.21; import { IBaseWorld } from "@latticexyz/world/src/codegen/interfaces/IBaseWorld.sol"; - import { Module } from "@latticexyz/world/src/Module.sol"; import { revertWithBytes } from "@latticexyz/world/src/revertWithBytes.sol"; +import { WorldResourceIdInstance } from "@latticexyz/world/src/WorldResourceId.sol"; + +import { ResourceId } from "@latticexyz/store/src/ResourceId.sol"; import { PuppetFactorySystem } from "./PuppetFactorySystem.sol"; import { PuppetDelegationControl } from "./PuppetDelegationControl.sol"; -import { MODULE_NAME, PUPPET_DELEGATION, PUPPET_FACTORY, PUPPET_TABLE_ID } from "./constants.sol"; +import { MODULE_NAME, PUPPET_DELEGATION, PUPPET_FACTORY, PUPPET_TABLE_ID, NAMESPACE_ID } from "./constants.sol"; import { PuppetRegistry } from "./tables/PuppetRegistry.sol"; @@ -16,6 +18,8 @@ import { PuppetRegistry } from "./tables/PuppetRegistry.sol"; * This module registers tables and delegation control systems required for puppet delegations */ contract PuppetModule is Module { + using WorldResourceIdInstance for ResourceId; + PuppetDelegationControl private immutable puppetDelegationControl = new PuppetDelegationControl(); PuppetFactorySystem private immutable puppetFactorySystem = new PuppetFactorySystem(); @@ -26,11 +30,23 @@ contract PuppetModule is Module { function installRoot(bytes memory) public { IBaseWorld world = IBaseWorld(_world()); + // Register namespace + (bool success, bytes memory returnData) = address(world).delegatecall( + abi.encodeCall(world.registerNamespace, (NAMESPACE_ID)) + ); + if (!success) revertWithBytes(returnData); + // Register table PuppetRegistry.register(PUPPET_TABLE_ID); - // Register system - (bool success, bytes memory returnData) = address(world).delegatecall( + // Register puppet factory + (success, returnData) = address(world).delegatecall( + abi.encodeCall(world.registerSystem, (PUPPET_FACTORY, puppetFactorySystem, true)) + ); + if (!success) revertWithBytes(returnData); + + // Register puppet delegation control + (success, returnData) = address(world).delegatecall( abi.encodeCall(world.registerSystem, (PUPPET_DELEGATION, puppetDelegationControl, true)) ); if (!success) revertWithBytes(returnData); @@ -39,6 +55,9 @@ contract PuppetModule is Module { function install(bytes memory) public { IBaseWorld world = IBaseWorld(_world()); + // Register namespace + world.registerNamespace(NAMESPACE_ID); + // Register table PuppetRegistry.register(PUPPET_TABLE_ID); diff --git a/packages/world-modules/src/modules/puppet/constants.sol b/packages/world-modules/src/modules/puppet/constants.sol index c441dd03b9..88efb01632 100644 --- a/packages/world-modules/src/modules/puppet/constants.sol +++ b/packages/world-modules/src/modules/puppet/constants.sol @@ -3,12 +3,14 @@ pragma solidity >=0.8.21; import { ResourceId } from "@latticexyz/store/src/ResourceId.sol"; import { RESOURCE_TABLE } from "@latticexyz/store/src/storeResourceTypes.sol"; -import { RESOURCE_SYSTEM } from "@latticexyz/world/src/worldResourceTypes.sol"; +import { RESOURCE_SYSTEM, RESOURCE_NAMESPACE } from "@latticexyz/world/src/worldResourceTypes.sol"; import { ROOT_NAMESPACE } from "@latticexyz/world/src/constants.sol"; bytes16 constant MODULE_NAME = bytes16("puppet"); bytes14 constant NAMESPACE = bytes14("puppet"); +ResourceId constant NAMESPACE_ID = ResourceId.wrap(bytes32(abi.encodePacked(RESOURCE_NAMESPACE, NAMESPACE))); + ResourceId constant PUPPET_DELEGATION = ResourceId.wrap( bytes32(abi.encodePacked(RESOURCE_SYSTEM, NAMESPACE, bytes16("Delegation"))) ); diff --git a/packages/world-modules/src/modules/uniqueentity/UniqueEntityModule.sol b/packages/world-modules/src/modules/uniqueentity/UniqueEntityModule.sol index 36ddb01e8a..1ba3c0f3f4 100644 --- a/packages/world-modules/src/modules/uniqueentity/UniqueEntityModule.sol +++ b/packages/world-modules/src/modules/uniqueentity/UniqueEntityModule.sol @@ -11,7 +11,7 @@ import { revertWithBytes } from "@latticexyz/world/src/revertWithBytes.sol"; import { UniqueEntity } from "./tables/UniqueEntity.sol"; import { UniqueEntitySystem } from "./UniqueEntitySystem.sol"; -import { MODULE_NAME, TABLE_ID, SYSTEM_ID } from "./constants.sol"; +import { MODULE_NAME, TABLE_ID, SYSTEM_ID, NAMESPACE_ID } from "./constants.sol"; /** * This module creates a table that stores a nonce, and @@ -33,11 +33,17 @@ contract UniqueEntityModule is Module { IBaseWorld world = IBaseWorld(_world()); + // Register namespace + (bool success, bytes memory data) = address(world).delegatecall( + abi.encodeCall(world.registerNamespace, (NAMESPACE_ID)) + ); + if (!success) revertWithBytes(data); + // Register table UniqueEntity._register(TABLE_ID); // Register system - (bool success, bytes memory data) = address(world).delegatecall( + (success, data) = address(world).delegatecall( abi.encodeCall(world.registerSystem, (SYSTEM_ID, uniqueEntitySystem, true)) ); if (!success) revertWithBytes(data); @@ -56,6 +62,9 @@ contract UniqueEntityModule is Module { IBaseWorld world = IBaseWorld(_world()); + // Register namespace + world.registerNamespace(NAMESPACE_ID); + // Register table UniqueEntity.register(TABLE_ID); diff --git a/packages/world-modules/src/modules/uniqueentity/constants.sol b/packages/world-modules/src/modules/uniqueentity/constants.sol index 07c4691423..ebefa29b7b 100644 --- a/packages/world-modules/src/modules/uniqueentity/constants.sol +++ b/packages/world-modules/src/modules/uniqueentity/constants.sol @@ -2,12 +2,13 @@ pragma solidity >=0.8.21; import { ResourceId } from "@latticexyz/world/src/WorldResourceId.sol"; -import { RESOURCE_TABLE, RESOURCE_SYSTEM } from "@latticexyz/world/src/worldResourceTypes.sol"; +import { RESOURCE_TABLE, RESOURCE_SYSTEM, RESOURCE_NAMESPACE } from "@latticexyz/world/src/worldResourceTypes.sol"; bytes14 constant NAMESPACE = bytes14("uniqueEntity"); bytes16 constant MODULE_NAME = bytes16("uniqueEntity"); bytes16 constant SYSTEM_NAME = bytes16("system"); bytes16 constant TABLE_NAME = bytes16("table"); +ResourceId constant NAMESPACE_ID = ResourceId.wrap(bytes32(abi.encodePacked(RESOURCE_NAMESPACE, NAMESPACE))); ResourceId constant TABLE_ID = ResourceId.wrap(bytes32(abi.encodePacked(RESOURCE_TABLE, NAMESPACE, TABLE_NAME))); ResourceId constant SYSTEM_ID = ResourceId.wrap((bytes32(abi.encodePacked(RESOURCE_SYSTEM, NAMESPACE, SYSTEM_NAME)))); diff --git a/packages/world-modules/test/PuppetModule.t.sol b/packages/world-modules/test/PuppetModule.t.sol index 7e29308620..5025d49866 100644 --- a/packages/world-modules/test/PuppetModule.t.sol +++ b/packages/world-modules/test/PuppetModule.t.sol @@ -49,9 +49,25 @@ contract PuppetModuleTest is Test, GasReporter { function setUp() public { world = IBaseWorld(address(new World())); world.initialize(new CoreModule()); + } + + function _setupPuppet() internal { world.installModule(new PuppetModule(), new bytes(0)); + // Register a new namespace and system + world.registerNamespace(systemId.getNamespaceId()); + PuppetTestSystem system = new PuppetTestSystem(); + world.registerSystem(systemId, system, true); + + // Connect the puppet + puppet = PuppetTestSystem(createPuppet(world, systemId)); + } + + function _setupRootPuppet() internal { + world.installRootModule(new PuppetModule(), new bytes(0)); + // Register a new system + world.registerNamespace(systemId.getNamespaceId()); PuppetTestSystem system = new PuppetTestSystem(); world.registerSystem(systemId, system, true); @@ -60,6 +76,8 @@ contract PuppetModuleTest is Test, GasReporter { } function testEmitOnPuppet() public { + _setupPuppet(); + vm.expectEmit(true, true, true, true); emit Hello("hello world"); string memory result = puppet.echoAndEmit("hello world"); @@ -67,6 +85,23 @@ contract PuppetModuleTest is Test, GasReporter { } function testMsgSender() public { + _setupPuppet(); + + assertEq(puppet.msgSender(), address(this)); + } + + function testEmitOnRootPuppet() public { + _setupRootPuppet(); + + vm.expectEmit(true, true, true, true); + emit Hello("hello world"); + string memory result = puppet.echoAndEmit("hello world"); + assertEq(result, "hello world"); + } + + function testMsgSenderRootPuppet() public { + _setupRootPuppet(); + assertEq(puppet.msgSender(), address(this)); } } diff --git a/packages/world-modules/test/StandardDelegationsModule.t.sol b/packages/world-modules/test/StandardDelegationsModule.t.sol index fd7b4a65de..adfc7d613e 100644 --- a/packages/world-modules/test/StandardDelegationsModule.t.sol +++ b/packages/world-modules/test/StandardDelegationsModule.t.sol @@ -25,6 +25,7 @@ import { CALLBOUND_DELEGATION, SYSTEMBOUND_DELEGATION, TIMEBOUND_DELEGATION } fr import { WorldTestSystem, WorldTestSystemReturn } from "@latticexyz/world/test/World.t.sol"; contract StandardDelegationsModuleTest is Test, GasReporter { + using WorldResourceIdInstance for ResourceId; IBaseWorld private world; ResourceId private systemId = WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: "namespace", name: "testSystem" }); @@ -38,6 +39,7 @@ contract StandardDelegationsModuleTest is Test, GasReporter { // Register a new system WorldTestSystem system = new WorldTestSystem(); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); } diff --git a/packages/world-modules/test/SystemSwitch.t.sol b/packages/world-modules/test/SystemSwitch.t.sol index cce9879230..6f8cf04f76 100644 --- a/packages/world-modules/test/SystemSwitch.t.sol +++ b/packages/world-modules/test/SystemSwitch.t.sol @@ -8,7 +8,7 @@ import { System } from "@latticexyz/world/src/System.sol"; import { IBaseWorld } from "@latticexyz/world/src/codegen/interfaces/IBaseWorld.sol"; import { World } from "@latticexyz/world/src/World.sol"; import { CoreModule } from "@latticexyz/world/src/modules/core/CoreModule.sol"; -import { ResourceId, WorldResourceIdLib } from "@latticexyz/world/src/WorldResourceId.sol"; +import { ResourceId, WorldResourceIdLib, WorldResourceIdInstance } from "@latticexyz/world/src/WorldResourceId.sol"; import { RESOURCE_SYSTEM } from "@latticexyz/world/src/worldResourceTypes.sol"; import { ROOT_NAMESPACE } from "@latticexyz/world/src/constants.sol"; import { SystemSwitch } from "../src/utils/SystemSwitch.sol"; @@ -38,6 +38,8 @@ contract EchoSystem is System { address constant caller = address(4232); contract SystemSwitchTest is Test, GasReporter { + using WorldResourceIdInstance for ResourceId; + IBaseWorld world; EchoSystem systemA; @@ -68,6 +70,10 @@ contract SystemSwitchTest is Test, GasReporter { rootSystemAId = WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: ROOT_NAMESPACE, name: "systemA" }); rootSystemBId = WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: ROOT_NAMESPACE, name: "systemB" }); + // Register namespaces + world.registerNamespace(systemAId.getNamespaceId()); + world.registerNamespace(systemBId.getNamespaceId()); + // Register systems world.registerSystem(systemAId, systemA, true); world.registerSystem(systemBId, systemB, true); diff --git a/packages/world-modules/test/UniqueEntityModule.t.sol b/packages/world-modules/test/UniqueEntityModule.t.sol index 5979a67435..9808e28743 100644 --- a/packages/world-modules/test/UniqueEntityModule.t.sol +++ b/packages/world-modules/test/UniqueEntityModule.t.sol @@ -126,6 +126,7 @@ contract UniqueEntityModuleTest is Test, GasReporter { namespace: "somens", name: "echoUniqueEntity" }); + world.registerNamespace(uniqueEntityTestSystemId.getNamespaceId()); world.registerSystem(uniqueEntityTestSystemId, uniqueEntityTestSystem, true); // Execute `getUniqueEntity` from the context of a World diff --git a/packages/world/gas-report.json b/packages/world/gas-report.json index 09c68697d8..d1cb3e329a 100644 --- a/packages/world/gas-report.json +++ b/packages/world/gas-report.json @@ -69,7 +69,7 @@ "file": "test/World.t.sol", "test": "testCallFromUnlimitedDelegation", "name": "register an unlimited delegation", - "gasUsed": 47638 + "gasUsed": 47632 }, { "file": "test/World.t.sol", @@ -93,7 +93,7 @@ "file": "test/World.t.sol", "test": "testRegisterFunctionSelector", "name": "Register a function selector", - "gasUsed": 83149 + "gasUsed": 83138 }, { "file": "test/World.t.sol", @@ -105,19 +105,19 @@ "file": "test/World.t.sol", "test": "testRegisterRootFunctionSelector", "name": "Register a root function selector", - "gasUsed": 80433 + "gasUsed": 80444 }, { "file": "test/World.t.sol", "test": "testRegisterSystem", "name": "register a system", - "gasUsed": 164283 + "gasUsed": 164317 }, { "file": "test/World.t.sol", "test": "testRegisterTable", "name": "Register a new table in the namespace", - "gasUsed": 636507 + "gasUsed": 528361 }, { "file": "test/World.t.sol", diff --git a/packages/world/src/AccessControl.sol b/packages/world/src/AccessControl.sol index 988fd1b7f1..0933bec75c 100644 --- a/packages/world/src/AccessControl.sol +++ b/packages/world/src/AccessControl.sol @@ -1,6 +1,8 @@ // SPDX-License-Identifier: MIT pragma solidity >=0.8.21; +import { ResourceIds } from "@latticexyz/store/src/codegen/tables/ResourceIds.sol"; + import { ResourceId, WorldResourceIdInstance } from "./WorldResourceId.sol"; import { IWorldErrors } from "./IWorldErrors.sol"; @@ -50,4 +52,15 @@ library AccessControl { revert IWorldErrors.World_AccessDenied(resourceId.toString(), caller); } } + + /** + * @notice Check for existence of the given resource ID. + * @dev Reverts with IWorldErrors.World_ResourceNotFound if the resource does not exist. + * @param resourceId The resource ID to check existence for. + */ + function requireExistence(ResourceId resourceId) internal view { + if (!ResourceIds._getExists(resourceId)) { + revert IWorldErrors.World_ResourceNotFound(resourceId, resourceId.toString()); + } + } } diff --git a/packages/world/src/modules/core/implementations/StoreRegistrationSystem.sol b/packages/world/src/modules/core/implementations/StoreRegistrationSystem.sol index 1df3ca8a96..c380ce0cdd 100644 --- a/packages/world/src/modules/core/implementations/StoreRegistrationSystem.sol +++ b/packages/world/src/modules/core/implementations/StoreRegistrationSystem.sol @@ -5,7 +5,6 @@ import { IStoreHook, STORE_HOOK_INTERFACE_ID } from "@latticexyz/store/src/IStor import { StoreCore } from "@latticexyz/store/src/StoreCore.sol"; import { FieldLayout } from "@latticexyz/store/src/FieldLayout.sol"; import { Schema } from "@latticexyz/store/src/Schema.sol"; -import { ResourceIds } from "@latticexyz/store/src/codegen/tables/ResourceIds.sol"; import { System } from "../../../System.sol"; import { ResourceId, WorldResourceIdInstance } from "../../../WorldResourceId.sol"; @@ -14,8 +13,6 @@ import { AccessControl } from "../../../AccessControl.sol"; import { requireInterface } from "../../../requireInterface.sol"; import { revertWithBytes } from "../../../revertWithBytes.sol"; -import { Systems } from "../../../codegen/tables/Systems.sol"; - import { IWorldErrors } from "../../../IWorldErrors.sol"; import { CORE_SYSTEM_ID } from "../constants.sol"; @@ -49,23 +46,16 @@ contract StoreRegistrationSystem is System, IWorldErrors { string[] calldata fieldNames ) public virtual { // Require the name to not be the namespace's root name - if (tableId.getName() == ROOT_NAME) revert World_InvalidResourceId(tableId, tableId.toString()); - - // If the namespace doesn't exist yet, register it - ResourceId namespaceId = tableId.getNamespaceId(); - if (!ResourceIds._getExists(namespaceId)) { - // Since this is a root system, we're in the context of the World contract already, - // so we can use delegatecall to register the namespace - address coreSystemAddress = Systems._getSystem(CORE_SYSTEM_ID); - (bool success, bytes memory data) = coreSystemAddress.delegatecall( - abi.encodeCall(WorldRegistrationSystem.registerNamespace, (namespaceId)) - ); - if (!success) revertWithBytes(data); - } else { - // otherwise require caller to own the namespace - AccessControl.requireOwner(namespaceId, _msgSender()); + if (tableId.getName() == ROOT_NAME) { + revert World_InvalidResourceId(tableId, tableId.toString()); } + // Require the table's namespace to exist + AccessControl.requireExistence(tableId.getNamespaceId()); + + // Require the caller to own the table's namespace + AccessControl.requireOwner(tableId, _msgSender()); + // Register the table StoreCore.registerTable(tableId, fieldLayout, keySchema, valueSchema, keyNames, fieldNames); } diff --git a/packages/world/src/modules/core/implementations/WorldRegistrationSystem.sol b/packages/world/src/modules/core/implementations/WorldRegistrationSystem.sol index 21025a33c5..11cd866970 100644 --- a/packages/world/src/modules/core/implementations/WorldRegistrationSystem.sol +++ b/packages/world/src/modules/core/implementations/WorldRegistrationSystem.sol @@ -113,6 +113,13 @@ contract WorldRegistrationSystem is System, IWorldErrors { revert World_InvalidResourceType(RESOURCE_SYSTEM, systemId, systemId.toString()); } + // Require the system's namespace to exist + ResourceId namespaceId = systemId.getNamespaceId(); + AccessControl.requireExistence(namespaceId); + + // Require the caller to own the namespace + AccessControl.requireOwner(namespaceId, _msgSender()); + // Require the provided address to implement the WorldContextConsumer interface requireInterface(address(system), WORLD_CONTEXT_CONSUMER_INTERFACE_ID); @@ -127,15 +134,6 @@ contract WorldRegistrationSystem is System, IWorldErrors { revert World_SystemAlreadyExists(address(system)); } - // If the namespace doesn't exist yet, register it - ResourceId namespaceId = systemId.getNamespaceId(); - if (!ResourceIds._getExists(namespaceId)) { - registerNamespace(namespaceId); - } else { - // otherwise require caller to own the namespace - AccessControl.requireOwner(namespaceId, _msgSender()); - } - // Check if a system already exists at this system ID address existingSystem = Systems._getSystem(systemId); diff --git a/packages/world/test/BatchCall.t.sol b/packages/world/test/BatchCall.t.sol index 099f32ae59..e90b461f9f 100644 --- a/packages/world/test/BatchCall.t.sol +++ b/packages/world/test/BatchCall.t.sol @@ -8,7 +8,7 @@ import { StoreSwitch } from "@latticexyz/store/src/StoreSwitch.sol"; import { World } from "../src/World.sol"; import { System } from "../src/System.sol"; import { UNLIMITED_DELEGATION } from "../src/constants.sol"; -import { ResourceId, WorldResourceIdLib } from "../src/WorldResourceId.sol"; +import { ResourceId, WorldResourceIdLib, WorldResourceIdInstance } from "../src/WorldResourceId.sol"; import { RESOURCE_SYSTEM } from "../src/worldResourceTypes.sol"; import { IWorldErrors } from "../src/IWorldErrors.sol"; @@ -47,6 +47,8 @@ contract TestSystem is System { } contract BatchCallTest is Test, GasReporter { + using WorldResourceIdInstance for ResourceId; + IBaseWorld world; bytes14 namespace = "namespace"; bytes16 name = "testSystem"; @@ -56,6 +58,7 @@ contract BatchCallTest is Test, GasReporter { function setUp() public { world = IBaseWorld(address(new World())); world.initialize(new CoreModule()); + world.registerNamespace(systemId.getNamespaceId()); } function testBatchCall() public { diff --git a/packages/world/test/Utils.t.sol b/packages/world/test/Utils.t.sol index b0458c9613..b2660f5210 100644 --- a/packages/world/test/Utils.t.sol +++ b/packages/world/test/Utils.t.sol @@ -9,6 +9,8 @@ import { World } from "../src/World.sol"; import { IBaseWorld } from "../src/codegen/interfaces/IBaseWorld.sol"; import { ResourceId, WorldResourceIdLib, WorldResourceIdInstance } from "../src/WorldResourceId.sol"; import { RESOURCE_SYSTEM } from "../src/worldResourceTypes.sol"; +import { ResourceIds } from "@latticexyz/store/src/codegen/tables/ResourceIds.sol"; +import { StoreSwitch } from "@latticexyz/store/src/StoreSwitch.sol"; import { CoreModule } from "../src/modules/core/CoreModule.sol"; @@ -27,12 +29,17 @@ contract UtilsTest is Test { function setUp() public { world = IBaseWorld(address(new World())); world.initialize(new CoreModule()); + StoreSwitch.setStoreAddress(address(world)); } function _registerAndGetNamespace(bytes14 namespace) internal returns (bytes16 returnedNamespace) { UtilsTestSystem testSystem = new UtilsTestSystem(); bytes16 name = "testSystem"; ResourceId systemId = WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: namespace, name: name }); + ResourceId namespaceId = WorldResourceIdLib.encodeNamespace(namespace); + if (!ResourceIds.getExists(namespaceId)) { + world.registerNamespace(namespaceId); + } world.registerSystem(systemId, testSystem, true); bytes memory data = world.call(systemId, abi.encodeCall(UtilsTestSystem.systemNamespace, ())); diff --git a/packages/world/test/World.t.sol b/packages/world/test/World.t.sol index 678e276c8b..c6651b360e 100644 --- a/packages/world/test/World.t.sol +++ b/packages/world/test/World.t.sol @@ -308,6 +308,7 @@ contract WorldTest is Test, GasReporter { name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, false); bytes memory result = world.call(systemId, abi.encodeCall(WorldTestSystem.getStoreAddress, ())); @@ -431,6 +432,7 @@ contract WorldTest is Test, GasReporter { fieldNames[0] = "value1"; fieldNames[1] = "value2"; fieldNames[2] = "value3"; + world.registerNamespace(tableId.getNamespaceId()); startGasReport("Register a new table in the namespace"); world.registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, keyNames, fieldNames); @@ -460,7 +462,7 @@ contract WorldTest is Test, GasReporter { namespace: namespace, name: "otherTable" }); - _expectAccessDenied(address(0x01), namespace, "", RESOURCE_NAMESPACE); + _expectAccessDenied(address(0x01), namespace, "otherTable", RESOURCE_TABLE); world.registerTable(otherTableId, fieldLayout, defaultKeySchema, valueSchema, keyNames, fieldNames); // Expect the World to not be allowed to call registerTable via an external call @@ -505,15 +507,22 @@ contract WorldTest is Test, GasReporter { // Expect the system to have access to its own namespace assertTrue(ResourceAccess.get({ resourceId: namespaceId, caller: address(system) })); - ResourceId newNamespaceId = WorldResourceIdLib.encodeNamespace("newNamespace"); - // Expect the namespace to be created if it doesn't exist yet - assertEq(NamespaceOwner.get(newNamespaceId), address(0)); - world.registerSystem( - WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: "newNamespace", name: "testSystem" }), - new System(), - false + // Expect the registration to fail if the namespace does not exist yet + System newSystem = new System(); + ResourceId invalidNamespaceSystemId = WorldResourceIdLib.encode({ + typeId: RESOURCE_SYSTEM, + namespace: "newNamespace", + name: "testSystem" + }); + assertEq(NamespaceOwner.get(invalidNamespaceSystemId.getNamespaceId()), address(0)); + vm.expectRevert( + abi.encodeWithSelector( + IWorldErrors.World_ResourceNotFound.selector, + invalidNamespaceSystemId.getNamespaceId(), + invalidNamespaceSystemId.getNamespaceId().toString() + ) ); - assertEq(NamespaceOwner.get(newNamespaceId), address(this)); + world.registerSystem(invalidNamespaceSystemId, newSystem, false); // Expect an error when registering an existing system at a new system ID vm.expectRevert(abi.encodeWithSelector(IWorldErrors.World_SystemAlreadyExists.selector, address(system))); @@ -527,7 +536,6 @@ contract WorldTest is Test, GasReporter { world.registerSystem(systemId, system, true); // Expect an error when registering a system at an existing resource ID of a different type - System newSystem = new System(); ResourceId tableId = WorldResourceIdLib.encode({ typeId: RESOURCE_TABLE, namespace: "", name: "testTable" }); world.registerTable( tableId, @@ -576,7 +584,7 @@ contract WorldTest is Test, GasReporter { ) ); world.registerSystem( - WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: "someNamespace", name: "invalidSystem" }), + WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: "", name: "invalidSystem" }), System(address(world)), true ); @@ -591,6 +599,7 @@ contract WorldTest is Test, GasReporter { namespace: namespace, name: systemName }); + world.registerNamespace(systemId.getNamespaceId()); // Register a system System oldSystem = new System(); @@ -631,6 +640,10 @@ contract WorldTest is Test, GasReporter { } function testInvalidIds() public { + // Register the namespaces + world.registerNamespace(WorldResourceIdLib.encodeNamespace("namespace")); + world.registerNamespace(WorldResourceIdLib.encodeNamespace("namespace2")); + // Register a new table ResourceId tableId = WorldResourceIdLib.encode({ typeId: RESOURCE_TABLE, namespace: "namespace", name: "name" }); world.registerTable( @@ -706,6 +719,7 @@ contract WorldTest is Test, GasReporter { namespace: "testSetRecord", name: "testTable" }); + world.registerNamespace(tableId.getNamespaceId()); // Register a new table world.registerTable( tableId, @@ -741,6 +755,7 @@ contract WorldTest is Test, GasReporter { ResourceId tableId = WorldResourceIdLib.encode({ typeId: RESOURCE_TABLE, namespace: namespace, name: name }); FieldLayout fieldLayout = Bool.getFieldLayout(); Schema valueSchema = Bool.getValueSchema(); + world.registerNamespace(tableId.getNamespaceId()); // Register a new table world.registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](1)); @@ -772,6 +787,7 @@ contract WorldTest is Test, GasReporter { Schema valueSchema = AddressArray.getValueSchema(); // Register a new table + world.registerNamespace(tableId.getNamespaceId()); world.registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](1)); // Create data @@ -817,6 +833,7 @@ contract WorldTest is Test, GasReporter { Schema valueSchema = Bool.getValueSchema(); // Register a new table + world.registerNamespace(tableId.getNamespaceId()); world.registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](1)); // Write data to the table and expect it to be written @@ -855,6 +872,7 @@ contract WorldTest is Test, GasReporter { namespace: "namespace", name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, false); // Call a system function without arguments via the World @@ -933,6 +951,7 @@ contract WorldTest is Test, GasReporter { namespace: "namespace", name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); address caller = address(1); @@ -954,6 +973,7 @@ contract WorldTest is Test, GasReporter { namespace: "namespace", name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); // Register an unlimited delegation @@ -983,6 +1003,7 @@ contract WorldTest is Test, GasReporter { namespace: "namespace", name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); // Expect a revert when attempting to perform a call on behalf of an address that doesn't have a delegation @@ -1005,6 +1026,7 @@ contract WorldTest is Test, GasReporter { namespace: "namespace", name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); // Register a limited delegation @@ -1022,6 +1044,7 @@ contract WorldTest is Test, GasReporter { namespace: "namespace", name: "testSystem" }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); // Register a delegation control mock system @@ -1031,6 +1054,7 @@ contract WorldTest is Test, GasReporter { namespace: "delegation", name: "mock" }); + world.registerNamespace(delegationControlMockId.getNamespaceId()); world.registerSystem(delegationControlMockId, delegationControlMock, true); address delegator = address(1); @@ -1224,6 +1248,7 @@ contract WorldTest is Test, GasReporter { // Register a new system WorldTestSystem system = new WorldTestSystem(); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); // Expect the registration to fail if the contract does not implement the system hook interface @@ -1268,6 +1293,7 @@ contract WorldTest is Test, GasReporter { }); // Register a new system + world.registerNamespace(systemId.getNamespaceId()); WorldTestSystem system = new WorldTestSystem(); world.registerSystem(systemId, system, false); @@ -1309,6 +1335,7 @@ contract WorldTest is Test, GasReporter { name: "testTable" }); // Register a new table + world.registerNamespace(tableId.getNamespaceId()); world.registerTable( tableId, Bool.getFieldLayout(), @@ -1337,6 +1364,7 @@ contract WorldTest is Test, GasReporter { name: "testTable" }); // Register a new table + world.registerNamespace(tableId.getNamespaceId()); world.registerTable( tableId, Bool.getFieldLayout(), @@ -1382,6 +1410,7 @@ contract WorldTest is Test, GasReporter { }); // Register a new non-root system WorldTestSystem system = new WorldTestSystem(); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, false); // Call the sysyem @@ -1397,6 +1426,7 @@ contract WorldTest is Test, GasReporter { // Register a new system WorldTestSystem system = new WorldTestSystem(); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); startGasReport("Register a function selector"); @@ -1428,6 +1458,7 @@ contract WorldTest is Test, GasReporter { // Register a new system WorldTestSystem system = new WorldTestSystem(); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); string memory worldFunc = "testSelector()"; @@ -1495,6 +1526,7 @@ contract WorldTest is Test, GasReporter { bytes16 name = "testSystem"; ResourceId systemId = WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: namespace, name: name }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); world.registerRootFunctionSelector(systemId, "receiveEther()", WorldTestSystem.receiveEther.selector); @@ -1521,6 +1553,7 @@ contract WorldTest is Test, GasReporter { bytes14 namespace = "noroot"; bytes16 name = "testSystem"; ResourceId systemId = WorldResourceIdLib.encode({ typeId: RESOURCE_SYSTEM, namespace: namespace, name: name }); + world.registerNamespace(systemId.getNamespaceId()); world.registerSystem(systemId, system, true); world.registerRootFunctionSelector(systemId, "msgSender()", WorldTestSystem.msgSender.selector); diff --git a/packages/world/test/WorldBalance.t.sol b/packages/world/test/WorldBalance.t.sol index c43c09b0dc..10c44dfd45 100644 --- a/packages/world/test/WorldBalance.t.sol +++ b/packages/world/test/WorldBalance.t.sol @@ -39,6 +39,8 @@ contract WorldBalanceTest is Test, GasReporter { world.initialize(new CoreModule()); StoreSwitch.setStoreAddress(address(world)); + world.registerNamespace(namespaceId); + world.registerSystem(rootSystemId, rootSystem, true); world.registerSystem(nonRootSystemId, nonRootSystem, true); diff --git a/packages/world/test/WorldDynamicUpdate.t.sol b/packages/world/test/WorldDynamicUpdate.t.sol index f8680e6fa0..a70d6e7b5e 100644 --- a/packages/world/test/WorldDynamicUpdate.t.sol +++ b/packages/world/test/WorldDynamicUpdate.t.sol @@ -64,6 +64,7 @@ contract UpdateInDynamicFieldTest is Test, GasReporter { tableId = WorldResourceIdLib.encode({ typeId: RESOURCE_TABLE, namespace: namespace, name: name }); // Register a new table + world.registerNamespace(tableId.getNamespaceId()); world.registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](1)); // Create data From 73f8dcae3f68ba6f49a0115089dd62edf3c26a32 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Thu, 11 Jan 2024 14:43:31 +0000 Subject: [PATCH 16/23] refactor(common,store): clean up codegen naming [N-04] (#2115) --- .../src/codegen/render-solidity/common.ts | 4 +- packages/store/ts/codegen/record.ts | 64 +++++++++++-------- packages/store/ts/codegen/renderTable.ts | 4 +- .../world/ts/node/render-solidity/index.ts | 2 +- ...renderWorld.ts => renderWorldInterface.ts} | 2 +- .../world/ts/node/render-solidity/worldgen.ts | 4 +- 6 files changed, 44 insertions(+), 36 deletions(-) rename packages/world/ts/node/render-solidity/{renderWorld.ts => renderWorldInterface.ts} (94%) diff --git a/packages/common/src/codegen/render-solidity/common.ts b/packages/common/src/codegen/render-solidity/common.ts index dae876ee4a..96041373fb 100644 --- a/packages/common/src/codegen/render-solidity/common.ts +++ b/packages/common/src/codegen/render-solidity/common.ts @@ -125,7 +125,7 @@ export function renderWithStore( _store: string; _commentSuffix: string; _methodNamePrefix: string; - _internal?: boolean; + _useExplicitFieldLayout?: boolean; }) => string ): string { let result = ""; @@ -135,7 +135,7 @@ export function renderWithStore( _store: "StoreCore", _commentSuffix: "", _methodNamePrefix: "_", - _internal: true, + _useExplicitFieldLayout: true, }); if (storeArgument) { diff --git a/packages/store/ts/codegen/record.ts b/packages/store/ts/codegen/record.ts index 2b6d307f6e..33d726ffdc 100644 --- a/packages/store/ts/codegen/record.ts +++ b/packages/store/ts/codegen/record.ts @@ -39,18 +39,21 @@ export function renderRecordMethods(options: RenderTableOptions) { ); } - result += renderWithStore(storeArgument, ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _internal }) => { - const externalArguments = renderArguments([ - _typedStore, - _typedTableId, - _typedKeyArgs, - renderArguments(options.fields.map(({ name, typeWithLocation }) => `${typeWithLocation} ${name}`)), - ]); - - const internalArguments = - "_tableId, _keyTuple, _staticData, _encodedLengths, _dynamicData" + (_internal ? ", _fieldLayout" : ""); - - return ` + result += renderWithStore( + storeArgument, + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _useExplicitFieldLayout }) => { + const externalArguments = renderArguments([ + _typedStore, + _typedTableId, + _typedKeyArgs, + renderArguments(options.fields.map(({ name, typeWithLocation }) => `${typeWithLocation} ${name}`)), + ]); + + const internalArguments = + "_tableId, _keyTuple, _staticData, _encodedLengths, _dynamicData" + + (_useExplicitFieldLayout ? ", _fieldLayout" : ""); + + return ` /** * @notice Set the full data using individual values${_commentSuffix}. */ @@ -62,12 +65,13 @@ export function renderRecordMethods(options: RenderTableOptions) { ${_store}.setRecord(${internalArguments}); } `; - }); + } + ); if (structName !== undefined) { result += renderWithStore( storeArgument, - ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _internal }) => { + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _useExplicitFieldLayout }) => { const externalArguments = renderArguments([ _typedStore, _typedTableId, @@ -76,7 +80,8 @@ export function renderRecordMethods(options: RenderTableOptions) { ]); const internalArguments = - "_tableId, _keyTuple, _staticData, _encodedLengths, _dynamicData" + (_internal ? ", _fieldLayout" : ""); + "_tableId, _keyTuple, _staticData, _encodedLengths, _dynamicData" + + (_useExplicitFieldLayout ? ", _fieldLayout" : ""); return ` /** @@ -134,20 +139,23 @@ export function renderDeleteRecordMethods(options: RenderTableOptions) { const { storeArgument } = options; const { _typedTableId, _typedKeyArgs, _keyTupleDefinition } = renderCommonData(options); - return renderWithStore(storeArgument, ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _internal }) => { - const externalArguments = renderArguments([_typedStore, _typedTableId, _typedKeyArgs]); - const internalArguments = "_tableId, _keyTuple" + (_internal ? ", _fieldLayout" : ""); + return renderWithStore( + storeArgument, + ({ _typedStore, _store, _commentSuffix, _methodNamePrefix, _useExplicitFieldLayout }) => { + const externalArguments = renderArguments([_typedStore, _typedTableId, _typedKeyArgs]); + const internalArguments = "_tableId, _keyTuple" + (_useExplicitFieldLayout ? ", _fieldLayout" : ""); - return ` - /** - * @notice Delete all data for given keys${_commentSuffix}. - */ - function ${_methodNamePrefix}deleteRecord(${externalArguments}) internal { - ${_keyTupleDefinition} - ${_store}.deleteRecord(${internalArguments}); - } - `; - }); + return ` + /** + * @notice Delete all data for given keys${_commentSuffix}. + */ + function ${_methodNamePrefix}deleteRecord(${externalArguments}) internal { + ${_keyTupleDefinition} + ${_store}.deleteRecord(${internalArguments}); + } + `; + } + ); } // Renders the `decode` function that parses a bytes blob into the table data diff --git a/packages/store/ts/codegen/renderTable.ts b/packages/store/ts/codegen/renderTable.ts index a0c03bade7..52ba690c3a 100644 --- a/packages/store/ts/codegen/renderTable.ts +++ b/packages/store/ts/codegen/renderTable.ts @@ -144,7 +144,7 @@ export function renderTable(options: RenderTableOptions) { ${renderEncodeStatic(staticFields)} - ${renderEncodedLengths(dynamicFields)} + ${renderEncodeLengths(dynamicFields)} ${renderEncodeDynamic(dynamicFields)} @@ -191,7 +191,7 @@ function renderEncodeStatic(staticFields: RenderStaticField[]) { `; } -function renderEncodedLengths(dynamicFields: RenderDynamicField[]) { +function renderEncodeLengths(dynamicFields: RenderDynamicField[]) { if (dynamicFields.length === 0) return ""; return ` diff --git a/packages/world/ts/node/render-solidity/index.ts b/packages/world/ts/node/render-solidity/index.ts index 9411b7acef..67ab048d2d 100644 --- a/packages/world/ts/node/render-solidity/index.ts +++ b/packages/world/ts/node/render-solidity/index.ts @@ -1,4 +1,4 @@ export * from "./renderSystemInterface"; -export * from "./renderWorld"; +export * from "./renderWorldInterface"; export * from "./types"; export * from "./worldgen"; diff --git a/packages/world/ts/node/render-solidity/renderWorld.ts b/packages/world/ts/node/render-solidity/renderWorldInterface.ts similarity index 94% rename from packages/world/ts/node/render-solidity/renderWorld.ts rename to packages/world/ts/node/render-solidity/renderWorldInterface.ts index 42495ef9c6..e5e4eaee16 100644 --- a/packages/world/ts/node/render-solidity/renderWorld.ts +++ b/packages/world/ts/node/render-solidity/renderWorldInterface.ts @@ -7,7 +7,7 @@ import { } from "@latticexyz/common/codegen"; import type { RenderWorldOptions } from "./types"; -export function renderWorld(options: RenderWorldOptions) { +export function renderWorldInterface(options: RenderWorldOptions) { const { interfaceName, storeImportPath, worldImportPath, imports } = options; const baseImports: AbsoluteImportDatum[] = interfaceName === "IBaseWorld" diff --git a/packages/world/ts/node/render-solidity/worldgen.ts b/packages/world/ts/node/render-solidity/worldgen.ts index 0bb8672393..a65d63ed8e 100644 --- a/packages/world/ts/node/render-solidity/worldgen.ts +++ b/packages/world/ts/node/render-solidity/worldgen.ts @@ -3,7 +3,7 @@ import path from "path"; import { formatAndWriteSolidity, contractToInterface, type RelativeImportDatum } from "@latticexyz/common/codegen"; import { StoreConfig } from "@latticexyz/store"; import { renderSystemInterface } from "./renderSystemInterface"; -import { renderWorld } from "./renderWorld"; +import { renderWorldInterface } from "./renderWorldInterface"; import { resolveWorldConfig } from "../../config/resolveWorldConfig"; import { WorldConfig } from "../../config/types"; @@ -64,7 +64,7 @@ export async function worldgen( } // render IWorld - const output = renderWorld({ + const output = renderWorldInterface({ interfaceName: config.worldInterfaceName, imports: systemInterfaceImports, storeImportPath: config.storeImportPath, From 103f635ebc20ac1aecc5c526c4bcb928e860a7ed Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Thu, 11 Jan 2024 14:53:55 +0000 Subject: [PATCH 17/23] feat(store): improve FieldLayout errors [N-03] (#2114) --- .changeset/young-poets-beam.md | 12 ++++++++++++ packages/store/src/FieldLayout.sol | 17 +++++++++++------ packages/store/test/StoreCore.t.sol | 5 +++-- packages/store/ts/codegen/renderFieldLayout.ts | 4 ++-- packages/world/gas-report.json | 2 +- 5 files changed, 29 insertions(+), 11 deletions(-) create mode 100644 .changeset/young-poets-beam.md diff --git a/.changeset/young-poets-beam.md b/.changeset/young-poets-beam.md new file mode 100644 index 0000000000..0430659a3c --- /dev/null +++ b/.changeset/young-poets-beam.md @@ -0,0 +1,12 @@ +--- +"@latticexyz/store": minor +--- + +Improved error messages for invalid `FieldLayout`s + +```diff +-error FieldLayoutLib_InvalidLength(uint256 length); ++error FieldLayoutLib_TooManyFields(uint256 numFields, uint256 maxFields); ++error FieldLayoutLib_TooManyDynamicFields(uint256 numFields, uint256 maxFields); ++error FieldLayoutLib_Empty(); +``` diff --git a/packages/store/src/FieldLayout.sol b/packages/store/src/FieldLayout.sol index 7e93bc28b1..bf88c76dd2 100644 --- a/packages/store/src/FieldLayout.sol +++ b/packages/store/src/FieldLayout.sol @@ -25,7 +25,9 @@ using FieldLayoutInstance for FieldLayout global; * various constraints regarding the length and size of the fields. */ library FieldLayoutLib { - error FieldLayoutLib_InvalidLength(uint256 length); + error FieldLayoutLib_TooManyFields(uint256 numFields, uint256 maxFields); + error FieldLayoutLib_TooManyDynamicFields(uint256 numFields, uint256 maxFields); + error FieldLayoutLib_Empty(); error FieldLayoutLib_StaticLengthIsZero(); error FieldLayoutLib_StaticLengthDoesNotFitInAWord(); @@ -41,8 +43,9 @@ library FieldLayoutLib { uint256 fieldLayout; uint256 totalLength; uint256 totalFields = _staticFields.length + numDynamicFields; - if (totalFields > MAX_TOTAL_FIELDS) revert FieldLayoutLib_InvalidLength(totalFields); - if (numDynamicFields > MAX_DYNAMIC_FIELDS) revert FieldLayoutLib_InvalidLength(numDynamicFields); + if (totalFields > MAX_TOTAL_FIELDS) revert FieldLayoutLib_TooManyFields(totalFields, MAX_TOTAL_FIELDS); + if (numDynamicFields > MAX_DYNAMIC_FIELDS) + revert FieldLayoutLib_TooManyDynamicFields(numDynamicFields, MAX_DYNAMIC_FIELDS); // Compute the total static length and store the field lengths in the encoded fieldLayout for (uint256 i = 0; i < _staticFields.length; ) { @@ -149,16 +152,18 @@ library FieldLayoutInstance { */ function validate(FieldLayout fieldLayout, bool allowEmpty) internal pure { // FieldLayout must not be empty - if (!allowEmpty && fieldLayout.isEmpty()) revert FieldLayoutLib.FieldLayoutLib_InvalidLength(0); + if (!allowEmpty && fieldLayout.isEmpty()) revert FieldLayoutLib.FieldLayoutLib_Empty(); // FieldLayout must have no more than MAX_DYNAMIC_FIELDS uint256 _numDynamicFields = fieldLayout.numDynamicFields(); - if (_numDynamicFields > MAX_DYNAMIC_FIELDS) revert FieldLayoutLib.FieldLayoutLib_InvalidLength(_numDynamicFields); + if (_numDynamicFields > MAX_DYNAMIC_FIELDS) + revert FieldLayoutLib.FieldLayoutLib_TooManyDynamicFields(_numDynamicFields, MAX_DYNAMIC_FIELDS); uint256 _numStaticFields = fieldLayout.numStaticFields(); // FieldLayout must not have more than MAX_TOTAL_FIELDS in total uint256 _numTotalFields = _numStaticFields + _numDynamicFields; - if (_numTotalFields > MAX_TOTAL_FIELDS) revert FieldLayoutLib.FieldLayoutLib_InvalidLength(_numTotalFields); + if (_numTotalFields > MAX_TOTAL_FIELDS) + revert FieldLayoutLib.FieldLayoutLib_TooManyFields(_numTotalFields, MAX_TOTAL_FIELDS); // Static lengths must be valid for (uint256 i; i < _numStaticFields; ) { diff --git a/packages/store/test/StoreCore.t.sol b/packages/store/test/StoreCore.t.sol index fea7a9054e..7638fac612 100644 --- a/packages/store/test/StoreCore.t.sol +++ b/packages/store/test/StoreCore.t.sol @@ -131,8 +131,9 @@ contract StoreCoreTest is Test, StoreMock { vm.expectRevert( abi.encodeWithSelector( - FieldLayoutLib.FieldLayoutLib_InvalidLength.selector, - invalidFieldLayout.numDynamicFields() + FieldLayoutLib.FieldLayoutLib_TooManyDynamicFields.selector, + invalidFieldLayout.numDynamicFields(), + 5 ) ); IStore(this).registerTable( diff --git a/packages/store/ts/codegen/renderFieldLayout.ts b/packages/store/ts/codegen/renderFieldLayout.ts index cb112a875c..b40cc998cb 100644 --- a/packages/store/ts/codegen/renderFieldLayout.ts +++ b/packages/store/ts/codegen/renderFieldLayout.ts @@ -14,8 +14,8 @@ export function encodeFieldLayout(fields: RenderType[]) { let totalLength = 0; const totalFields = fields.length; - if (totalFields > MAX_TOTAL_FIELDS) throw new Error(`FieldLayout: invalid length ${totalFields}`); - if (numDynamicFields > MAX_DYNAMIC_FIELDS) throw new Error(`FieldLayout: invalid length ${numDynamicFields}`); + if (totalFields > MAX_TOTAL_FIELDS) throw new Error("FieldLayout: too many fields"); + if (numDynamicFields > MAX_DYNAMIC_FIELDS) throw new Error("FieldLayout: too many dynamic fields"); for (let i = 0; i < staticFields.length; i++) { const { isDynamic, staticByteLength } = fields[i]; diff --git a/packages/world/gas-report.json b/packages/world/gas-report.json index d1cb3e329a..9caba2f8e9 100644 --- a/packages/world/gas-report.json +++ b/packages/world/gas-report.json @@ -117,7 +117,7 @@ "file": "test/World.t.sol", "test": "testRegisterTable", "name": "Register a new table in the namespace", - "gasUsed": 528361 + "gasUsed": 528393 }, { "file": "test/World.t.sol", From 971807737951f1fc64ec97a07147296b0c0a269d Mon Sep 17 00:00:00 2001 From: yonada Date: Fri, 12 Jan 2024 15:17:57 +0000 Subject: [PATCH 18/23] docs: fix misleading documentation [M-06] (#2100) --- packages/store/src/PackedCounter.sol | 8 ++++---- packages/store/src/Storage.sol | 8 ++++---- packages/store/src/tightcoder/TightCoder.sol | 2 +- 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/packages/store/src/PackedCounter.sol b/packages/store/src/PackedCounter.sol index 083383ff12..bff5867e9f 100644 --- a/packages/store/src/PackedCounter.sol +++ b/packages/store/src/PackedCounter.sol @@ -48,7 +48,7 @@ library PackedCounterLib { } /** - * @notice Packs a single value into a PackedCounter. + * @notice Packs two values into a PackedCounter. * @dev Encodes the given values 'a'-'b' into the structure of a PackedCounter. * @param a The length of the first dynamic field's data. * @param b The length of the second dynamic field's data. @@ -65,7 +65,7 @@ library PackedCounterLib { } /** - * @notice Packs a single value into a PackedCounter. + * @notice Packs three values into a PackedCounter. * @dev Encodes the given values 'a'-'c' into the structure of a PackedCounter. * @param a The length of the first dynamic field's data. * @param b The length of the second dynamic field's data. @@ -84,7 +84,7 @@ library PackedCounterLib { } /** - * @notice Packs a single value into a PackedCounter. + * @notice Packs four values into a PackedCounter. * @dev Encodes the given values 'a'-'d' into the structure of a PackedCounter. * @param a The length of the first dynamic field's data. * @param b The length of the second dynamic field's data. @@ -105,7 +105,7 @@ library PackedCounterLib { } /** - * @notice Packs a single value into a PackedCounter. + * @notice Packs five values into a PackedCounter. * @dev Encodes the given values 'a'-'e' into the structure of a PackedCounter. * @param a The length of the first dynamic field's data. * @param b The length of the second dynamic field's data. diff --git a/packages/store/src/Storage.sol b/packages/store/src/Storage.sol index 65f7221f88..c0f4fa2b10 100644 --- a/packages/store/src/Storage.sol +++ b/packages/store/src/Storage.sol @@ -121,7 +121,7 @@ library Storage { /** * @notice Set multiple storage locations to zero. * @param storagePointer The starting storage location. - * @param length The number of storage locations to set to zero. + * @param length The number of storage locations to set to zero, in bytes */ function zero(uint256 storagePointer, uint256 length) internal { // Ceil division to round up to the nearest word @@ -213,9 +213,9 @@ library Storage { mstore( memoryPointer, or( - // store the middle part + // store the left part and(offsetData, mask), - // preserve the surrounding parts + // preserve the right parts and(mload(memoryPointer), not(mask)) ) ) @@ -271,7 +271,7 @@ library Storage { * @param storagePointer The base storage location. * @param length Length of the data in bytes. * @param offset Offset within the storage location. - * @return result The loaded bytes, left-aligned bytes. Bytes beyond the length are zeroed. + * @return result The loaded bytes, left-aligned bytes. Bytes beyond the length are not zeroed. */ function loadField(uint256 storagePointer, uint256 length, uint256 offset) internal view returns (bytes32 result) { if (offset >= 32) { diff --git a/packages/store/src/tightcoder/TightCoder.sol b/packages/store/src/tightcoder/TightCoder.sol index 7bd5858665..320792cbed 100644 --- a/packages/store/src/tightcoder/TightCoder.sol +++ b/packages/store/src/tightcoder/TightCoder.sol @@ -13,7 +13,7 @@ library TightCoder { * @dev Copies the array to a new bytes array, tightly packing its elements. * @param array The array to encode. * @param elementSize The size of each element in bytes. - * @param leftPaddingBits The number of bits to pad on the left for each element. + * @param leftPaddingBits The amount to shift each element to the left. * @return data A tightly packed byte array. * @notice elementSize and leftPaddingBits must be correctly provided by the caller based on the array's element type. */ From 1c7b84256cc1d259a4693b5befd72c50f129122a Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Fri, 12 Jan 2024 15:27:58 +0000 Subject: [PATCH 19/23] style(common): reuse properties from common type [L-05] (#2121) --- packages/store/ts/codegen/tightcoder/renderFunctions.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/store/ts/codegen/tightcoder/renderFunctions.ts b/packages/store/ts/codegen/tightcoder/renderFunctions.ts index 7b8b591e54..eefaa21805 100644 --- a/packages/store/ts/codegen/tightcoder/renderFunctions.ts +++ b/packages/store/ts/codegen/tightcoder/renderFunctions.ts @@ -1,6 +1,6 @@ -import { getLeftPaddingBits } from "@latticexyz/common/codegen"; +import { RenderType, getLeftPaddingBits } from "@latticexyz/common/codegen"; -export function renderTightCoderDecode(element: { internalTypeId: string; staticByteLength: number }) { +export function renderTightCoderDecode(element: Pick) { return ` /** * @notice Decodes a slice into an array of ${element.internalTypeId}. @@ -25,7 +25,7 @@ export function renderTightCoderDecode(element: { internalTypeId: string; static `; } -export function renderTightCoderEncode(element: { internalTypeId: string; staticByteLength: number }) { +export function renderTightCoderEncode(element: Pick) { return ` /** From dd724bea66ad6256e7780ac7635cea567c9d498c Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Fri, 12 Jan 2024 15:28:29 +0000 Subject: [PATCH 20/23] test: increase forge test verbosity in ci (#2119) --- .github/workflows/test.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index d61aae167f..d7f7ee3a9b 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -6,6 +6,9 @@ on: - main pull_request: +env: + FOUNDRY_VERBOSITY: 5 + jobs: test: name: Run tests From aee8020a65ca5cfebb2ca479357a535bbf07269b Mon Sep 17 00:00:00 2001 From: yonada Date: Fri, 12 Jan 2024 15:30:09 +0000 Subject: [PATCH 21/23] fix(world): check namespace exists before balance transfer [L-03] (#2095) Co-authored-by: alvarius Co-authored-by: Kevin Ingersoll --- .changeset/seven-carpets-develop.md | 5 ++++ .../implementations/BalanceTransferSystem.sol | 4 +++ packages/world/test/WorldBalance.t.sol | 26 +++++++++++++++++++ 3 files changed, 35 insertions(+) create mode 100644 .changeset/seven-carpets-develop.md diff --git a/.changeset/seven-carpets-develop.md b/.changeset/seven-carpets-develop.md new file mode 100644 index 0000000000..c1ab18bf56 --- /dev/null +++ b/.changeset/seven-carpets-develop.md @@ -0,0 +1,5 @@ +--- +"@latticexyz/world": patch +--- + +Namespace balances can no longer be transferred to non-existent namespaces. diff --git a/packages/world/src/modules/core/implementations/BalanceTransferSystem.sol b/packages/world/src/modules/core/implementations/BalanceTransferSystem.sol index 085e98db7f..53b7c88fc0 100644 --- a/packages/world/src/modules/core/implementations/BalanceTransferSystem.sol +++ b/packages/world/src/modules/core/implementations/BalanceTransferSystem.sol @@ -2,6 +2,7 @@ pragma solidity >=0.8.21; import { ResourceId, ResourceIdInstance } from "@latticexyz/store/src/ResourceId.sol"; +import { ResourceIds } from "@latticexyz/store/src/codegen/tables/ResourceIds.sol"; import { System } from "../../../System.sol"; import { revertWithBytes } from "../../../revertWithBytes.sol"; @@ -37,6 +38,9 @@ contract BalanceTransferSystem is System, IWorldErrors { revert World_InvalidResourceType(RESOURCE_NAMESPACE, toNamespaceId, toNamespaceId.toString()); } + // Require the namespace to exist + AccessControl.requireExistence(toNamespaceId); + // Require caller to have access to the namespace AccessControl.requireAccess(fromNamespaceId, _msgSender()); diff --git a/packages/world/test/WorldBalance.t.sol b/packages/world/test/WorldBalance.t.sol index 10c44dfd45..3e92b2b4bf 100644 --- a/packages/world/test/WorldBalance.t.sol +++ b/packages/world/test/WorldBalance.t.sol @@ -276,6 +276,32 @@ contract WorldBalanceTest is Test, GasReporter { assertEq(Balances.get(invalidNamespace), 0); } + function testTransferBalanceToNamespaceRevertResourceNotFound() public { + bytes14 toNamespace = "not_registered"; + ResourceId toNamespaceId = WorldResourceIdLib.encodeNamespace(toNamespace); + + uint256 value = 1 ether; + + // Expect the root namespace to have no balance + assertEq(Balances.get(ROOT_NAMESPACE_ID), 0); + + // Send balance to root namespace + vm.deal(caller, value); + vm.prank(caller); + (bool success, bytes memory data) = address(world).call{ value: value }(""); + assertTrue(success); + assertEq(data.length, 0); + + // Expect the root namespace to have the value as balance + assertEq(Balances.get(ROOT_NAMESPACE_ID), value); + + // Expect revert when attempting to transfer to a non-existent namespace + vm.expectRevert( + abi.encodeWithSelector(IWorldErrors.World_ResourceNotFound.selector, toNamespaceId, toNamespaceId.toString()) + ); + world.transferBalanceToNamespace(ROOT_NAMESPACE_ID, toNamespaceId, value); + } + function testTransferBalanceToAddress() public { uint256 value = 1 ether; From ca01b29cb8932fd4af4eae56134b5cfa9610dd2a Mon Sep 17 00:00:00 2001 From: yonada Date: Fri, 12 Jan 2024 15:37:39 +0000 Subject: [PATCH 22/23] test(store): add tests and gas reports for using offchain tables (#2117) Co-authored-by: Kevin Ingersoll --- packages/store/gas-report.json | 24 +++++-- packages/store/test/StoreCore.t.sol | 87 +++++++++++++++++++++++- packages/store/test/StoreCoreGas.t.sol | 91 +++++++++++++++++++++++++- 3 files changed, 194 insertions(+), 8 deletions(-) diff --git a/packages/store/gas-report.json b/packages/store/gas-report.json index 7e1e3b9316..e6d78e4c3a 100644 --- a/packages/store/gas-report.json +++ b/packages/store/gas-report.json @@ -719,6 +719,12 @@ "name": "delete record (complex data, 3 slots)", "gasUsed": 8015 }, + { + "file": "test/StoreCoreGas.t.sol", + "test": "testDeleteDataOffchainTable", + "name": "StoreCore: delete record in offchain table", + "gasUsed": 4687 + }, { "file": "test/StoreCoreGas.t.sol", "test": "testGetStaticDataLocation", @@ -753,19 +759,19 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "set record on table with subscriber", - "gasUsed": 72296 + "gasUsed": 72361 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "set static field on table with subscriber", - "gasUsed": 19791 + "gasUsed": 19813 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "delete record on table with subscriber", - "gasUsed": 18617 + "gasUsed": 18595 }, { "file": "test/StoreCoreGas.t.sol", @@ -777,19 +783,19 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "set (dynamic) record on table with subscriber", - "gasUsed": 165451 + "gasUsed": 165516 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "set (dynamic) field on table with subscriber", - "gasUsed": 24469 + "gasUsed": 24424 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "delete (dynamic) record on table with subscriber", - "gasUsed": 20283 + "gasUsed": 20261 }, { "file": "test/StoreCoreGas.t.sol", @@ -941,6 +947,12 @@ "name": "get static record (2 slots)", "gasUsed": 1740 }, + { + "file": "test/StoreCoreGas.t.sol", + "test": "testSetDataOffchainTable", + "name": "StoreCore: set record in offchain table", + "gasUsed": 8082 + }, { "file": "test/StoreCoreGas.t.sol", "test": "testUpdateInDynamicField", diff --git a/packages/store/test/StoreCore.t.sol b/packages/store/test/StoreCore.t.sol index 7638fac612..15a3983363 100644 --- a/packages/store/test/StoreCore.t.sol +++ b/packages/store/test/StoreCore.t.sol @@ -17,7 +17,7 @@ import { StoreSwitch } from "../src/StoreSwitch.sol"; import { IStoreHook } from "../src/IStoreHook.sol"; import { Tables, ResourceIds, TablesTableId } from "../src/codegen/index.sol"; import { ResourceId, ResourceIdLib, ResourceIdInstance } from "../src/ResourceId.sol"; -import { RESOURCE_TABLE } from "../src/storeResourceTypes.sol"; +import { RESOURCE_TABLE, RESOURCE_OFFCHAIN_TABLE } from "../src/storeResourceTypes.sol"; import { FieldLayoutEncodeHelper } from "./FieldLayoutEncodeHelper.sol"; import { BEFORE_SET_RECORD, AFTER_SET_RECORD, BEFORE_SPLICE_STATIC_DATA, AFTER_SPLICE_STATIC_DATA, BEFORE_SPLICE_DYNAMIC_DATA, AFTER_SPLICE_DYNAMIC_DATA, BEFORE_DELETE_RECORD, AFTER_DELETE_RECORD, ALL, BEFORE_ALL, AFTER_ALL } from "../src/storeHookTypes.sol"; import { SchemaEncodeHelper } from "./SchemaEncodeHelper.sol"; @@ -44,6 +44,7 @@ contract StoreCoreTest is Test, StoreMock { string[] defaultKeyNames = new string[](1); ResourceId _tableId = ResourceIdLib.encode({ typeId: RESOURCE_TABLE, name: "some table" }); ResourceId _tableId2 = ResourceIdLib.encode({ typeId: RESOURCE_TABLE, name: "some other table" }); + ResourceId _tableId3 = ResourceIdLib.encode({ typeId: RESOURCE_OFFCHAIN_TABLE, name: "some offchain table" }); function testGetStaticDataLocation() public { ResourceId tableId = _tableId; @@ -1318,4 +1319,88 @@ contract StoreCoreTest is Test, StoreMock { assertEq(loadedData.encodedLengths.unwrap(), bytes32(0)); assertEq(loadedData.dynamicData, ""); } + + function testSetDataOffchainTable() public { + ResourceId tableId = _tableId3; + + // Register offchain table + FieldLayout fieldLayout = FieldLayoutEncodeHelper.encode(1, 2, 1, 2, 0); + Schema valueSchema = SchemaEncodeHelper.encode( + SchemaType.UINT8, + SchemaType.UINT16, + SchemaType.UINT8, + SchemaType.UINT16 + ); + + IStore(this).registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](4)); + + // Set data + bytes memory staticData = abi.encodePacked(bytes1(0x01), bytes2(0x0203), bytes1(0x04), bytes2(0x0506)); + + bytes32[] memory keyTuple = new bytes32[](1); + keyTuple[0] = "some key"; + + // Expect a Store_SetRecord event to be emitted + vm.expectEmit(true, true, true, true); + emit Store_SetRecord(tableId, keyTuple, staticData, PackedCounter.wrap(bytes32(0)), new bytes(0)); + + IStore(this).setRecord(tableId, keyTuple, staticData, PackedCounter.wrap(bytes32(0)), new bytes(0)); + } + + function testDeleteDataOffchainTable() public { + ResourceId tableId = _tableId3; + + // Register table + FieldLayout fieldLayout = FieldLayoutEncodeHelper.encode(16, 2); + { + Schema valueSchema = SchemaEncodeHelper.encode( + SchemaType.UINT128, + SchemaType.UINT32_ARRAY, + SchemaType.UINT32_ARRAY + ); + IStore(this).registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](3)); + } + + bytes16 firstDataBytes = bytes16(0x0102030405060708090a0b0c0d0e0f10); + + bytes memory secondDataBytes; + { + uint32[] memory secondData = new uint32[](2); + secondData[0] = 0x11121314; + secondData[1] = 0x15161718; + secondDataBytes = EncodeArray.encode(secondData); + } + + bytes memory thirdDataBytes; + { + uint32[] memory thirdData = new uint32[](3); + thirdData[0] = 0x191a1b1c; + thirdData[1] = 0x1d1e1f20; + thirdData[2] = 0x21222324; + thirdDataBytes = EncodeArray.encode(thirdData); + } + + PackedCounter encodedDynamicLength; + { + encodedDynamicLength = PackedCounterLib.pack(uint40(secondDataBytes.length), uint40(thirdDataBytes.length)); + } + + // Concat data + bytes memory staticData = abi.encodePacked(firstDataBytes); + bytes memory dynamicData = abi.encodePacked(secondDataBytes, thirdDataBytes); + + // Create keyTuple + bytes32[] memory keyTuple = new bytes32[](1); + keyTuple[0] = bytes32("some key"); + + // Set data + IStore(this).setRecord(tableId, keyTuple, staticData, encodedDynamicLength, dynamicData); + + // Expect a Store_DeleteRecord event to be emitted + vm.expectEmit(true, true, true, true); + emit Store_DeleteRecord(tableId, keyTuple); + + // Delete data + IStore(this).deleteRecord(tableId, keyTuple); + } } diff --git a/packages/store/test/StoreCoreGas.t.sol b/packages/store/test/StoreCoreGas.t.sol index 3b65a03fc2..67962c21b1 100644 --- a/packages/store/test/StoreCoreGas.t.sol +++ b/packages/store/test/StoreCoreGas.t.sol @@ -16,7 +16,7 @@ import { IStoreErrors } from "../src/IStoreErrors.sol"; import { IStore } from "../src/IStore.sol"; import { ResourceId, ResourceIdLib } from "../src/ResourceId.sol"; import { ResourceIds } from "../src/codegen/tables/ResourceIds.sol"; -import { RESOURCE_TABLE } from "../src/storeResourceTypes.sol"; +import { RESOURCE_TABLE, RESOURCE_OFFCHAIN_TABLE } from "../src/storeResourceTypes.sol"; import { FieldLayoutEncodeHelper } from "./FieldLayoutEncodeHelper.sol"; import { SchemaEncodeHelper } from "./SchemaEncodeHelper.sol"; import { StoreMock } from "./StoreMock.sol"; @@ -37,6 +37,7 @@ contract StoreCoreGasTest is Test, GasReporter, StoreMock { Schema defaultKeySchema = SchemaEncodeHelper.encode(SchemaType.BYTES32); ResourceId _tableId = ResourceIdLib.encode({ typeId: RESOURCE_TABLE, name: "some table" }); ResourceId _tableId2 = ResourceIdLib.encode({ typeId: RESOURCE_TABLE, name: "some other table" }); + ResourceId _tableId3 = ResourceIdLib.encode({ typeId: RESOURCE_OFFCHAIN_TABLE, name: "some offchain table" }); function testGetStaticDataLocation() public { ResourceId tableId = _tableId; @@ -724,4 +725,92 @@ contract StoreCoreGasTest is Test, GasReporter, StoreMock { StoreCore.deleteRecord(tableId, keyTuple); endGasReport(); } + + function testSetDataOffchainTable() public { + ResourceId tableId = _tableId3; + + // Register offchain table + FieldLayout fieldLayout = FieldLayoutEncodeHelper.encode(1, 2, 1, 2, 0); + Schema valueSchema = SchemaEncodeHelper.encode( + SchemaType.UINT8, + SchemaType.UINT16, + SchemaType.UINT8, + SchemaType.UINT16 + ); + + IStore(this).registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](4)); + + // Set data + bytes memory staticData = abi.encodePacked(bytes1(0x01), bytes2(0x0203), bytes1(0x04), bytes2(0x0506)); + + bytes32[] memory keyTuple = new bytes32[](1); + keyTuple[0] = "some key"; + + // Expect a Store_SetRecord event to be emitted + vm.expectEmit(true, true, true, true); + emit Store_SetRecord(tableId, keyTuple, staticData, PackedCounter.wrap(bytes32(0)), new bytes(0)); + + startGasReport("StoreCore: set record in offchain table"); + IStore(this).setRecord(tableId, keyTuple, staticData, PackedCounter.wrap(bytes32(0)), new bytes(0)); + endGasReport(); + } + + function testDeleteDataOffchainTable() public { + ResourceId tableId = _tableId3; + + // Register table + FieldLayout fieldLayout = FieldLayoutEncodeHelper.encode(16, 2); + { + Schema valueSchema = SchemaEncodeHelper.encode( + SchemaType.UINT128, + SchemaType.UINT32_ARRAY, + SchemaType.UINT32_ARRAY + ); + IStore(this).registerTable(tableId, fieldLayout, defaultKeySchema, valueSchema, new string[](1), new string[](3)); + } + + bytes16 firstDataBytes = bytes16(0x0102030405060708090a0b0c0d0e0f10); + + bytes memory secondDataBytes; + { + uint32[] memory secondData = new uint32[](2); + secondData[0] = 0x11121314; + secondData[1] = 0x15161718; + secondDataBytes = EncodeArray.encode(secondData); + } + + bytes memory thirdDataBytes; + { + uint32[] memory thirdData = new uint32[](3); + thirdData[0] = 0x191a1b1c; + thirdData[1] = 0x1d1e1f20; + thirdData[2] = 0x21222324; + thirdDataBytes = EncodeArray.encode(thirdData); + } + + PackedCounter encodedDynamicLength; + { + encodedDynamicLength = PackedCounterLib.pack(uint40(secondDataBytes.length), uint40(thirdDataBytes.length)); + } + + // Concat data + bytes memory staticData = abi.encodePacked(firstDataBytes); + bytes memory dynamicData = abi.encodePacked(secondDataBytes, thirdDataBytes); + + // Create keyTuple + bytes32[] memory keyTuple = new bytes32[](1); + keyTuple[0] = bytes32("some key"); + + // Set data + IStore(this).setRecord(tableId, keyTuple, staticData, encodedDynamicLength, dynamicData); + + // Expect a Store_DeleteRecord event to be emitted + vm.expectEmit(true, true, true, true); + emit Store_DeleteRecord(tableId, keyTuple); + + // Delete data + startGasReport("StoreCore: delete record in offchain table"); + IStore(this).deleteRecord(tableId, keyTuple); + endGasReport(); + } } From c6c13f2ea7e405cac2bc9cf77659d2d66bfdc0d2 Mon Sep 17 00:00:00 2001 From: yonada Date: Fri, 12 Jan 2024 15:57:32 +0000 Subject: [PATCH 23/23] fix(store): emit event after calling beforeSetRecord hook [L-02] (#2017) Co-authored-by: Kevin Ingersoll --- .changeset/beige-ads-melt.md | 5 ++ packages/store/gas-report.json | 78 ++++++++++++------------ packages/store/src/StoreCore.sol | 68 +++++++++++---------- packages/world-modules/gas-report.json | 82 +++++++++++++------------- packages/world/gas-report.json | 28 ++++----- 5 files changed, 136 insertions(+), 125 deletions(-) create mode 100644 .changeset/beige-ads-melt.md diff --git a/.changeset/beige-ads-melt.md b/.changeset/beige-ads-melt.md new file mode 100644 index 0000000000..01846b4024 --- /dev/null +++ b/.changeset/beige-ads-melt.md @@ -0,0 +1,5 @@ +--- +"@latticexyz/store": patch +--- + +Storage events are now emitted after "before" hooks, so that the resulting logs are now correctly ordered and reflect onchain logic. This resolves issues with store writes and event emissions happening in "before" hooks. diff --git a/packages/store/gas-report.json b/packages/store/gas-report.json index e6d78e4c3a..1d4043de4c 100644 --- a/packages/store/gas-report.json +++ b/packages/store/gas-report.json @@ -63,7 +63,7 @@ "file": "test/Callbacks.t.sol", "test": "testSetAndGet", "name": "Callbacks: set field", - "gasUsed": 56256 + "gasUsed": 56271 }, { "file": "test/Callbacks.t.sol", @@ -75,7 +75,7 @@ "file": "test/Callbacks.t.sol", "test": "testSetAndGet", "name": "Callbacks: push 1 element", - "gasUsed": 32514 + "gasUsed": 32530 }, { "file": "test/FieldLayout.t.sol", @@ -369,7 +369,7 @@ "file": "test/KeyEncoding.t.sol", "test": "testRegisterAndGetFieldLayout", "name": "register KeyEncoding table", - "gasUsed": 719032 + "gasUsed": 719062 }, { "file": "test/Mixed.t.sol", @@ -393,7 +393,7 @@ "file": "test/Mixed.t.sol", "test": "testSetGetDeleteExternal", "name": "set record in Mixed (external, cold)", - "gasUsed": 108527 + "gasUsed": 108528 }, { "file": "test/Mixed.t.sol", @@ -411,7 +411,7 @@ "file": "test/Mixed.t.sol", "test": "testSetGetDeleteInternal", "name": "set record in Mixed (internal, cold)", - "gasUsed": 103281 + "gasUsed": 103282 }, { "file": "test/Mixed.t.sol", @@ -669,25 +669,25 @@ "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromSecondField", "name": "pop from field (cold, 1 slot, 1 uint32 item)", - "gasUsed": 18032 + "gasUsed": 18041 }, { "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromSecondField", "name": "pop from field (warm, 1 slot, 1 uint32 item)", - "gasUsed": 12039 + "gasUsed": 12049 }, { "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromThirdField", "name": "pop from field (cold, 2 slots, 10 uint32 items)", - "gasUsed": 15800 + "gasUsed": 15810 }, { "file": "test/StoreCoreDynamic.t.sol", "test": "testPopFromThirdField", "name": "pop from field (warm, 2 slots, 10 uint32 items)", - "gasUsed": 11807 + "gasUsed": 11818 }, { "file": "test/StoreCoreGas.t.sol", @@ -759,13 +759,13 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "set record on table with subscriber", - "gasUsed": 72361 + "gasUsed": 72385 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooks", "name": "set static field on table with subscriber", - "gasUsed": 19813 + "gasUsed": 19837 }, { "file": "test/StoreCoreGas.t.sol", @@ -783,13 +783,13 @@ "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "set (dynamic) record on table with subscriber", - "gasUsed": 165516 + "gasUsed": 165525 }, { "file": "test/StoreCoreGas.t.sol", "test": "testHooksDynamicData", "name": "set (dynamic) field on table with subscriber", - "gasUsed": 24424 + "gasUsed": 24446 }, { "file": "test/StoreCoreGas.t.sol", @@ -801,19 +801,19 @@ "file": "test/StoreCoreGas.t.sol", "test": "testPushToDynamicField", "name": "push to field (1 slot, 1 uint32 item)", - "gasUsed": 9459 + "gasUsed": 9475 }, { "file": "test/StoreCoreGas.t.sol", "test": "testPushToDynamicField", "name": "push to field (2 slots, 10 uint32 items)", - "gasUsed": 32129 + "gasUsed": 32150 }, { "file": "test/StoreCoreGas.t.sol", "test": "testRegisterAndGetFieldLayout", "name": "StoreCore: register table", - "gasUsed": 640914 + "gasUsed": 640944 }, { "file": "test/StoreCoreGas.t.sol", @@ -837,7 +837,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetDynamicData", "name": "set complex record with dynamic data (4 slots)", - "gasUsed": 102512 + "gasUsed": 102513 }, { "file": "test/StoreCoreGas.t.sol", @@ -879,7 +879,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set static field (1 slot)", - "gasUsed": 31224 + "gasUsed": 31252 }, { "file": "test/StoreCoreGas.t.sol", @@ -891,7 +891,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set static field (overlap 2 slot)", - "gasUsed": 29877 + "gasUsed": 29908 }, { "file": "test/StoreCoreGas.t.sol", @@ -903,7 +903,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set dynamic field (1 slot, first dynamic field)", - "gasUsed": 53945 + "gasUsed": 53961 }, { "file": "test/StoreCoreGas.t.sol", @@ -915,7 +915,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetField", "name": "set dynamic field (1 slot, second dynamic field)", - "gasUsed": 32168 + "gasUsed": 32186 }, { "file": "test/StoreCoreGas.t.sol", @@ -927,7 +927,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetStaticData", "name": "set static record (1 slot)", - "gasUsed": 32756 + "gasUsed": 32782 }, { "file": "test/StoreCoreGas.t.sol", @@ -939,7 +939,7 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetAndGetStaticDataSpanningWords", "name": "set static record (2 slots)", - "gasUsed": 55260 + "gasUsed": 55289 }, { "file": "test/StoreCoreGas.t.sol", @@ -951,19 +951,19 @@ "file": "test/StoreCoreGas.t.sol", "test": "testSetDataOffchainTable", "name": "StoreCore: set record in offchain table", - "gasUsed": 8082 + "gasUsed": 8093 }, { "file": "test/StoreCoreGas.t.sol", "test": "testUpdateInDynamicField", "name": "update in field (1 slot, 1 uint32 item)", - "gasUsed": 8807 + "gasUsed": 8835 }, { "file": "test/StoreCoreGas.t.sol", "test": "testUpdateInDynamicField", "name": "push to field (2 slots, 6 uint64 items)", - "gasUsed": 9253 + "gasUsed": 9282 }, { "file": "test/StoreHook.t.sol", @@ -993,13 +993,13 @@ "file": "test/StoreHooks.t.sol", "test": "testOneSlot", "name": "StoreHooks: set field with one elements (cold)", - "gasUsed": 58262 + "gasUsed": 58275 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: set field (cold)", - "gasUsed": 58262 + "gasUsed": 58275 }, { "file": "test/StoreHooks.t.sol", @@ -1011,25 +1011,25 @@ "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: push 1 element (cold)", - "gasUsed": 12612 + "gasUsed": 12626 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: pop 1 element (warm)", - "gasUsed": 9946 + "gasUsed": 9958 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: push 1 element (warm)", - "gasUsed": 10629 + "gasUsed": 10646 }, { "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: update 1 element (warm)", - "gasUsed": 29855 + "gasUsed": 29886 }, { "file": "test/StoreHooks.t.sol", @@ -1041,19 +1041,19 @@ "file": "test/StoreHooks.t.sol", "test": "testTable", "name": "StoreHooks: set field (warm)", - "gasUsed": 30407 + "gasUsed": 30427 }, { "file": "test/StoreHooks.t.sol", "test": "testThreeSlots", "name": "StoreHooks: set field with three elements (cold)", - "gasUsed": 80950 + "gasUsed": 80966 }, { "file": "test/StoreHooks.t.sol", "test": "testTwoSlots", "name": "StoreHooks: set field with two elements (cold)", - "gasUsed": 80862 + "gasUsed": 80878 }, { "file": "test/StoreHooksColdLoad.t.sol", @@ -1083,13 +1083,13 @@ "file": "test/StoreHooksColdLoad.t.sol", "test": "testPop", "name": "StoreHooks: pop 1 element (cold)", - "gasUsed": 18381 + "gasUsed": 18390 }, { "file": "test/StoreHooksColdLoad.t.sol", "test": "testUpdate", "name": "StoreHooks: update 1 element (cold)", - "gasUsed": 20310 + "gasUsed": 20333 }, { "file": "test/StoreSwitch.t.sol", @@ -1155,13 +1155,13 @@ "file": "test/Vector2.t.sol", "test": "testRegisterAndGetFieldLayout", "name": "register Vector2 field layout", - "gasUsed": 442418 + "gasUsed": 442447 }, { "file": "test/Vector2.t.sol", "test": "testSetAndGet", "name": "set Vector2 record", - "gasUsed": 33658 + "gasUsed": 33684 }, { "file": "test/Vector2.t.sol", diff --git a/packages/store/src/StoreCore.sol b/packages/store/src/StoreCore.sol index 50b2028d7d..ff64ea8598 100644 --- a/packages/store/src/StoreCore.sol +++ b/packages/store/src/StoreCore.sol @@ -298,11 +298,10 @@ library StoreCore { bytes memory dynamicData, FieldLayout fieldLayout ) internal { - // Emit event to notify indexers - emit Store_SetRecord(tableId, keyTuple, staticData, encodedLengths, dynamicData); - // Early return if the table is an offchain table - if (tableId.getType() != RESOURCE_TABLE) { + if (tableId.getType() == RESOURCE_OFFCHAIN_TABLE) { + // Emit event to notify indexers + emit Store_SetRecord(tableId, keyTuple, staticData, encodedLengths, dynamicData); return; } @@ -322,6 +321,9 @@ library StoreCore { } } + // Emit event to notify indexers + emit Store_SetRecord(tableId, keyTuple, staticData, encodedLengths, dynamicData); + // Store the static data at the static data location uint256 staticDataLocation = StoreCoreInternal._getStaticDataLocation(tableId, keyTuple); uint256 memoryPointer = Memory.dataPointer(staticData); @@ -387,16 +389,15 @@ library StoreCore { * @param data The data to write to the static data of the record at the start byte. */ function spliceStaticData(ResourceId tableId, bytes32[] memory keyTuple, uint48 start, bytes memory data) internal { - uint256 location = StoreCoreInternal._getStaticDataLocation(tableId, keyTuple); - - // Emit event to notify offchain indexers - emit StoreCore.Store_SpliceStaticData({ tableId: tableId, keyTuple: keyTuple, start: start, data: data }); - // Early return if the table is an offchain table - if (tableId.getType() != RESOURCE_TABLE) { + if (tableId.getType() == RESOURCE_OFFCHAIN_TABLE) { + // Emit event to notify offchain indexers + emit StoreCore.Store_SpliceStaticData({ tableId: tableId, keyTuple: keyTuple, start: start, data: data }); return; } + uint256 location = StoreCoreInternal._getStaticDataLocation(tableId, keyTuple); + // Call onBeforeSpliceStaticData hooks (before actually modifying the state, so observers have access to the previous state if needed) bytes21[] memory hooks = StoreHooks._get(tableId); for (uint256 i; i < hooks.length; i++) { @@ -411,6 +412,9 @@ library StoreCore { } } + // Emit event to notify offchain indexers + emit StoreCore.Store_SpliceStaticData({ tableId: tableId, keyTuple: keyTuple, start: start, data: data }); + // Store the provided value in storage Storage.store({ storagePointer: location, offset: start, data: data }); @@ -583,11 +587,10 @@ library StoreCore { * @param fieldLayout The field layout for the record. */ function deleteRecord(ResourceId tableId, bytes32[] memory keyTuple, FieldLayout fieldLayout) internal { - // Emit event to notify indexers - emit Store_DeleteRecord(tableId, keyTuple); - // Early return if the table is an offchain table - if (tableId.getType() != RESOURCE_TABLE) { + if (tableId.getType() == RESOURCE_OFFCHAIN_TABLE) { + // Emit event to notify indexers + emit Store_DeleteRecord(tableId, keyTuple); return; } @@ -600,6 +603,9 @@ library StoreCore { } } + // Emit event to notify indexers + emit Store_DeleteRecord(tableId, keyTuple); + // Delete static data uint256 staticDataLocation = StoreCoreInternal._getStaticDataLocation(tableId, keyTuple); Storage.store({ storagePointer: staticDataLocation, offset: 0, data: new bytes(fieldLayout.staticDataLength()) }); @@ -989,6 +995,23 @@ library StoreCoreInternal { // Update the encoded length PackedCounter updatedEncodedLengths = previousEncodedLengths.setAtIndex(dynamicFieldIndex, updatedFieldLength); + // Call onBeforeSpliceDynamicData hooks (before actually modifying the state, so observers have access to the previous state if needed) + bytes21[] memory hooks = StoreHooks._get(tableId); + for (uint256 i; i < hooks.length; i++) { + Hook hook = Hook.wrap(hooks[i]); + if (hook.isEnabled(BEFORE_SPLICE_DYNAMIC_DATA)) { + IStoreHook(hook.getAddress()).onBeforeSpliceDynamicData({ + tableId: tableId, + keyTuple: keyTuple, + dynamicFieldIndex: dynamicFieldIndex, + startWithinField: startWithinField, + deleteCount: deleteCount, + encodedLengths: previousEncodedLengths, + data: data + }); + } + } + { // Compute start index for the splice uint256 start = startWithinField; @@ -1010,23 +1033,6 @@ library StoreCoreInternal { }); } - // Call onBeforeSpliceDynamicData hooks (before actually modifying the state, so observers have access to the previous state if needed) - bytes21[] memory hooks = StoreHooks._get(tableId); - for (uint256 i; i < hooks.length; i++) { - Hook hook = Hook.wrap(hooks[i]); - if (hook.isEnabled(BEFORE_SPLICE_DYNAMIC_DATA)) { - IStoreHook(hook.getAddress()).onBeforeSpliceDynamicData({ - tableId: tableId, - keyTuple: keyTuple, - dynamicFieldIndex: dynamicFieldIndex, - startWithinField: startWithinField, - deleteCount: deleteCount, - encodedLengths: previousEncodedLengths, - data: data - }); - } - } - // Store the updated encoded lengths in storage if (previousFieldLength != updatedFieldLength) { uint256 dynamicSchemaLengthSlot = _getDynamicDataLengthLocation(tableId, keyTuple); diff --git a/packages/world-modules/gas-report.json b/packages/world-modules/gas-report.json index 87d1a9b1d4..885089bac9 100644 --- a/packages/world-modules/gas-report.json +++ b/packages/world-modules/gas-report.json @@ -3,139 +3,139 @@ "file": "test/ERC20.t.sol", "test": "testApprove", "name": "approve", - "gasUsed": 114344 + "gasUsed": 114366 }, { "file": "test/ERC20.t.sol", "test": "testBurn", "name": "burn", - "gasUsed": 75896 + "gasUsed": 75931 }, { "file": "test/ERC20.t.sol", "test": "testMint", "name": "mint", - "gasUsed": 161735 + "gasUsed": 161770 }, { "file": "test/ERC20.t.sol", "test": "testTransfer", "name": "transfer", - "gasUsed": 92978 + "gasUsed": 93016 }, { "file": "test/ERC20.t.sol", "test": "testTransferFrom", "name": "transferFrom", - "gasUsed": 130295 + "gasUsed": 130355 }, { "file": "test/ERC721.t.sol", "test": "testApproveAllGas", "name": "setApprovalForAll", - "gasUsed": 113971 + "gasUsed": 113993 }, { "file": "test/ERC721.t.sol", "test": "testApproveGas", "name": "approve", - "gasUsed": 87980 + "gasUsed": 87999 }, { "file": "test/ERC721.t.sol", "test": "testBurnGas", "name": "burn", - "gasUsed": 101880 + "gasUsed": 101937 }, { "file": "test/ERC721.t.sol", "test": "testMintGas", "name": "mint", - "gasUsed": 169463 + "gasUsed": 169501 }, { "file": "test/ERC721.t.sol", "test": "testSafeMintToEOAGas", "name": "safeMint", - "gasUsed": 169734 + "gasUsed": 169772 }, { "file": "test/ERC721.t.sol", "test": "testSafeTransferFromToEOAGas", "name": "safeTransferFrom", - "gasUsed": 143698 + "gasUsed": 143774 }, { "file": "test/ERC721.t.sol", "test": "testTransferFromGas", "name": "transferFrom", - "gasUsed": 136858 + "gasUsed": 136934 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallComposite", "name": "install keys in table module", - "gasUsed": 1413282 + "gasUsed": 1413433 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallGas", "name": "install keys in table module", - "gasUsed": 1413282 + "gasUsed": 1413433 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallGas", "name": "set a record on a table with keysInTableModule installed", - "gasUsed": 158829 + "gasUsed": 158854 }, { "file": "test/KeysInTableModule.t.sol", "test": "testInstallSingleton", "name": "install keys in table module", - "gasUsed": 1413282 + "gasUsed": 1413433 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookCompositeGas", "name": "install keys in table module", - "gasUsed": 1413282 + "gasUsed": 1413433 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookCompositeGas", "name": "change a composite record on a table with keysInTableModule installed", - "gasUsed": 22497 + "gasUsed": 22492 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookCompositeGas", "name": "delete a composite record on a table with keysInTableModule installed", - "gasUsed": 155864 + "gasUsed": 155971 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookGas", "name": "install keys in table module", - "gasUsed": 1413282 + "gasUsed": 1413433 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookGas", "name": "change a record on a table with keysInTableModule installed", - "gasUsed": 21219 + "gasUsed": 21214 }, { "file": "test/KeysInTableModule.t.sol", "test": "testSetAndDeleteRecordHookGas", "name": "delete a record on a table with keysInTableModule installed", - "gasUsed": 85046 + "gasUsed": 85089 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testGetKeysWithValueGas", "name": "install keys with value module", - "gasUsed": 668083 + "gasUsed": 668206 }, { "file": "test/KeysWithValueModule.t.sol", @@ -153,49 +153,49 @@ "file": "test/KeysWithValueModule.t.sol", "test": "testInstall", "name": "install keys with value module", - "gasUsed": 668083 + "gasUsed": 668206 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testInstall", "name": "set a record on a table with KeysWithValueModule installed", - "gasUsed": 135435 + "gasUsed": 135437 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "install keys with value module", - "gasUsed": 668083 + "gasUsed": 668206 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "change a record on a table with KeysWithValueModule installed", - "gasUsed": 103819 + "gasUsed": 103840 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetAndDeleteRecordHook", "name": "delete a record on a table with KeysWithValueModule installed", - "gasUsed": 36501 + "gasUsed": 36489 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "install keys with value module", - "gasUsed": 668083 + "gasUsed": 668206 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "set a field on a table with KeysWithValueModule installed", - "gasUsed": 146671 + "gasUsed": 146672 }, { "file": "test/KeysWithValueModule.t.sol", "test": "testSetField", "name": "change a field on a table with KeysWithValueModule installed", - "gasUsed": 111430 + "gasUsed": 111431 }, { "file": "test/query.t.sol", @@ -267,60 +267,60 @@ "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromCallboundDelegation", "name": "register a callbound delegation", - "gasUsed": 118319 + "gasUsed": 118347 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromCallboundDelegation", "name": "call a system via a callbound delegation", - "gasUsed": 36703 + "gasUsed": 36697 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromSystemDelegation", "name": "register a systembound delegation", - "gasUsed": 115875 + "gasUsed": 115900 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromSystemDelegation", "name": "call a system via a systembound delegation", - "gasUsed": 33846 + "gasUsed": 33869 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromTimeboundDelegation", "name": "register a timebound delegation", - "gasUsed": 112813 + "gasUsed": 112823 }, { "file": "test/StandardDelegationsModule.t.sol", "test": "testCallFromTimeboundDelegation", "name": "call a system via a timebound delegation", - "gasUsed": 26815 + "gasUsed": 26809 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstall", "name": "install unique entity module", - "gasUsed": 694780 + "gasUsed": 694917 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstall", "name": "get a unique entity nonce (non-root module)", - "gasUsed": 51041 + "gasUsed": 51045 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstallRoot", "name": "installRoot unique entity module", - "gasUsed": 663774 + "gasUsed": 663866 }, { "file": "test/UniqueEntityModule.t.sol", "test": "testInstallRoot", "name": "get a unique entity nonce (root module)", - "gasUsed": 51041 + "gasUsed": 51045 } ] diff --git a/packages/world/gas-report.json b/packages/world/gas-report.json index 9caba2f8e9..5c87427efd 100644 --- a/packages/world/gas-report.json +++ b/packages/world/gas-report.json @@ -69,7 +69,7 @@ "file": "test/World.t.sol", "test": "testCallFromUnlimitedDelegation", "name": "register an unlimited delegation", - "gasUsed": 47632 + "gasUsed": 47651 }, { "file": "test/World.t.sol", @@ -81,79 +81,79 @@ "file": "test/World.t.sol", "test": "testDeleteRecord", "name": "Delete record", - "gasUsed": 9852 + "gasUsed": 9846 }, { "file": "test/World.t.sol", "test": "testPushToDynamicField", "name": "Push data to the table", - "gasUsed": 85826 + "gasUsed": 85842 }, { "file": "test/World.t.sol", "test": "testRegisterFunctionSelector", "name": "Register a function selector", - "gasUsed": 83138 + "gasUsed": 83156 }, { "file": "test/World.t.sol", "test": "testRegisterNamespace", "name": "Register a new namespace", - "gasUsed": 120932 + "gasUsed": 120962 }, { "file": "test/World.t.sol", "test": "testRegisterRootFunctionSelector", "name": "Register a root function selector", - "gasUsed": 80444 + "gasUsed": 80462 }, { "file": "test/World.t.sol", "test": "testRegisterSystem", "name": "register a system", - "gasUsed": 164317 + "gasUsed": 164349 }, { "file": "test/World.t.sol", "test": "testRegisterTable", "name": "Register a new table in the namespace", - "gasUsed": 528393 + "gasUsed": 528422 }, { "file": "test/World.t.sol", "test": "testSetField", "name": "Write data to a table field", - "gasUsed": 36896 + "gasUsed": 36912 }, { "file": "test/World.t.sol", "test": "testSetRecord", "name": "Write data to the table", - "gasUsed": 39039 + "gasUsed": 39056 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testPopFromDynamicField", "name": "pop 1 address (cold)", - "gasUsed": 25617 + "gasUsed": 25627 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testPopFromDynamicField", "name": "pop 1 address (warm)", - "gasUsed": 12763 + "gasUsed": 12773 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testSpliceDynamicData", "name": "update in field 1 item (cold)", - "gasUsed": 25968 + "gasUsed": 25990 }, { "file": "test/WorldDynamicUpdate.t.sol", "test": "testSpliceDynamicData", "name": "update in field 1 item (warm)", - "gasUsed": 13169 + "gasUsed": 13191 }, { "file": "test/WorldResourceId.t.sol",