From c542fcce430c994da13e5ca1d5be4cb1e99483a5 Mon Sep 17 00:00:00 2001 From: byakuren-hijiri <159621697+byakuren-hijiri@users.noreply.github.com> Date: Tue, 7 May 2024 03:29:54 -0300 Subject: [PATCH] feat(store): Refactor API to access AST store for third-party tools (#326) Refers to #314 * feat(store): Remove `openMockContext` API users can simply create an `ASTProgram` containing the desired top-level entries and use `createContext` instead. * feat(store): Change the default behavior of `openContext` --- CHANGELOG.md | 1 + src/grammar/store.ts | 94 ++++++++++++++++++++++---------------- src/pipeline/precompile.ts | 2 +- 3 files changed, 56 insertions(+), 41 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ce0f05157..cc9e12b77 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,6 +12,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changed - Refactor AST types to simplify access to third-party tools: PR [#325](https://github.com/tact-lang/tact/pull/325) +- Refactor the compiler API used to access AST store: PR [#326](https://github.com/tact-lang/tact/pull/326) ### Fixed diff --git a/src/grammar/store.ts b/src/grammar/store.ts index 1fd8b01d7..2ed610d8a 100644 --- a/src/grammar/store.ts +++ b/src/grammar/store.ts @@ -1,10 +1,24 @@ -import { ASTConstant, ASTFunction, ASTNativeFunction, ASTType } from "./ast"; +import { + ASTProgram, + ASTConstant, + ASTFunction, + ASTNativeFunction, + ASTType, +} from "./ast"; import { CompilerContext, createContextStore } from "../context"; import { parse } from "./grammar"; import { TypeOrigin } from "../types/types"; -type ASTStore = { - sources: { code: string; path: string }[]; +export type TactSource = { code: string; path: string; origin: TypeOrigin }; + +/** + * Represents the storage for all AST-related data within the compiler context. + * @property functions AST entries representing top-level functions. + * @property constants AST entries representing top-level constant definitions. + * @property types AST entries representing structures, contracts, and traits. + */ +export type ASTStore = { + sources: TactSource[]; funcSources: { code: string; path: string }[]; functions: (ASTFunction | ASTNativeFunction)[]; constants: ASTConstant[]; @@ -13,6 +27,12 @@ type ASTStore = { const store = createContextStore(); +/** + * Retrieves the raw AST for the given context. + * @param ctx The compiler context from which the AST is retrieved. + * @throws Will throw an error if the AST is not found in the context. + * @returns The AST types associated with the context. + */ export function getRawAST(ctx: CompilerContext) { const r = store.get(ctx, "types"); if (!r) { @@ -21,33 +41,47 @@ export function getRawAST(ctx: CompilerContext) { return r; } +/** + * Parses multiple Tact source files into AST programs. + */ +export function parsePrograms(sources: TactSource[]): ASTProgram[] { + return sources.map((source) => + parse(source.code, source.path, source.origin), + ); +} + +/** + * Extends the compiler context by adding AST entries and source information from + * given sources and parsed programs. + * @param parsedPrograms An optional array of previously parsed programs. If not defined, they will be parsed from `sources`. + * @returns The updated compiler context. + */ export function openContext( ctx: CompilerContext, - sources: { code: string; path: string; origin: TypeOrigin }[], + sources: TactSource[], funcSources: { code: string; path: string }[], -) { - const asts = sources.map((source) => - parse(source.code, source.path, source.origin), - ); + parsedPrograms?: ASTProgram[], +): CompilerContext { + const programs = parsedPrograms ? parsedPrograms : parsePrograms(sources); const types: ASTType[] = []; const functions: (ASTNativeFunction | ASTFunction)[] = []; const constants: ASTConstant[] = []; - for (const a of asts) { - for (const e of a.entries) { + for (const program of programs) { + for (const entry of program.entries) { if ( - e.kind === "def_struct" || - e.kind === "def_contract" || - e.kind === "def_trait" || - e.kind === "primitive" + entry.kind === "def_struct" || + entry.kind === "def_contract" || + entry.kind === "def_trait" || + entry.kind === "primitive" ) { - types.push(e); + types.push(entry); } else if ( - e.kind === "def_function" || - e.kind === "def_native_function" + entry.kind === "def_function" || + entry.kind === "def_native_function" ) { - functions.push(e); - } else if (e.kind === "def_constant") { - constants.push(e); + functions.push(entry); + } else if (entry.kind === "def_constant") { + constants.push(entry); } } } @@ -60,23 +94,3 @@ export function openContext( }); return ctx; } - -// Creates a mock context with the given AST elements needed for testing -// purposes -export function openMockContext( - ctx: CompilerContext, - types: ASTType[], - functions: (ASTNativeFunction | ASTFunction)[], - constants: ASTConstant[], -) { - const sources: { code: string; path: string }[] = []; - const funcSources: { code: string; path: string }[] = []; - ctx = store.set(ctx, "types", { - sources, - funcSources, - functions, - constants, - types, - }); - return ctx; -} diff --git a/src/pipeline/precompile.ts b/src/pipeline/precompile.ts index 7d951e873..eca8f3533 100644 --- a/src/pipeline/precompile.ts +++ b/src/pipeline/precompile.ts @@ -17,7 +17,7 @@ export function precompile( // Load all sources const imported = resolveImports({ entrypoint, project, stdlib }); - // Perform initial compiler steps + // Add information about all the source code entries to the context ctx = openContext(ctx, imported.tact, imported.func); // First load type descriptors and check that