UNPKG

@taml/parser

Version:

Parser for TAML (Terminal ANSI Markup Language) that generates AST nodes

251 lines 17.5 kB
/** * Validator for TAML tokens and syntax */ import { isValidTag } from "@taml/ast"; import { InvalidTagError, MismatchedTagError, UnclosedTagError, } from "./errors.js"; import { isCloseTagToken, isEofToken, isOpenTagToken, isTextToken, } from "./tokenizer.js"; /** * TAML Validator class */ export class TamlValidator { source; constructor(source) { this.source = source; } /** * Validate a sequence of tokens */ validateTokens(tokens) { const context = { tokens, position: 0, source: this.source, tagStack: [], errors: [], }; try { this.validateTokenSequence(context); // Check for unclosed tags at the end if (context.tagStack.length > 0) { const unclosedTag = context.tagStack[context.tagStack.length - 1]; if (unclosedTag) { const error = new UnclosedTagError(unclosedTag.tagName, unclosedTag.token.start, unclosedTag.token.line, unclosedTag.token.column, this.source); context.errors.push(error); } } return { valid: context.errors.length === 0, errors: context.errors, }; } catch (error) { if (error instanceof Error) { context.errors.push(error); } return { valid: false, errors: context.errors, }; } } /** * Validate the sequence of tokens */ validateTokenSequence(context) { while (context.position < context.tokens.length) { const token = context.tokens[context.position]; if (!token) { break; } if (isEofToken(token)) { break; } if (isOpenTagToken(token)) { this.validateOpenTag(context, token); } else if (isCloseTagToken(token)) { this.validateCloseTag(context, token); } else if (isTextToken(token)) { // Text tokens are always valid context.position++; } else { // Skip unexpected tokens context.position++; } } } /** * Validate an opening tag token */ validateOpenTag(context, token) { if (!isOpenTagToken(token)) { context.position++; return; } // Validate tag name against TAML specification if (!isValidTag(token.tagName)) { const error = new InvalidTagError(token.tagName, token.start, token.line, token.column, this.source); context.errors.push(error); context.position++; return; } // Push tag onto stack for tracking context.tagStack.push({ tagName: token.tagName, token, }); context.position++; } /** * Validate a closing tag token */ validateCloseTag(context, token) { if (!isCloseTagToken(token)) { context.position++; return; } // Validate tag name against TAML specification if (!isValidTag(token.tagName)) { const error = new InvalidTagError(token.tagName, token.start, token.line, token.column, this.source); context.errors.push(error); context.position++; return; } // Check if there's a matching opening tag if (context.tagStack.length === 0) { // No opening tag to match - this is an extra closing tag const error = new MismatchedTagError("(none)", token.tagName, token.start, token.line, token.column, this.source); context.errors.push(error); context.position++; return; } // Check if the closing tag matches the most recent opening tag const lastOpenTag = context.tagStack[context.tagStack.length - 1]; if (lastOpenTag && lastOpenTag.tagName !== token.tagName) { const error = new MismatchedTagError(lastOpenTag.tagName, token.tagName, token.start, token.line, token.column, this.source); context.errors.push(error); context.position++; return; } // Valid closing tag - pop from stack context.tagStack.pop(); context.position++; } /** * Get current validation state for debugging */ getDebugInfo(context) { return { position: context.position, currentToken: context.position < context.tokens.length ? (context.tokens[context.position] ?? null) : null, tagStack: context.tagStack.map((entry) => entry.tagName), errorCount: context.errors.length, }; } } /** * Convenience function to validate TAML tokens */ export function validateTamlTokens(tokens, source) { const validator = new TamlValidator(source); return validator.validateTokens(tokens); } /** * Validate tag name against TAML specification */ export function validateTagName(tagName) { return isValidTag(tagName); } /** * Check if nesting is valid (stack-based validation) */ export function validateNesting(tokens) { const tagStack = []; const unclosedTags = []; const mismatchedTags = []; for (const token of tokens) { if (isOpenTagToken(token)) { if (isValidTag(token.tagName)) { tagStack.push(token.tagName); } } else if (isCloseTagToken(token)) { if (isValidTag(token.tagName)) { if (tagStack.length === 0) { mismatchedTags.push({ expected: "(none)", actual: token.tagName }); } else { const lastTag = tagStack[tagStack.length - 1]; if (lastTag && lastTag === token.tagName) { tagStack.pop(); } else if (lastTag) { mismatchedTags.push({ expected: lastTag, actual: token.tagName }); } } } } } // Any remaining tags in stack are unclosed unclosedTags.push(...tagStack); return { valid: unclosedTags.length === 0 && mismatchedTags.length === 0, unclosedTags, mismatchedTags, }; } /** * Validate proper tag closure */ export function validateTagClosure(tokens) { const tagStack = []; const issues = []; for (const token of tokens) { if (isOpenTagToken(token)) { if (isValidTag(token.tagName)) { tagStack.push({ tagName: token.tagName, position: token.start }); } } else if (isCloseTagToken(token)) { if (isValidTag(token.tagName)) { if (tagStack.length === 0) { issues.push({ type: "extra", tagName: token.tagName, position: token.start, }); } else { const lastTag = tagStack[tagStack.length - 1]; if (lastTag && lastTag.tagName === token.tagName) { tagStack.pop(); } else { issues.push({ type: "mismatched", tagName: token.tagName, position: token.start, }); } } } } } // Any remaining tags in stack are unclosed for (const unclosedTag of tagStack) { issues.push({ type: "unclosed", tagName: unclosedTag.tagName, position: unclosedTag.position, }); } return { valid: issues.length === 0, issues, }; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validator.js","sourceRoot":"","sources":["../ts/validator.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAgB,UAAU,EAAE,MAAM,WAAW,CAAC;AACrD,OAAO,EACL,eAAe,EACf,kBAAkB,EAClB,gBAAgB,GACjB,MAAM,aAAa,CAAC;AACrB,OAAO,EAEL,eAAe,EACf,UAAU,EACV,cAAc,EACd,WAAW,GACZ,MAAM,gBAAgB,CAAC;AAqBxB;;GAEG;AACH,MAAM,OAAO,aAAa;IACK;IAA7B,YAA6B,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;IAAG,CAAC;IAE/C;;OAEG;IACH,cAAc,CAAC,MAAmB;QAChC,MAAM,OAAO,GAAsB;YACjC,MAAM;YACN,QAAQ,EAAE,CAAC;YACX,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,EAAE;YACZ,MAAM,EAAE,EAAE;SACX,CAAC;QAEF,IAAI,CAAC;YACH,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;YAEpC,qCAAqC;YACrC,IAAI,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAChC,MAAM,WAAW,GAAG,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;gBAClE,IAAI,WAAW,EAAE,CAAC;oBAChB,MAAM,KAAK,GAAG,IAAI,gBAAgB,CAChC,WAAW,CAAC,OAAO,EACnB,WAAW,CAAC,KAAK,CAAC,KAAK,EACvB,WAAW,CAAC,KAAK,CAAC,IAAI,EACtB,WAAW,CAAC,KAAK,CAAC,MAAM,EACxB,IAAI,CAAC,MAAM,CACZ,CAAC;oBACF,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC7B,CAAC;YACH,CAAC;YAED,OAAO;gBACL,KAAK,EAAE,OAAO,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC;gBAClC,MAAM,EAAE,OAAO,CAAC,MAAM;aACvB,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;gBAC3B,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC7B,CAAC;YACD,OAAO;gBACL,KAAK,EAAE,KAAK;gBACZ,MAAM,EAAE,OAAO,CAAC,MAAM;aACvB,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;OAEG;IACK,qBAAqB,CAAC,OAA0B;QACtD,OAAO,OAAO,CAAC,QAAQ,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;YAChD,MAAM,KAAK,GAAG,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YAE/C,IAAI,CAAC,KAAK,EAAE,CAAC;gBACX,MAAM;YACR,CAAC;YAED,IAAI,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;gBACtB,MAAM;YACR,CAAC;YAED,IAAI,cAAc,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC1B,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;YACvC,CAAC;iBAAM,IAAI,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC;gBAClC,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;YACxC,CAAC;iBAAM,IAAI,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC9B,+BAA+B;gBAC/B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACrB,CAAC;iBAAM,CAAC;gBACN,yBAAyB;gBACzB,OAAO,CAAC,QAAQ,EAAE,CAAC;YACrB,CAAC;QACH,CAAC;IACH,CAAC;IAED;;OAEG;IACK,eAAe,CAAC,OAA0B,EAAE,KAAgB;QAClE,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,EAAE,CAAC;YAC3B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,+CAA+C;QAC/C,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;YAC/B,MAAM,KAAK,GAAG,IAAI,eAAe,CAC/B,KAAK,CAAC,OAAO,EACb,KAAK,CAAC,KAAK,EACX,KAAK,CAAC,IAAI,EACV,KAAK,CAAC,MAAM,EACZ,IAAI,CAAC,MAAM,CACZ,CAAC;YACF,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,mCAAmC;QACnC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC;YACpB,OAAO,EAAE,KAAK,CAAC,OAAkB;YACjC,KAAK;SACN,CAAC,CAAC;QAEH,OAAO,CAAC,QAAQ,EAAE,CAAC;IACrB,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,OAA0B,EAAE,KAAgB;QACnE,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC;YAC5B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,+CAA+C;QAC/C,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;YAC/B,MAAM,KAAK,GAAG,IAAI,eAAe,CAC/B,KAAK,CAAC,OAAO,EACb,KAAK,CAAC,KAAK,EACX,KAAK,CAAC,IAAI,EACV,KAAK,CAAC,MAAM,EACZ,IAAI,CAAC,MAAM,CACZ,CAAC;YACF,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,0CAA0C;QAC1C,IAAI,OAAO,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAClC,yDAAyD;YACzD,MAAM,KAAK,GAAG,IAAI,kBAAkB,CAClC,QAAQ,EACR,KAAK,CAAC,OAAO,EACb,KAAK,CAAC,KAAK,EACX,KAAK,CAAC,IAAI,EACV,KAAK,CAAC,MAAM,EACZ,IAAI,CAAC,MAAM,CACZ,CAAC;YACF,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,+DAA+D;QAC/D,MAAM,WAAW,GAAG,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAClE,IAAI,WAAW,IAAI,WAAW,CAAC,OAAO,KAAK,KAAK,CAAC,OAAO,EAAE,CAAC;YACzD,MAAM,KAAK,GAAG,IAAI,kBAAkB,CAClC,WAAW,CAAC,OAAO,EACnB,KAAK,CAAC,OAAO,EACb,KAAK,CAAC,KAAK,EACX,KAAK,CAAC,IAAI,EACV,KAAK,CAAC,MAAM,EACZ,IAAI,CAAC,MAAM,CACZ,CAAC;YACF,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,qCAAqC;QACrC,OAAO,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;QACvB,OAAO,CAAC,QAAQ,EAAE,CAAC;IACrB,CAAC;IAED;;OAEG;IACH,YAAY,CAAC,OAA0B;QAMrC,OAAO;YACL,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,YAAY,EACV,OAAO,CAAC,QAAQ,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM;gBACtC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC;gBAC5C,CAAC,CAAC,IAAI;YACV,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC;YACxD,UAAU,EAAE,OAAO,CAAC,MAAM,CAAC,MAAM;SAClC,CAAC;IACJ,CAAC;CACF;AAED;;GAEG;AACH,MAAM,UAAU,kBAAkB,CAChC,MAAmB,EACnB,MAAc;IAEd,MAAM,SAAS,GAAG,IAAI,aAAa,CAAC,MAAM,CAAC,CAAC;IAC5C,OAAO,SAAS,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;AAC1C,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,eAAe,CAAC,OAAe;IAC7C,OAAO,UAAU,CAAC,OAAO,CAAC,CAAC;AAC7B,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,eAAe,CAAC,MAAmB;IAKjD,MAAM,QAAQ,GAAa,EAAE,CAAC;IAC9B,MAAM,YAAY,GAAa,EAAE,CAAC;IAClC,MAAM,cAAc,GAAgD,EAAE,CAAC;IAEvE,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,IAAI,cAAc,CAAC,KAAK,CAAC,EAAE,CAAC;YAC1B,IAAI,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC9B,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAC/B,CAAC;QACH,CAAC;aAAM,IAAI,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC;YAClC,IAAI,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC9B,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBAC1B,cAAc,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;gBACrE,CAAC;qBAAM,CAAC;oBACN,MAAM,OAAO,GAAG,QAAQ,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;oBAC9C,IAAI,OAAO,IAAI,OAAO,KAAK,KAAK,CAAC,OAAO,EAAE,CAAC;wBACzC,QAAQ,CAAC,GAAG,EAAE,CAAC;oBACjB,CAAC;yBAAM,IAAI,OAAO,EAAE,CAAC;wBACnB,cAAc,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;oBACpE,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,2CAA2C;IAC3C,YAAY,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC;IAE/B,OAAO;QACL,KAAK,EAAE,YAAY,CAAC,MAAM,KAAK,CAAC,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC;QAC/D,YAAY;QACZ,cAAc;KACf,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,kBAAkB,CAAC,MAAmB;IAQpD,MAAM,QAAQ,GAAiD,EAAE,CAAC;IAClE,MAAM,MAAM,GAIP,EAAE,CAAC;IAER,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,IAAI,cAAc,CAAC,KAAK,CAAC,EAAE,CAAC;YAC1B,IAAI,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC9B,QAAQ,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC;YACnE,CAAC;QACH,CAAC;aAAM,IAAI,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC;YAClC,IAAI,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC9B,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBAC1B,MAAM,CAAC,IAAI,CAAC;wBACV,IAAI,EAAE,OAAO;wBACb,OAAO,EAAE,KAAK,CAAC,OAAO;wBACtB,QAAQ,EAAE,KAAK,CAAC,KAAK;qBACtB,CAAC,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACN,MAAM,OAAO,GAAG,QAAQ,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;oBAC9C,IAAI,OAAO,IAAI,OAAO,CAAC,OAAO,KAAK,KAAK,CAAC,OAAO,EAAE,CAAC;wBACjD,QAAQ,CAAC,GAAG,EAAE,CAAC;oBACjB,CAAC;yBAAM,CAAC;wBACN,MAAM,CAAC,IAAI,CAAC;4BACV,IAAI,EAAE,YAAY;4BAClB,OAAO,EAAE,KAAK,CAAC,OAAO;4BACtB,QAAQ,EAAE,KAAK,CAAC,KAAK;yBACtB,CAAC,CAAC;oBACL,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,2CAA2C;IAC3C,KAAK,MAAM,WAAW,IAAI,QAAQ,EAAE,CAAC;QACnC,MAAM,CAAC,IAAI,CAAC;YACV,IAAI,EAAE,UAAU;YAChB,OAAO,EAAE,WAAW,CAAC,OAAO;YAC5B,QAAQ,EAAE,WAAW,CAAC,QAAQ;SAC/B,CAAC,CAAC;IACL,CAAC;IAED,OAAO;QACL,KAAK,EAAE,MAAM,CAAC,MAAM,KAAK,CAAC;QAC1B,MAAM;KACP,CAAC;AACJ,CAAC"}