From 8bbb3f4461962f57f1bfdbaff664f03bb987346b Mon Sep 17 00:00:00 2001 From: Gusarich Date: Thu, 11 Jul 2024 19:22:11 +0300 Subject: [PATCH 1/5] check getter name collisions in wrappers --- CHANGELOG.md | 1 + src/bindings/writeTypescript.ts | 10 +++++++++- .../contract-duplicate-getter-pascalcase.spec.ts | 14 ++++++++++++++ .../contract-duplicate-getter-pascalcase.tact | 12 ++++++++++++ src/test/compilation-failed/tact.config.json | 5 +++++ 5 files changed, 41 insertions(+), 1 deletion(-) create mode 100644 src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts create mode 100644 src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact diff --git a/CHANGELOG.md b/CHANGELOG.md index 4e8822ff0..2be3061f2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -37,6 +37,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Usage of `initOf` inside of `init()` does not cause error `135` anymore: PR [#521](https://github.com/tact-lang/tact/issues/521) - Usage of `newAddress` with hash parts shorter than 64 hexadecimal digits does not cause constant evaluation error `Invalid address hash length` anymore: PR [#525](https://github.com/tact-lang/tact/pull/525) - Introduced a streamlined error logger for compilation pipeline to support third-party tools: PR [#509](https://github.com/tact-lang/tact/pull/509) +- Collisions of PascalCase getter names in generated wrappers are now checked: PR [#556](https://github.com/tact-lang/tact/pull/556) ## [1.4.0] - 2024-06-21 diff --git a/src/bindings/writeTypescript.ts b/src/bindings/writeTypescript.ts index 092d1b49f..f887ffb45 100644 --- a/src/bindings/writeTypescript.ts +++ b/src/bindings/writeTypescript.ts @@ -561,10 +561,18 @@ export function writeTypescript( } // Getters + const getterNames: Set = new Set(); if (abi.getters) { for (const g of abi.getters) { + const pascalCaseName = changeCase.pascalCase(g.name); + if (getterNames.has(pascalCaseName)) { + throw Error( + `Getter with name '${g.name}' already exists in pascal case form`, + ); + } + getterNames.add(pascalCaseName); w.append( - `async get${changeCase.pascalCase(g.name)}(${["provider: ContractProvider", ...writeArguments(g.arguments ? g.arguments : [])].join(", ")}) {`, + `async get${pascalCaseName}(${["provider: ContractProvider", ...writeArguments(g.arguments ? g.arguments : [])].join(", ")}) {`, ); w.inIndent(() => { w.append(`let builder = new TupleBuilder();`); diff --git a/src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts b/src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts new file mode 100644 index 000000000..8310febd4 --- /dev/null +++ b/src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts @@ -0,0 +1,14 @@ +import { __DANGER_resetNodeId } from "../../grammar/ast"; +import { itShouldNotCompile } from "./util"; + +describe("contract-duplicate-getter-pascalcase", () => { + beforeEach(() => { + __DANGER_resetNodeId(); + }); + + itShouldNotCompile({ + testName: "contract-duplicate-getter-pascalcase", + errorMessage: + "Bindings compiler crashed: Getter with name 'test_getter' already exists in pascal case form", + }); +}); diff --git a/src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact b/src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact new file mode 100644 index 000000000..1e54fe678 --- /dev/null +++ b/src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact @@ -0,0 +1,12 @@ +contract Test { + receive () {} + + get fun testGetter(): Int { + return 123; + } + + get fun test_getter(): Int { + return 123; + } +} + diff --git a/src/test/compilation-failed/tact.config.json b/src/test/compilation-failed/tact.config.json index dfc144919..822ba64d4 100644 --- a/src/test/compilation-failed/tact.config.json +++ b/src/test/compilation-failed/tact.config.json @@ -131,6 +131,11 @@ "name": "contract-duplicate-receiver-opcode", "path": "./contracts/contract-duplicate-receiver-opcode.tact", "output": "./contracts/output" + }, + { + "name": "contract-duplicate-getter-pascalcase", + "path": "./contracts/contract-duplicate-getter-pascalcase.tact", + "output": "./contracts/output" } ] } From 0a242a6205965bef198591058ea3b533f246beb4 Mon Sep 17 00:00:00 2001 From: Gusarich Date: Thu, 11 Jul 2024 19:23:25 +0300 Subject: [PATCH 2/5] add `pascalcase` to cspell ignore list --- cspell.json | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/cspell.json b/cspell.json index 6df46b2af..479f4c0b7 100644 --- a/cspell.json +++ b/cspell.json @@ -77,7 +77,8 @@ "привет", "PUSHREF", "PUSHSLICE", - "SETINDEXVARQ" + "SETINDEXVARQ", + "pascalcase" ], "flagWords": [], "ignorePaths": [ From 9f33339987fbb5b637df3d8d8b28abe1c6658f59 Mon Sep 17 00:00:00 2001 From: Gusarich Date: Thu, 11 Jul 2024 19:58:11 +0300 Subject: [PATCH 3/5] change behaviour to conflcit resolving instead of error throwing --- src/bindings/writeTypescript.ts | 12 +- ...ntract-duplicate-getter-pascalcase.spec.ts | 14 - src/test/compilation-failed/tact.config.json | 5 - .../getter-names-conflict.spec.ts.snap | 299 ++++++++++++++++++ .../contracts/getter-names-conflict.tact} | 8 +- .../getter-names-conflict.spec.ts | 23 ++ tact.config.json | 5 + 7 files changed, 338 insertions(+), 28 deletions(-) delete mode 100644 src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts create mode 100644 src/test/e2e-emulated/__snapshots__/getter-names-conflict.spec.ts.snap rename src/test/{compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact => e2e-emulated/contracts/getter-names-conflict.tact} (55%) create mode 100644 src/test/e2e-emulated/getter-names-conflict.spec.ts diff --git a/src/bindings/writeTypescript.ts b/src/bindings/writeTypescript.ts index f887ffb45..525203891 100644 --- a/src/bindings/writeTypescript.ts +++ b/src/bindings/writeTypescript.ts @@ -564,15 +564,13 @@ export function writeTypescript( const getterNames: Set = new Set(); if (abi.getters) { for (const g of abi.getters) { - const pascalCaseName = changeCase.pascalCase(g.name); - if (getterNames.has(pascalCaseName)) { - throw Error( - `Getter with name '${g.name}' already exists in pascal case form`, - ); + let getterName = changeCase.pascalCase(g.name); + if (getterNames.has(getterName)) { + getterName = g.name; } - getterNames.add(pascalCaseName); + getterNames.add(getterName); w.append( - `async get${pascalCaseName}(${["provider: ContractProvider", ...writeArguments(g.arguments ? g.arguments : [])].join(", ")}) {`, + `async get${getterName}(${["provider: ContractProvider", ...writeArguments(g.arguments ? g.arguments : [])].join(", ")}) {`, ); w.inIndent(() => { w.append(`let builder = new TupleBuilder();`); diff --git a/src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts b/src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts deleted file mode 100644 index 8310febd4..000000000 --- a/src/test/compilation-failed/contract-duplicate-getter-pascalcase.spec.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { __DANGER_resetNodeId } from "../../grammar/ast"; -import { itShouldNotCompile } from "./util"; - -describe("contract-duplicate-getter-pascalcase", () => { - beforeEach(() => { - __DANGER_resetNodeId(); - }); - - itShouldNotCompile({ - testName: "contract-duplicate-getter-pascalcase", - errorMessage: - "Bindings compiler crashed: Getter with name 'test_getter' already exists in pascal case form", - }); -}); diff --git a/src/test/compilation-failed/tact.config.json b/src/test/compilation-failed/tact.config.json index 822ba64d4..dfc144919 100644 --- a/src/test/compilation-failed/tact.config.json +++ b/src/test/compilation-failed/tact.config.json @@ -131,11 +131,6 @@ "name": "contract-duplicate-receiver-opcode", "path": "./contracts/contract-duplicate-receiver-opcode.tact", "output": "./contracts/output" - }, - { - "name": "contract-duplicate-getter-pascalcase", - "path": "./contracts/contract-duplicate-getter-pascalcase.tact", - "output": "./contracts/output" } ] } diff --git a/src/test/e2e-emulated/__snapshots__/getter-names-conflict.spec.ts.snap b/src/test/e2e-emulated/__snapshots__/getter-names-conflict.spec.ts.snap new file mode 100644 index 000000000..02f9615bf --- /dev/null +++ b/src/test/e2e-emulated/__snapshots__/getter-names-conflict.spec.ts.snap @@ -0,0 +1,299 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`getter-names-conflict should handle conflicts in getter names correctly 1`] = ` +Test { + "abi": { + "errors": { + "10": { + "message": "Dictionary error", + }, + "128": { + "message": "Null reference exception", + }, + "129": { + "message": "Invalid serialization prefix", + }, + "13": { + "message": "Out of gas error", + }, + "130": { + "message": "Invalid incoming message", + }, + "131": { + "message": "Constraints error", + }, + "132": { + "message": "Access denied", + }, + "133": { + "message": "Contract stopped", + }, + "134": { + "message": "Invalid argument", + }, + "135": { + "message": "Code of a contract was not found", + }, + "136": { + "message": "Invalid address", + }, + "137": { + "message": "Masterchain support is not enabled for this contract", + }, + "2": { + "message": "Stack underflow", + }, + "3": { + "message": "Stack overflow", + }, + "32": { + "message": "Method ID not found", + }, + "34": { + "message": "Action is invalid or not supported", + }, + "37": { + "message": "Not enough TON", + }, + "38": { + "message": "Not enough extra-currencies", + }, + "4": { + "message": "Integer overflow", + }, + "5": { + "message": "Integer out of expected range", + }, + "6": { + "message": "Invalid opcode", + }, + "7": { + "message": "Type check error", + }, + "8": { + "message": "Cell overflow", + }, + "9": { + "message": "Cell underflow", + }, + }, + "getters": [ + { + "arguments": [], + "name": "testGetter", + "returnType": { + "format": 257, + "kind": "simple", + "optional": false, + "type": "int", + }, + }, + { + "arguments": [], + "name": "test_getter", + "returnType": { + "format": 257, + "kind": "simple", + "optional": false, + "type": "int", + }, + }, + { + "arguments": [], + "name": "Test_getter", + "returnType": { + "format": 257, + "kind": "simple", + "optional": false, + "type": "int", + }, + }, + ], + "receivers": [ + { + "message": { + "kind": "empty", + }, + "receiver": "internal", + }, + ], + "types": [ + { + "fields": [ + { + "name": "code", + "type": { + "kind": "simple", + "optional": false, + "type": "cell", + }, + }, + { + "name": "data", + "type": { + "kind": "simple", + "optional": false, + "type": "cell", + }, + }, + ], + "header": null, + "name": "StateInit", + }, + { + "fields": [ + { + "name": "bounced", + "type": { + "kind": "simple", + "optional": false, + "type": "bool", + }, + }, + { + "name": "sender", + "type": { + "kind": "simple", + "optional": false, + "type": "address", + }, + }, + { + "name": "value", + "type": { + "format": 257, + "kind": "simple", + "optional": false, + "type": "int", + }, + }, + { + "name": "raw", + "type": { + "kind": "simple", + "optional": false, + "type": "slice", + }, + }, + ], + "header": null, + "name": "Context", + }, + { + "fields": [ + { + "name": "bounce", + "type": { + "kind": "simple", + "optional": false, + "type": "bool", + }, + }, + { + "name": "to", + "type": { + "kind": "simple", + "optional": false, + "type": "address", + }, + }, + { + "name": "value", + "type": { + "format": 257, + "kind": "simple", + "optional": false, + "type": "int", + }, + }, + { + "name": "mode", + "type": { + "format": 257, + "kind": "simple", + "optional": false, + "type": "int", + }, + }, + { + "name": "body", + "type": { + "kind": "simple", + "optional": true, + "type": "cell", + }, + }, + { + "name": "code", + "type": { + "kind": "simple", + "optional": true, + "type": "cell", + }, + }, + { + "name": "data", + "type": { + "kind": "simple", + "optional": true, + "type": "cell", + }, + }, + ], + "header": null, + "name": "SendParameters", + }, + ], + }, + "address": kQD80WcWxBhERIUXYb8hyvNv5hK-iToKR4Q5dEGUZ2NWl9Mx, + "init": { + "code": x{FF00F4A413F4BCF2C80B} + x{62_} + x{D001D0D3030171B0A301FA400120D74981010BBAF2E08820D70B0A208104FFBAF2D0898309BAF2E088545053036F04F86102F862DB3C59DB3CF2E08230C8F84301CC7F01CA00C9ED54} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{0192307FE07021D749C21F953020D70B1FDEC00001D749C121B0917FE070} + x{2_} + x{2_} + x{B9BDCDB3CDB3C31} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{73} + x{BBE1FDB3CDB3C31} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{71} + x{2_} + x{BB1A6DB3CDB3C31} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{72} + x{B82BEED44D0D20001}, + "data": x{4_} + x{C_} + x{A15891_} + x{FF00F4A413F4BCF2C80B} + x{62_} + x{D001D0D3030171B0A301FA400120D74981010BBAF2E08820D70B0A208104FFBAF2D0898309BAF2E088545053036F04F86102F862DB3C59DB3CF2E08230C8F84301CC7F01CA00C9ED54} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{0192307FE07021D749C21F953020D70B1FDEC00001D749C121B0917FE070} + x{2_} + x{2_} + x{B9BDCDB3CDB3C31} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{73} + x{BBE1FDB3CDB3C31} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{71} + x{2_} + x{BB1A6DB3CDB3C31} + x{ED44D0D401F863D20030916DE0F828D70B0A8309BAF2E089DB3C} + x{6D} + x{72} + x{B82BEED44D0D20001}, + }, +} +`; diff --git a/src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact b/src/test/e2e-emulated/contracts/getter-names-conflict.tact similarity index 55% rename from src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact rename to src/test/e2e-emulated/contracts/getter-names-conflict.tact index 1e54fe678..bcd694034 100644 --- a/src/test/compilation-failed/contracts/contract-duplicate-getter-pascalcase.tact +++ b/src/test/e2e-emulated/contracts/getter-names-conflict.tact @@ -2,11 +2,15 @@ contract Test { receive () {} get fun testGetter(): Int { - return 123; + return 1; } get fun test_getter(): Int { - return 123; + return 2; + } + + get fun Test_getter(): Int { + return 3; } } diff --git a/src/test/e2e-emulated/getter-names-conflict.spec.ts b/src/test/e2e-emulated/getter-names-conflict.spec.ts new file mode 100644 index 000000000..8b3d3ad42 --- /dev/null +++ b/src/test/e2e-emulated/getter-names-conflict.spec.ts @@ -0,0 +1,23 @@ +import { toNano } from "@ton/core"; +import { ContractSystem } from "@tact-lang/emulator"; +import { __DANGER_resetNodeId } from "../../grammar/ast"; +import { Test } from "./contracts/output/getter-names-conflict_Test"; + +describe("getter-names-conflict", () => { + beforeEach(() => { + __DANGER_resetNodeId(); + }); + it("should handle conflicts in getter names correctly", async () => { + // Init + const system = await ContractSystem.create(); + const treasure = system.treasure("treasure"); + const contract = system.open(await Test.fromInit()); + await contract.send(treasure, { value: toNano("10") }, null); + await system.run(); + expect(contract).toMatchSnapshot(); + + expect(await contract.getTestGetter()).toBe(1n); + expect(await contract.gettest_getter()).toBe(2n); + expect(await contract.getTest_getter()).toBe(3n); + }); +}); diff --git a/tact.config.json b/tact.config.json index 62d7a3055..205786a5c 100644 --- a/tact.config.json +++ b/tact.config.json @@ -365,6 +365,11 @@ "options": { "debug": true } + }, + { + "name": "getter-names-conflict", + "path": "./src/test/e2e-emulated/contracts/getter-names-conflict.tact", + "output": "./src/test/e2e-emulated/contracts/output" } ] } From 71a35cb078037277876f23e5bcaab623e3e27634 Mon Sep 17 00:00:00 2001 From: Gusarich Date: Thu, 11 Jul 2024 19:59:14 +0300 Subject: [PATCH 4/5] fix cspell --- cspell.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cspell.json b/cspell.json index 479f4c0b7..aeb676014 100644 --- a/cspell.json +++ b/cspell.json @@ -77,8 +77,7 @@ "привет", "PUSHREF", "PUSHSLICE", - "SETINDEXVARQ", - "pascalcase" + "SETINDEXVARQ" ], "flagWords": [], "ignorePaths": [ @@ -101,6 +100,7 @@ "src/test/e2e-emulated/contracts/intrinsics.tact", "src/test/e2e-emulated/contracts/strings.tact", "src/test/compilation-fail/fail-const-eval.spec.ts", + "src/test/e2e-emulated/getter-names-conflict.spec.ts", "stdlib/stdlib.fc" ] } From 9bc406d7a3179636daad04ecac18671d0c24bf83 Mon Sep 17 00:00:00 2001 From: Gusarich Date: Fri, 12 Jul 2024 14:31:46 +0300 Subject: [PATCH 5/5] add exported constant mapping of getter names --- src/bindings/writeTypescript.ts | 30 ++++++++++++++----- .../getter-names-conflict.spec.ts | 9 +++++- 2 files changed, 31 insertions(+), 8 deletions(-) diff --git a/src/bindings/writeTypescript.ts b/src/bindings/writeTypescript.ts index 525203891..1ff535a72 100644 --- a/src/bindings/writeTypescript.ts +++ b/src/bindings/writeTypescript.ts @@ -214,18 +214,40 @@ export function writeTypescript( w.append(`]`); w.append(); + const getterNames: Map = new Map(); + // Getters w.append(`const ${abi.name}_getters: ABIGetter[] = [`); w.inIndent(() => { if (abi.getters) { for (const t of abi.getters) { w.append(JSON.stringify(t) + ","); + + let getterName = changeCase.pascalCase(t.name); + if (Array.from(getterNames.values()).includes(getterName)) { + getterName = t.name; + } + getterNames.set(t.name, getterName); } } }); w.append(`]`); w.append(); + // Getter mapping + w.append( + `export const ${abi.name}_getterMapping: { [key: string]: string } = {`, + ); + w.inIndent(() => { + if (abi.getters) { + for (const t of abi.getters) { + w.append(`'${t.name}': 'get${getterNames.get(t.name)}',`); + } + } + }); + w.append(`}`); + w.append(); + // Receivers w.append(`const ${abi.name}_receivers: ABIReceiver[] = [`); w.inIndent(() => { @@ -561,16 +583,10 @@ export function writeTypescript( } // Getters - const getterNames: Set = new Set(); if (abi.getters) { for (const g of abi.getters) { - let getterName = changeCase.pascalCase(g.name); - if (getterNames.has(getterName)) { - getterName = g.name; - } - getterNames.add(getterName); w.append( - `async get${getterName}(${["provider: ContractProvider", ...writeArguments(g.arguments ? g.arguments : [])].join(", ")}) {`, + `async get${getterNames.get(g.name)}(${["provider: ContractProvider", ...writeArguments(g.arguments ? g.arguments : [])].join(", ")}) {`, ); w.inIndent(() => { w.append(`let builder = new TupleBuilder();`); diff --git a/src/test/e2e-emulated/getter-names-conflict.spec.ts b/src/test/e2e-emulated/getter-names-conflict.spec.ts index 8b3d3ad42..9eda79803 100644 --- a/src/test/e2e-emulated/getter-names-conflict.spec.ts +++ b/src/test/e2e-emulated/getter-names-conflict.spec.ts @@ -1,7 +1,10 @@ import { toNano } from "@ton/core"; import { ContractSystem } from "@tact-lang/emulator"; import { __DANGER_resetNodeId } from "../../grammar/ast"; -import { Test } from "./contracts/output/getter-names-conflict_Test"; +import { + Test, + Test_getterMapping, +} from "./contracts/output/getter-names-conflict_Test"; describe("getter-names-conflict", () => { beforeEach(() => { @@ -19,5 +22,9 @@ describe("getter-names-conflict", () => { expect(await contract.getTestGetter()).toBe(1n); expect(await contract.gettest_getter()).toBe(2n); expect(await contract.getTest_getter()).toBe(3n); + + expect(Test_getterMapping["testGetter"]).toBe("getTestGetter"); + expect(Test_getterMapping["test_getter"]).toBe("gettest_getter"); + expect(Test_getterMapping["Test_getter"]).toBe("getTest_getter"); }); });