From aaec66950d82543d89520ffa8a2b41d4aa698685 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Mon, 11 Sep 2023 10:32:48 +0100 Subject: [PATCH 1/6] feat(protocol-parser): add helpers to make it easier to work with keySchema/valueSchema --- packages/protocol-parser/src/common.ts | 17 ++++++++++++++++- packages/protocol-parser/src/decodeKey.ts | 15 +++++++++++++++ packages/protocol-parser/src/decodeKeyTuple.ts | 1 + .../protocol-parser/src/decodeRecord.test.ts | 12 ++++++++++++ packages/protocol-parser/src/decodeRecord.ts | 10 ++++++---- packages/protocol-parser/src/decodeValue.ts | 16 ++++++++++++++++ packages/protocol-parser/src/encodeField.ts | 11 +++++++---- packages/protocol-parser/src/encodeKey.ts | 10 ++++++++++ packages/protocol-parser/src/encodeKeyTuple.ts | 1 + .../protocol-parser/src/encodeRecord.test.ts | 8 +++++++- packages/protocol-parser/src/encodeRecord.ts | 7 ++++--- packages/protocol-parser/src/encodeValue.ts | 18 ++++++++++++++++++ .../protocol-parser/src/hexToPackedCounter.ts | 7 ++++--- packages/protocol-parser/src/index.ts | 7 +++++++ packages/protocol-parser/src/keySchemaToHex.ts | 8 ++++++++ .../protocol-parser/src/padSliceHex.test.ts | 15 +++++++++++++++ packages/protocol-parser/src/padSliceHex.ts | 8 ++++++++ packages/protocol-parser/src/schemaToHex.ts | 1 + .../protocol-parser/src/valueSchemaToHex.ts | 11 +++++++++++ 19 files changed, 167 insertions(+), 16 deletions(-) create mode 100644 packages/protocol-parser/src/decodeKey.ts create mode 100644 packages/protocol-parser/src/decodeValue.ts create mode 100644 packages/protocol-parser/src/encodeKey.ts create mode 100644 packages/protocol-parser/src/encodeValue.ts create mode 100644 packages/protocol-parser/src/keySchemaToHex.ts create mode 100644 packages/protocol-parser/src/padSliceHex.test.ts create mode 100644 packages/protocol-parser/src/padSliceHex.ts create mode 100644 packages/protocol-parser/src/valueSchemaToHex.ts diff --git a/packages/protocol-parser/src/common.ts b/packages/protocol-parser/src/common.ts index ba9cc72a69..9ff1790395 100644 --- a/packages/protocol-parser/src/common.ts +++ b/packages/protocol-parser/src/common.ts @@ -1,11 +1,26 @@ -import { DynamicAbiType, StaticAbiType } from "@latticexyz/schema-type"; +import { DynamicAbiType, SchemaAbiType, SchemaAbiTypeToPrimitiveType, StaticAbiType } from "@latticexyz/schema-type"; +/** @deprecated use `KeySchema` or `ValueSchema` instead */ export type Schema = { readonly staticFields: readonly StaticAbiType[]; readonly dynamicFields: readonly DynamicAbiType[]; }; +/** @deprecated use `KeySchema` and `ValueSchema` instead */ export type TableSchema = { readonly keySchema: Schema; // TODO: refine to set dynamicFields to [] readonly valueSchema: Schema; }; + +export type KeySchema = Record; +export type ValueSchema = Record; + +/** Map a table schema like `{ value: "uint256" }` to its primitive types like `{ value: bigint }` */ +export type SchemaToPrimitives = { + [key in keyof TSchema]: SchemaAbiTypeToPrimitiveType; +}; + +export type TableRecord = { + key: SchemaToPrimitives; + value: SchemaToPrimitives; +}; diff --git a/packages/protocol-parser/src/decodeKey.ts b/packages/protocol-parser/src/decodeKey.ts new file mode 100644 index 0000000000..2b11552ad8 --- /dev/null +++ b/packages/protocol-parser/src/decodeKey.ts @@ -0,0 +1,15 @@ +import { Hex } from "viem"; +import { SchemaToPrimitives, KeySchema } from "./common"; +import { decodeKeyTuple } from "./decodeKeyTuple"; + +export function decodeKey( + keySchema: TSchema, + data: readonly Hex[] +): SchemaToPrimitives { + // TODO: refactor and move all decodeKeyTuple logic into this method so we can delete decodeKeyTuple + const keyValues = decodeKeyTuple({ staticFields: Object.values(keySchema), dynamicFields: [] }, data); + + return Object.fromEntries( + Object.keys(keySchema).map((name, i) => [name, keyValues[i]]) + ) as SchemaToPrimitives; +} diff --git a/packages/protocol-parser/src/decodeKeyTuple.ts b/packages/protocol-parser/src/decodeKeyTuple.ts index 81342f13d4..623df1fac1 100644 --- a/packages/protocol-parser/src/decodeKeyTuple.ts +++ b/packages/protocol-parser/src/decodeKeyTuple.ts @@ -4,6 +4,7 @@ import { Schema } from "./common"; // key tuples are encoded in the same way as abi.encode, so we can decode them with viem +/** @deprecated use `decodeKey` instead */ export function decodeKeyTuple(keySchema: Schema, keyTuple: readonly Hex[]): StaticPrimitiveType[] { if (keySchema.staticFields.length !== keyTuple.length) { throw new Error( diff --git a/packages/protocol-parser/src/decodeRecord.test.ts b/packages/protocol-parser/src/decodeRecord.test.ts index a5bca516e6..390dcc6ade 100644 --- a/packages/protocol-parser/src/decodeRecord.test.ts +++ b/packages/protocol-parser/src/decodeRecord.test.ts @@ -21,4 +21,16 @@ describe("decodeRecord", () => { ] `); }); + + it("can decode an out of bounds array", () => { + const schema = { staticFields: [], dynamicFields: ["uint32[]"] } as const; + const values = decodeRecord(schema, "0x0000000000000000000000000000000000000000000000000400000000000004"); + expect(values).toMatchInlineSnapshot(` + [ + [ + 0, + ], + ] + `); + }); }); diff --git a/packages/protocol-parser/src/decodeRecord.ts b/packages/protocol-parser/src/decodeRecord.ts index 6e1a4fb46e..ce355acff8 100644 --- a/packages/protocol-parser/src/decodeRecord.ts +++ b/packages/protocol-parser/src/decodeRecord.ts @@ -4,20 +4,22 @@ import { staticAbiTypeToByteLength, dynamicAbiTypeToDefaultValue, } from "@latticexyz/schema-type"; -import { Hex, sliceHex } from "viem"; +import { Hex } from "viem"; import { Schema } from "./common"; import { decodeDynamicField } from "./decodeDynamicField"; import { decodeStaticField } from "./decodeStaticField"; import { hexToPackedCounter } from "./hexToPackedCounter"; import { staticDataLength } from "./staticDataLength"; +import { padSliceHex } from "./padSliceHex"; +/** @deprecated use `decodeValue` instead */ export function decodeRecord(schema: Schema, data: Hex): readonly (StaticPrimitiveType | DynamicPrimitiveType)[] { const values: (StaticPrimitiveType | DynamicPrimitiveType)[] = []; let bytesOffset = 0; schema.staticFields.forEach((fieldType) => { const fieldByteLength = staticAbiTypeToByteLength[fieldType]; - const value = decodeStaticField(fieldType, sliceHex(data, bytesOffset, bytesOffset + fieldByteLength)); + const value = decodeStaticField(fieldType, padSliceHex(data, bytesOffset, bytesOffset + fieldByteLength)); bytesOffset += fieldByteLength; values.push(value); }); @@ -37,13 +39,13 @@ export function decodeRecord(schema: Schema, data: Hex): readonly (StaticPrimiti } if (schema.dynamicFields.length > 0) { - const dataLayout = hexToPackedCounter(sliceHex(data, bytesOffset, bytesOffset + 32)); + const dataLayout = hexToPackedCounter(padSliceHex(data, bytesOffset, bytesOffset + 32)); bytesOffset += 32; schema.dynamicFields.forEach((fieldType, i) => { const dataLength = dataLayout.fieldByteLengths[i]; if (dataLength > 0) { - const value = decodeDynamicField(fieldType, sliceHex(data, bytesOffset, bytesOffset + dataLength)); + const value = decodeDynamicField(fieldType, padSliceHex(data, bytesOffset, bytesOffset + dataLength)); bytesOffset += dataLength; values.push(value); } else { diff --git a/packages/protocol-parser/src/decodeValue.ts b/packages/protocol-parser/src/decodeValue.ts new file mode 100644 index 0000000000..bd96093fed --- /dev/null +++ b/packages/protocol-parser/src/decodeValue.ts @@ -0,0 +1,16 @@ +import { isStaticAbiType, isDynamicAbiType } from "@latticexyz/schema-type"; +import { Hex } from "viem"; +import { SchemaToPrimitives, ValueSchema } from "./common"; +import { decodeRecord } from "./decodeRecord"; + +export function decodeValue(valueSchema: TSchema, data: Hex): SchemaToPrimitives { + const staticFields = Object.values(valueSchema).filter(isStaticAbiType); + const dynamicFields = Object.values(valueSchema).filter(isDynamicAbiType); + + // TODO: refactor and move all decodeRecord logic into this method so we can delete decodeRecord + const valueTuple = decodeRecord({ staticFields, dynamicFields }, data); + + return Object.fromEntries( + Object.keys(valueSchema).map((name, i) => [name, valueTuple[i]]) + ) as SchemaToPrimitives; +} diff --git a/packages/protocol-parser/src/encodeField.ts b/packages/protocol-parser/src/encodeField.ts index 89155d6531..99e977a850 100644 --- a/packages/protocol-parser/src/encodeField.ts +++ b/packages/protocol-parser/src/encodeField.ts @@ -8,10 +8,13 @@ export function encodeField( ): Hex { if (isArrayAbiType(fieldType) && Array.isArray(value)) { const staticFieldType = arrayAbiTypeToStaticAbiType(fieldType); - return encodePacked( - value.map(() => staticFieldType), - value - ); + // TODO: we can remove conditional once this is fixed: https://github.com/wagmi-dev/viem/pull/1147 + return value.length === 0 + ? "0x" + : encodePacked( + value.map(() => staticFieldType), + value + ); } return encodePacked([fieldType], [value]); } diff --git a/packages/protocol-parser/src/encodeKey.ts b/packages/protocol-parser/src/encodeKey.ts new file mode 100644 index 0000000000..f98ac04210 --- /dev/null +++ b/packages/protocol-parser/src/encodeKey.ts @@ -0,0 +1,10 @@ +import { isStaticAbiType } from "@latticexyz/schema-type"; +import { Hex } from "viem"; +import { SchemaToPrimitives, KeySchema } from "./common"; +import { encodeKeyTuple } from "./encodeKeyTuple"; + +export function encodeKey(keySchema: TSchema, key: SchemaToPrimitives): Hex[] { + const staticFields = Object.values(keySchema).filter(isStaticAbiType); + // TODO: refactor and move all encodeKey logic into this method so we can delete encodeKey + return encodeKeyTuple({ staticFields, dynamicFields: [] }, Object.values(key)); +} diff --git a/packages/protocol-parser/src/encodeKeyTuple.ts b/packages/protocol-parser/src/encodeKeyTuple.ts index c52ffae0ac..e81337cac0 100644 --- a/packages/protocol-parser/src/encodeKeyTuple.ts +++ b/packages/protocol-parser/src/encodeKeyTuple.ts @@ -2,6 +2,7 @@ import { StaticPrimitiveType } from "@latticexyz/schema-type"; import { Hex, encodeAbiParameters } from "viem"; import { Schema } from "./common"; +/** @deprecated use `encodeKey` instead */ export function encodeKeyTuple(keySchema: Schema, keyTuple: StaticPrimitiveType[]): Hex[] { return keyTuple.map((key, index) => encodeAbiParameters([{ type: keySchema.staticFields[index] }], [key])); } diff --git a/packages/protocol-parser/src/encodeRecord.test.ts b/packages/protocol-parser/src/encodeRecord.test.ts index e85bec0b39..fde765d9b6 100644 --- a/packages/protocol-parser/src/encodeRecord.test.ts +++ b/packages/protocol-parser/src/encodeRecord.test.ts @@ -6,7 +6,7 @@ describe("encodeRecord", () => { const schema = { staticFields: ["uint32", "uint128"], dynamicFields: ["uint32[]", "string"] } as const; const hex = encodeRecord(schema, [1, 2n, [3, 4], "some string"]); expect(hex).toBe( - "0x0000000100000000000000000000000000000002000000000000130000000008000000000b0000000000000000000000000000000000000300000004736f6d6520737472696e67" + "0x0000000100000000000000000000000000000002000000000000000000000000000000000000000b0000000008000000000000130000000300000004736f6d6520737472696e67" ); }); @@ -15,4 +15,10 @@ describe("encodeRecord", () => { const hex = encodeRecord(schema, [1, 2n]); expect(hex).toBe("0x0000000100000000000000000000000000000002"); }); + + it("can encode an array to hex", () => { + const schema = { staticFields: [], dynamicFields: ["uint32[]"] } as const; + const hex = encodeRecord(schema, [[42]]); + expect(hex).toBe("0x00000000000000000000000000000000000000000000000004000000000000040000002a"); + }); }); diff --git a/packages/protocol-parser/src/encodeRecord.ts b/packages/protocol-parser/src/encodeRecord.ts index 23a1c16fea..b224a6c532 100644 --- a/packages/protocol-parser/src/encodeRecord.ts +++ b/packages/protocol-parser/src/encodeRecord.ts @@ -3,6 +3,7 @@ import { Hex } from "viem"; import { encodeField } from "./encodeField"; import { Schema } from "./common"; +/** @deprecated use `encodeValue` instead */ export function encodeRecord(schema: Schema, values: readonly (StaticPrimitiveType | DynamicPrimitiveType)[]): Hex { const staticValues = values.slice(0, schema.staticFields.length) as readonly StaticPrimitiveType[]; const dynamicValues = values.slice(schema.staticFields.length) as readonly DynamicPrimitiveType[]; @@ -17,14 +18,14 @@ export function encodeRecord(schema: Schema, values: readonly (StaticPrimitiveTy encodeField(schema.dynamicFields[i], value).replace(/^0x/, "") ); - const dynamicFieldByteLengths = dynamicDataItems.map((value) => value.length / 2); + const dynamicFieldByteLengths = dynamicDataItems.map((value) => value.length / 2).reverse(); const dynamicTotalByteLength = dynamicFieldByteLengths.reduce((total, length) => total + BigInt(length), 0n); const dynamicData = dynamicDataItems.join(""); - const packedCounter = `${encodeField("uint56", dynamicTotalByteLength).replace(/^0x/, "")}${dynamicFieldByteLengths + const packedCounter = `${dynamicFieldByteLengths .map((length) => encodeField("uint40", length).replace(/^0x/, "")) - .join("")}`.padEnd(64, "0"); + .join("")}${encodeField("uint56", dynamicTotalByteLength).replace(/^0x/, "")}`.padStart(64, "0"); return `0x${staticData}${packedCounter}${dynamicData}`; } diff --git a/packages/protocol-parser/src/encodeValue.ts b/packages/protocol-parser/src/encodeValue.ts new file mode 100644 index 0000000000..6394f144b3 --- /dev/null +++ b/packages/protocol-parser/src/encodeValue.ts @@ -0,0 +1,18 @@ +import { isStaticAbiType, isDynamicAbiType } from "@latticexyz/schema-type"; +import { Hex } from "viem"; +import { SchemaToPrimitives, ValueSchema } from "./common"; +import { encodeRecord } from "./encodeRecord"; + +export function encodeValue( + valueSchema: TSchema, + value: SchemaToPrimitives +): Hex { + const staticFields = Object.values(valueSchema).filter(isStaticAbiType); + const dynamicFields = Object.values(valueSchema).filter(isDynamicAbiType); + + // TODO: refactor and move all encodeRecord logic into this method so we can delete encodeRecord + + // This currently assumes fields/values are ordered by static, dynamic + // TODO: make sure we preserve ordering based on schema definition + return encodeRecord({ staticFields, dynamicFields }, Object.values(value)); +} diff --git a/packages/protocol-parser/src/hexToPackedCounter.ts b/packages/protocol-parser/src/hexToPackedCounter.ts index 4a244e0cf5..eec1382cf8 100644 --- a/packages/protocol-parser/src/hexToPackedCounter.ts +++ b/packages/protocol-parser/src/hexToPackedCounter.ts @@ -1,7 +1,8 @@ -import { Hex, sliceHex } from "viem"; +import { Hex } from "viem"; import { decodeStaticField } from "./decodeStaticField"; import { decodeDynamicField } from "./decodeDynamicField"; import { InvalidHexLengthForPackedCounterError, PackedCounterLengthMismatchError } from "./errors"; +import { padSliceHex } from "./padSliceHex"; // Keep this logic in sync with PackedCounter.sol @@ -18,9 +19,9 @@ export function hexToPackedCounter(data: Hex): { throw new InvalidHexLengthForPackedCounterError(data); } - const totalByteLength = decodeStaticField("uint56", sliceHex(data, 32 - 7, 32)); + const totalByteLength = decodeStaticField("uint56", padSliceHex(data, 32 - 7, 32)); // TODO: use schema to make sure we only parse as many as we need (rather than zeroes at the end)? - const reversedFieldByteLengths = decodeDynamicField("uint40[]", sliceHex(data, 0, 32 - 7)); + const reversedFieldByteLengths = decodeDynamicField("uint40[]", padSliceHex(data, 0, 32 - 7)); // Reverse the lengths const fieldByteLengths = Object.freeze([...reversedFieldByteLengths].reverse()); diff --git a/packages/protocol-parser/src/index.ts b/packages/protocol-parser/src/index.ts index bffae37cd2..e88579ce82 100644 --- a/packages/protocol-parser/src/index.ts +++ b/packages/protocol-parser/src/index.ts @@ -2,16 +2,23 @@ export * from "./abiTypesToSchema"; export * from "./common"; export * from "./decodeDynamicField"; export * from "./decodeField"; +export * from "./decodeKey"; export * from "./decodeKeyTuple"; export * from "./decodeRecord"; export * from "./decodeStaticField"; +export * from "./decodeValue"; export * from "./encodeField"; +export * from "./encodeKey"; export * from "./encodeKeyTuple"; export * from "./encodeRecord"; +export * from "./encodeValue"; export * from "./errors"; export * from "./hexToPackedCounter"; export * from "./hexToSchema"; export * from "./hexToTableSchema"; +export * from "./keySchemaToHex"; +export * from "./padSliceHex"; export * from "./schemaIndexToAbiType"; export * from "./schemaToHex"; export * from "./staticDataLength"; +export * from "./valueSchemaToHex"; diff --git a/packages/protocol-parser/src/keySchemaToHex.ts b/packages/protocol-parser/src/keySchemaToHex.ts new file mode 100644 index 0000000000..0d5d3a697e --- /dev/null +++ b/packages/protocol-parser/src/keySchemaToHex.ts @@ -0,0 +1,8 @@ +import { isStaticAbiType } from "@latticexyz/schema-type"; +import { Hex } from "viem"; +import { KeySchema } from "./common"; +import { schemaToHex } from "./schemaToHex"; + +export function keySchemaToHex(schema: KeySchema): Hex { + return schemaToHex({ staticFields: Object.values(schema).filter(isStaticAbiType), dynamicFields: [] }); +} diff --git a/packages/protocol-parser/src/padSliceHex.test.ts b/packages/protocol-parser/src/padSliceHex.test.ts new file mode 100644 index 0000000000..5f100c628b --- /dev/null +++ b/packages/protocol-parser/src/padSliceHex.test.ts @@ -0,0 +1,15 @@ +import { describe, expect, it } from "vitest"; +import { padSliceHex } from "./padSliceHex"; + +describe("padSliceHex", () => { + it("can slice empty hex", () => { + expect(padSliceHex("0x", 6)).toBe("0x"); + expect(padSliceHex("0x", 6, 10)).toBe("0x00000000"); + }); + it("can slice hex out of bounds", () => { + expect(padSliceHex("0x000100", 1)).toBe("0x0100"); + expect(padSliceHex("0x000100", 1, 4)).toBe("0x010000"); + expect(padSliceHex("0x000100", 3)).toBe("0x"); + expect(padSliceHex("0x000100", 3, 4)).toBe("0x00"); + }); +}); diff --git a/packages/protocol-parser/src/padSliceHex.ts b/packages/protocol-parser/src/padSliceHex.ts new file mode 100644 index 0000000000..734459e66a --- /dev/null +++ b/packages/protocol-parser/src/padSliceHex.ts @@ -0,0 +1,8 @@ +import { Hex } from "viem"; + +export function padSliceHex(data: Hex, start: number, end?: number): Hex { + return `0x${data + .replace(/^0x/, "") + .slice(start * 2, end != null ? end * 2 : undefined) + .padEnd(((end ?? start) - start) * 2, "0")}`; +} diff --git a/packages/protocol-parser/src/schemaToHex.ts b/packages/protocol-parser/src/schemaToHex.ts index c153539c61..439994d17c 100644 --- a/packages/protocol-parser/src/schemaToHex.ts +++ b/packages/protocol-parser/src/schemaToHex.ts @@ -3,6 +3,7 @@ import { Hex } from "viem"; import { Schema } from "./common"; import { staticDataLength } from "./staticDataLength"; +/** @deprecated use `keySchemaToHex` or `valueSchemaToHex` instead */ export function schemaToHex(schema: Schema): Hex { const staticSchemaTypes = schema.staticFields.map((abiType) => schemaAbiTypes.indexOf(abiType)); const dynamicSchemaTypes = schema.dynamicFields.map((abiType) => schemaAbiTypes.indexOf(abiType)); diff --git a/packages/protocol-parser/src/valueSchemaToHex.ts b/packages/protocol-parser/src/valueSchemaToHex.ts new file mode 100644 index 0000000000..b7e24b7f7a --- /dev/null +++ b/packages/protocol-parser/src/valueSchemaToHex.ts @@ -0,0 +1,11 @@ +import { isDynamicAbiType, isStaticAbiType } from "@latticexyz/schema-type"; +import { Hex } from "viem"; +import { ValueSchema } from "./common"; +import { schemaToHex } from "./schemaToHex"; + +export function valueSchemaToHex(schema: ValueSchema): Hex { + return schemaToHex({ + staticFields: Object.values(schema).filter(isStaticAbiType), + dynamicFields: Object.values(schema).filter(isDynamicAbiType), + }); +} From 66d92fd60d849c7179329b7af08370ec82ab9395 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Mon, 11 Sep 2023 10:39:24 +0100 Subject: [PATCH 2/6] remove dupe types, update imports --- packages/store-sync/src/common.ts | 6 ++--- .../src/postgres/buildInternalTables.ts | 5 ++-- .../store-sync/src/postgres/buildTable.ts | 25 ++++++------------- packages/store-sync/src/recs/common.ts | 3 ++- packages/store-sync/src/recs/decodeEntity.ts | 5 ++-- packages/store-sync/src/recs/encodeEntity.ts | 5 ++-- packages/store-sync/src/schemaToDefaults.ts | 2 +- .../src/sqlite/createSqliteTable.ts | 25 ++++++------------- .../store-sync/src/sqlite/internalTables.ts | 6 ++--- packages/store/ts/common.ts | 15 +---------- 10 files changed, 32 insertions(+), 65 deletions(-) diff --git a/packages/store-sync/src/common.ts b/packages/store-sync/src/common.ts index 6feccfe802..d903b05ba1 100644 --- a/packages/store-sync/src/common.ts +++ b/packages/store-sync/src/common.ts @@ -1,17 +1,15 @@ -import { Address, Block, Hex, Log, PublicClient, TransactionReceipt } from "viem"; +import { Address, Block, Hex, Log, PublicClient } from "viem"; import { GroupLogsByBlockNumberResult } from "@latticexyz/block-logs-stream"; import { StoreConfig, - KeySchema, - ValueSchema, ConfigToKeyPrimitives as Key, ConfigToValuePrimitives as Value, - TableRecord, StoreEventsAbiItem, StoreEventsAbi, } from "@latticexyz/store"; import { Observable } from "rxjs"; import { BlockStorageOperations } from "./blockLogsToStorage"; +import { KeySchema, ValueSchema, TableRecord } from "@latticexyz/protocol-parser"; export type ChainId = number; export type WorldId = `${ChainId}:${Address}`; diff --git a/packages/store-sync/src/postgres/buildInternalTables.ts b/packages/store-sync/src/postgres/buildInternalTables.ts index c269a9f9d0..889566db44 100644 --- a/packages/store-sync/src/postgres/buildInternalTables.ts +++ b/packages/store-sync/src/postgres/buildInternalTables.ts @@ -2,6 +2,7 @@ import { integer, pgSchema, text } from "drizzle-orm/pg-core"; import { DynamicAbiType, StaticAbiType } from "@latticexyz/schema-type"; import { transformSchemaName } from "./transformSchemaName"; import { asAddress, asBigInt, asJson, asNumber } from "./columnTypes"; +import { KeySchema, ValueSchema } from "@latticexyz/protocol-parser"; // eslint-disable-next-line @typescript-eslint/explicit-function-return-type export function buildInternalTables() { @@ -22,8 +23,8 @@ export function buildInternalTables() { tableId: text("table_id").notNull(), namespace: text("namespace").notNull(), name: text("name").notNull(), - keySchema: asJson>("key_schema").notNull(), - valueSchema: asJson>("value_schema").notNull(), + keySchema: asJson("key_schema").notNull(), + valueSchema: asJson("value_schema").notNull(), lastUpdatedBlockNumber: asBigInt("last_updated_block_number", "numeric"), // TODO: last block hash? lastError: text("last_error"), diff --git a/packages/store-sync/src/postgres/buildTable.ts b/packages/store-sync/src/postgres/buildTable.ts index 8974900e2a..0a9e24cd42 100644 --- a/packages/store-sync/src/postgres/buildTable.ts +++ b/packages/store-sync/src/postgres/buildTable.ts @@ -1,8 +1,8 @@ import { AnyPgColumnBuilder, PgTableWithColumns, pgSchema } from "drizzle-orm/pg-core"; -import { SchemaAbiType, StaticAbiType } from "@latticexyz/schema-type"; import { buildColumn } from "./buildColumn"; import { Address, getAddress } from "viem"; import { transformSchemaName } from "./transformSchemaName"; +import { KeySchema, ValueSchema } from "@latticexyz/protocol-parser"; // TODO: convert camel case to snake case for DB storage? export const metaColumns = { @@ -12,10 +12,7 @@ export const metaColumns = { __isDeleted: buildColumn("__isDeleted", "bool").notNull(), } as const satisfies Record; -type PgTableFromSchema< - TKeySchema extends Record, - TValueSchema extends Record -> = PgTableWithColumns<{ +type PgTableFromSchema = PgTableWithColumns<{ name: string; schema: string; columns: { @@ -30,10 +27,7 @@ type PgTableFromSchema< }; }>; -type BuildTableOptions< - TKeySchema extends Record, - TValueSchema extends Record -> = { +type BuildTableOptions = { address: Address; namespace: string; name: string; @@ -41,15 +35,12 @@ type BuildTableOptions< valueSchema: TValueSchema; }; -type BuildTableResult< - TKeySchema extends Record, - TValueSchema extends Record -> = PgTableFromSchema; +type BuildTableResult = PgTableFromSchema< + TKeySchema, + TValueSchema +>; -export function buildTable< - TKeySchema extends Record, - TValueSchema extends Record ->({ +export function buildTable({ address, namespace, name, diff --git a/packages/store-sync/src/recs/common.ts b/packages/store-sync/src/recs/common.ts index 79292ac176..c384d5da13 100644 --- a/packages/store-sync/src/recs/common.ts +++ b/packages/store-sync/src/recs/common.ts @@ -1,7 +1,8 @@ -import { KeySchema, StoreConfig, ValueSchema } from "@latticexyz/store"; +import { StoreConfig } from "@latticexyz/store"; import { Component as RecsComponent, Metadata as RecsMetadata, Type as RecsType } from "@latticexyz/recs"; import { SchemaAbiTypeToRecsType } from "./schemaAbiTypeToRecsType"; import { SchemaAbiType } from "@latticexyz/schema-type"; +import { KeySchema, ValueSchema } from "@latticexyz/protocol-parser"; export type StoreComponentMetadata = RecsMetadata & { componentName: string; diff --git a/packages/store-sync/src/recs/decodeEntity.ts b/packages/store-sync/src/recs/decodeEntity.ts index cadfad5df3..90167726f0 100644 --- a/packages/store-sync/src/recs/decodeEntity.ts +++ b/packages/store-sync/src/recs/decodeEntity.ts @@ -1,10 +1,9 @@ import { Entity } from "@latticexyz/recs"; -import { StaticAbiType } from "@latticexyz/schema-type"; import { Hex, decodeAbiParameters } from "viem"; -import { SchemaToPrimitives } from "@latticexyz/store"; +import { KeySchema, SchemaToPrimitives } from "@latticexyz/protocol-parser"; import { entityToHexKeyTuple } from "./entityToHexKeyTuple"; -export function decodeEntity>( +export function decodeEntity( keySchema: TKeySchema, entity: Entity ): SchemaToPrimitives { diff --git a/packages/store-sync/src/recs/encodeEntity.ts b/packages/store-sync/src/recs/encodeEntity.ts index 92f4c839ef..e0e155371c 100644 --- a/packages/store-sync/src/recs/encodeEntity.ts +++ b/packages/store-sync/src/recs/encodeEntity.ts @@ -1,10 +1,9 @@ import { Entity } from "@latticexyz/recs"; -import { StaticAbiType } from "@latticexyz/schema-type"; import { encodeAbiParameters } from "viem"; -import { SchemaToPrimitives } from "@latticexyz/store"; +import { KeySchema, SchemaToPrimitives } from "@latticexyz/protocol-parser"; import { hexKeyTupleToEntity } from "./hexKeyTupleToEntity"; -export function encodeEntity>( +export function encodeEntity( keySchema: TKeySchema, key: SchemaToPrimitives ): Entity { diff --git a/packages/store-sync/src/schemaToDefaults.ts b/packages/store-sync/src/schemaToDefaults.ts index e964dc224a..36dab2b1ec 100644 --- a/packages/store-sync/src/schemaToDefaults.ts +++ b/packages/store-sync/src/schemaToDefaults.ts @@ -1,5 +1,5 @@ import { schemaAbiTypeToDefaultValue } from "@latticexyz/schema-type"; -import { ValueSchema, SchemaToPrimitives } from "@latticexyz/store"; +import { ValueSchema, SchemaToPrimitives } from "@latticexyz/protocol-parser"; export function schemaToDefaults(schema: TSchema): SchemaToPrimitives { return Object.fromEntries( diff --git a/packages/store-sync/src/sqlite/createSqliteTable.ts b/packages/store-sync/src/sqlite/createSqliteTable.ts index 79f238e9ed..e8e8667b7d 100644 --- a/packages/store-sync/src/sqlite/createSqliteTable.ts +++ b/packages/store-sync/src/sqlite/createSqliteTable.ts @@ -1,8 +1,8 @@ import { AnySQLiteColumnBuilder, SQLiteTableWithColumns, sqliteTable } from "drizzle-orm/sqlite-core"; -import { SchemaAbiType, StaticAbiType } from "@latticexyz/schema-type"; import { buildSqliteColumn } from "./buildSqliteColumn"; import { Address } from "viem"; import { getTableName } from "./getTableName"; +import { KeySchema, ValueSchema } from "@latticexyz/protocol-parser"; export const metaColumns = { __key: buildSqliteColumn("__key", "bytes").notNull().primaryKey(), @@ -11,10 +11,7 @@ export const metaColumns = { __isDeleted: buildSqliteColumn("__isDeleted", "bool").notNull(), } as const satisfies Record; -type SQLiteTableFromSchema< - TKeySchema extends Record, - TValueSchema extends Record -> = SQLiteTableWithColumns<{ +type SQLiteTableFromSchema = SQLiteTableWithColumns<{ name: string; schema: string | undefined; columns: { @@ -29,10 +26,7 @@ type SQLiteTableFromSchema< }; }>; -type CreateSqliteTableOptions< - TKeySchema extends Record, - TValueSchema extends Record -> = { +type CreateSqliteTableOptions = { address: Address; namespace: string; name: string; @@ -40,15 +34,12 @@ type CreateSqliteTableOptions< valueSchema: TValueSchema; }; -type CreateSqliteTableResult< - TKeySchema extends Record, - TValueSchema extends Record -> = SQLiteTableFromSchema; +type CreateSqliteTableResult = SQLiteTableFromSchema< + TKeySchema, + TValueSchema +>; -export function createSqliteTable< - TKeySchema extends Record, - TValueSchema extends Record ->({ +export function createSqliteTable({ address, namespace, name, diff --git a/packages/store-sync/src/sqlite/internalTables.ts b/packages/store-sync/src/sqlite/internalTables.ts index 057336839a..d2774a48e0 100644 --- a/packages/store-sync/src/sqlite/internalTables.ts +++ b/packages/store-sync/src/sqlite/internalTables.ts @@ -1,6 +1,6 @@ import { blob, integer, sqliteTable, text } from "drizzle-orm/sqlite-core"; import { address, json } from "./columnTypes"; -import { DynamicAbiType, StaticAbiType } from "@latticexyz/schema-type"; +import { KeySchema, ValueSchema } from "@latticexyz/protocol-parser"; export const chainState = sqliteTable("__chainState", { schemaVersion: integer("schema_version").notNull().primaryKey(), @@ -17,8 +17,8 @@ export const mudStoreTables = sqliteTable("__mudStoreTables", { tableId: text("table_id").notNull(), namespace: text("namespace").notNull(), name: text("name").notNull(), - keySchema: json>("key_schema").notNull(), - valueSchema: json>("value_schema").notNull(), + keySchema: json("key_schema").notNull(), + valueSchema: json("value_schema").notNull(), lastUpdatedBlockNumber: blob("last_updated_block_number", { mode: "bigint" }), // TODO: last block hash? lastError: text("last_error"), diff --git a/packages/store/ts/common.ts b/packages/store/ts/common.ts index 9d825f35fd..380709deec 100644 --- a/packages/store/ts/common.ts +++ b/packages/store/ts/common.ts @@ -1,19 +1,6 @@ -import { SchemaAbiType, SchemaAbiTypeToPrimitiveType, StaticAbiType } from "@latticexyz/schema-type"; +import { SchemaAbiType, SchemaAbiTypeToPrimitiveType } from "@latticexyz/schema-type"; import { FieldData, FullSchemaConfig, StoreConfig } from "./config"; -export type KeySchema = Record; -export type ValueSchema = Record; - -/** Map a table schema like `{ value: "uint256" }` to its primitive types like `{ value: bigint }` */ -export type SchemaToPrimitives = { - [key in keyof TSchema]: SchemaAbiTypeToPrimitiveType; -}; - -export type TableRecord = { - key: SchemaToPrimitives; - value: SchemaToPrimitives; -}; - export type ConfigFieldTypeToSchemaAbiType> = T extends SchemaAbiType ? T : T extends `${string}[${string}]` From 6d3a4b35877e596ec209015436b5dcba1138a09f Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Mon, 11 Sep 2023 03:57:16 -0700 Subject: [PATCH 3/6] Update packages/protocol-parser/src/encodeKey.ts Co-authored-by: alvarius --- packages/protocol-parser/src/encodeKey.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/protocol-parser/src/encodeKey.ts b/packages/protocol-parser/src/encodeKey.ts index f98ac04210..f520536eae 100644 --- a/packages/protocol-parser/src/encodeKey.ts +++ b/packages/protocol-parser/src/encodeKey.ts @@ -5,6 +5,6 @@ import { encodeKeyTuple } from "./encodeKeyTuple"; export function encodeKey(keySchema: TSchema, key: SchemaToPrimitives): Hex[] { const staticFields = Object.values(keySchema).filter(isStaticAbiType); - // TODO: refactor and move all encodeKey logic into this method so we can delete encodeKey + // TODO: refactor and move all encodeKeyTuple logic into this method so we can delete encodeKeyTuple return encodeKeyTuple({ staticFields, dynamicFields: [] }, Object.values(key)); } From 075a8ecbef6becec3700f876abec42f83f637ee0 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Mon, 11 Sep 2023 14:57:59 +0100 Subject: [PATCH 4/6] add typedoc and rename for clarity --- packages/protocol-parser/src/decodeRecord.ts | 8 ++++---- .../protocol-parser/src/hexToPackedCounter.ts | 6 +++--- packages/protocol-parser/src/index.ts | 2 +- packages/protocol-parser/src/padSliceHex.test.ts | 15 --------------- packages/protocol-parser/src/padSliceHex.ts | 8 -------- packages/protocol-parser/src/readHex.test.ts | 15 +++++++++++++++ packages/protocol-parser/src/readHex.ts | 15 +++++++++++++++ 7 files changed, 38 insertions(+), 31 deletions(-) delete mode 100644 packages/protocol-parser/src/padSliceHex.test.ts delete mode 100644 packages/protocol-parser/src/padSliceHex.ts create mode 100644 packages/protocol-parser/src/readHex.test.ts create mode 100644 packages/protocol-parser/src/readHex.ts diff --git a/packages/protocol-parser/src/decodeRecord.ts b/packages/protocol-parser/src/decodeRecord.ts index ce355acff8..2f8536287d 100644 --- a/packages/protocol-parser/src/decodeRecord.ts +++ b/packages/protocol-parser/src/decodeRecord.ts @@ -10,7 +10,7 @@ import { decodeDynamicField } from "./decodeDynamicField"; import { decodeStaticField } from "./decodeStaticField"; import { hexToPackedCounter } from "./hexToPackedCounter"; import { staticDataLength } from "./staticDataLength"; -import { padSliceHex } from "./padSliceHex"; +import { readHex } from "./readHex"; /** @deprecated use `decodeValue` instead */ export function decodeRecord(schema: Schema, data: Hex): readonly (StaticPrimitiveType | DynamicPrimitiveType)[] { @@ -19,7 +19,7 @@ export function decodeRecord(schema: Schema, data: Hex): readonly (StaticPrimiti let bytesOffset = 0; schema.staticFields.forEach((fieldType) => { const fieldByteLength = staticAbiTypeToByteLength[fieldType]; - const value = decodeStaticField(fieldType, padSliceHex(data, bytesOffset, bytesOffset + fieldByteLength)); + const value = decodeStaticField(fieldType, readHex(data, bytesOffset, bytesOffset + fieldByteLength)); bytesOffset += fieldByteLength; values.push(value); }); @@ -39,13 +39,13 @@ export function decodeRecord(schema: Schema, data: Hex): readonly (StaticPrimiti } if (schema.dynamicFields.length > 0) { - const dataLayout = hexToPackedCounter(padSliceHex(data, bytesOffset, bytesOffset + 32)); + const dataLayout = hexToPackedCounter(readHex(data, bytesOffset, bytesOffset + 32)); bytesOffset += 32; schema.dynamicFields.forEach((fieldType, i) => { const dataLength = dataLayout.fieldByteLengths[i]; if (dataLength > 0) { - const value = decodeDynamicField(fieldType, padSliceHex(data, bytesOffset, bytesOffset + dataLength)); + const value = decodeDynamicField(fieldType, readHex(data, bytesOffset, bytesOffset + dataLength)); bytesOffset += dataLength; values.push(value); } else { diff --git a/packages/protocol-parser/src/hexToPackedCounter.ts b/packages/protocol-parser/src/hexToPackedCounter.ts index eec1382cf8..6ab5920bbd 100644 --- a/packages/protocol-parser/src/hexToPackedCounter.ts +++ b/packages/protocol-parser/src/hexToPackedCounter.ts @@ -2,7 +2,7 @@ import { Hex } from "viem"; import { decodeStaticField } from "./decodeStaticField"; import { decodeDynamicField } from "./decodeDynamicField"; import { InvalidHexLengthForPackedCounterError, PackedCounterLengthMismatchError } from "./errors"; -import { padSliceHex } from "./padSliceHex"; +import { readHex } from "./readHex"; // Keep this logic in sync with PackedCounter.sol @@ -19,9 +19,9 @@ export function hexToPackedCounter(data: Hex): { throw new InvalidHexLengthForPackedCounterError(data); } - const totalByteLength = decodeStaticField("uint56", padSliceHex(data, 32 - 7, 32)); + const totalByteLength = decodeStaticField("uint56", readHex(data, 32 - 7, 32)); // TODO: use schema to make sure we only parse as many as we need (rather than zeroes at the end)? - const reversedFieldByteLengths = decodeDynamicField("uint40[]", padSliceHex(data, 0, 32 - 7)); + const reversedFieldByteLengths = decodeDynamicField("uint40[]", readHex(data, 0, 32 - 7)); // Reverse the lengths const fieldByteLengths = Object.freeze([...reversedFieldByteLengths].reverse()); diff --git a/packages/protocol-parser/src/index.ts b/packages/protocol-parser/src/index.ts index e88579ce82..c501303621 100644 --- a/packages/protocol-parser/src/index.ts +++ b/packages/protocol-parser/src/index.ts @@ -17,7 +17,7 @@ export * from "./hexToPackedCounter"; export * from "./hexToSchema"; export * from "./hexToTableSchema"; export * from "./keySchemaToHex"; -export * from "./padSliceHex"; +export * from "./readHex"; export * from "./schemaIndexToAbiType"; export * from "./schemaToHex"; export * from "./staticDataLength"; diff --git a/packages/protocol-parser/src/padSliceHex.test.ts b/packages/protocol-parser/src/padSliceHex.test.ts deleted file mode 100644 index 5f100c628b..0000000000 --- a/packages/protocol-parser/src/padSliceHex.test.ts +++ /dev/null @@ -1,15 +0,0 @@ -import { describe, expect, it } from "vitest"; -import { padSliceHex } from "./padSliceHex"; - -describe("padSliceHex", () => { - it("can slice empty hex", () => { - expect(padSliceHex("0x", 6)).toBe("0x"); - expect(padSliceHex("0x", 6, 10)).toBe("0x00000000"); - }); - it("can slice hex out of bounds", () => { - expect(padSliceHex("0x000100", 1)).toBe("0x0100"); - expect(padSliceHex("0x000100", 1, 4)).toBe("0x010000"); - expect(padSliceHex("0x000100", 3)).toBe("0x"); - expect(padSliceHex("0x000100", 3, 4)).toBe("0x00"); - }); -}); diff --git a/packages/protocol-parser/src/padSliceHex.ts b/packages/protocol-parser/src/padSliceHex.ts deleted file mode 100644 index 734459e66a..0000000000 --- a/packages/protocol-parser/src/padSliceHex.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { Hex } from "viem"; - -export function padSliceHex(data: Hex, start: number, end?: number): Hex { - return `0x${data - .replace(/^0x/, "") - .slice(start * 2, end != null ? end * 2 : undefined) - .padEnd(((end ?? start) - start) * 2, "0")}`; -} diff --git a/packages/protocol-parser/src/readHex.test.ts b/packages/protocol-parser/src/readHex.test.ts new file mode 100644 index 0000000000..d9cb7f88ad --- /dev/null +++ b/packages/protocol-parser/src/readHex.test.ts @@ -0,0 +1,15 @@ +import { describe, expect, it } from "vitest"; +import { readHex } from "./readHex"; + +describe("readHex", () => { + it("can slice empty hex", () => { + expect(readHex("0x", 6)).toBe("0x"); + expect(readHex("0x", 6, 10)).toBe("0x00000000"); + }); + it("can slice hex out of bounds", () => { + expect(readHex("0x000100", 1)).toBe("0x0100"); + expect(readHex("0x000100", 1, 4)).toBe("0x010000"); + expect(readHex("0x000100", 3)).toBe("0x"); + expect(readHex("0x000100", 3, 4)).toBe("0x00"); + }); +}); diff --git a/packages/protocol-parser/src/readHex.ts b/packages/protocol-parser/src/readHex.ts new file mode 100644 index 0000000000..cf7661802f --- /dev/null +++ b/packages/protocol-parser/src/readHex.ts @@ -0,0 +1,15 @@ +import { Hex } from "viem"; + +/** + * Get the hex value at start/end positions. This will always return a valid hex string. + * + * If `start` is out of range, this returns `"0x"`. + * + * If `end` is specified and out of range, the result is right zero-padded to the desired length (`end - start`). + */ +export function readHex(data: Hex, start: number, end?: number): Hex { + return `0x${data + .replace(/^0x/, "") + .slice(start * 2, end != null ? end * 2 : undefined) + .padEnd(((end ?? start) - start) * 2, "0")}`; +} From d7b6ef08170685a47c9b7acb8bb29ec7e9cf800d Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 12 Sep 2023 01:54:45 -0700 Subject: [PATCH 5/6] Create tricky-comics-remain.md --- .changeset/tricky-comics-remain.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .changeset/tricky-comics-remain.md diff --git a/.changeset/tricky-comics-remain.md b/.changeset/tricky-comics-remain.md new file mode 100644 index 0000000000..99495e47a8 --- /dev/null +++ b/.changeset/tricky-comics-remain.md @@ -0,0 +1,5 @@ +--- +"@latticexyz/store": minor +--- + +Moved `KeySchema`, `ValueSchema`, `SchemaToPrimitives` and `TableRecord` types into `@latticexyz/protocol-parser` From f2be76b03ea787cf173b9d843d00cea0d5d190ae Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 12 Sep 2023 01:58:53 -0700 Subject: [PATCH 6/6] Create wicked-tigers-return.md --- .changeset/wicked-tigers-return.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .changeset/wicked-tigers-return.md diff --git a/.changeset/wicked-tigers-return.md b/.changeset/wicked-tigers-return.md new file mode 100644 index 0000000000..0a191f7399 --- /dev/null +++ b/.changeset/wicked-tigers-return.md @@ -0,0 +1,5 @@ +--- +"@latticexyz/protocol-parser": minor +--- + +Adds `decodeKey`, `decodeValue`, `encodeKey`, and `encodeValue` helpers to decode/encode from key/value schemas. Deprecates previous methods that use a schema object with static/dynamic field arrays, originally attempting to model our on-chain behavior but ended up not very ergonomic when working with table configs.