UNPKG

prettier-plugin-solidity

Version:

A Prettier Plugin for automatically formatting your Solidity code.

83 lines (66 loc) 2.51 kB
import { coerce, satisfies } from 'semver'; import { NonterminalKind } from '@nomicfoundation/slang/cst'; import { printFunction } from '../slang-printers/print-function.js'; import { getNodeMetadata, updateMetadata } from '../slang-utils/metadata.js'; import { FunctionName } from './FunctionName.js'; import { ParametersDeclaration } from './ParametersDeclaration.js'; import { FunctionAttributes } from './FunctionAttributes.js'; import { ReturnsDeclaration } from './ReturnsDeclaration.js'; import { FunctionBody } from './FunctionBody.js'; import type * as ast from '@nomicfoundation/slang/ast'; import type { AstPath, Doc, ParserOptions } from 'prettier'; import type { AstNode } from './types.d.ts'; import type { PrintFunction, SlangNode } from '../types.d.ts'; export class FunctionDefinition implements SlangNode { readonly kind = NonterminalKind.FunctionDefinition; comments; loc; name: FunctionName; parameters: ParametersDeclaration; attributes: FunctionAttributes; returns?: ReturnsDeclaration; body: FunctionBody; constructor(ast: ast.FunctionDefinition, options: ParserOptions<AstNode>) { let metadata = getNodeMetadata(ast); this.name = new FunctionName(ast.name); this.parameters = new ParametersDeclaration(ast.parameters, options); this.attributes = new FunctionAttributes(ast.attributes, options); if (ast.returns) { this.returns = new ReturnsDeclaration(ast.returns, options); } this.body = new FunctionBody(ast.body, options); metadata = updateMetadata(metadata, [ this.name, this.parameters, this.attributes, this.returns, this.body ]); this.comments = metadata.comments; this.loc = metadata.loc; // Older versions of Solidity defined a constructor as a function having // the same name as the contract. const compiler = coerce(options.compiler); if (compiler && satisfies(compiler, '>=0.5.0')) { this.cleanModifierInvocationArguments(); } } cleanModifierInvocationArguments(): void { for (const { variant: attribute } of this.attributes.items) { if ( typeof attribute !== 'string' && attribute.kind === NonterminalKind.ModifierInvocation ) { attribute.cleanModifierInvocationArguments(); } } } print(path: AstPath<FunctionDefinition>, print: PrintFunction): Doc { return printFunction( ['function ', path.call(print, 'name')], this, path, print ); } }