rawsql-ts
Version:
High-performance SQL parser and AST analyzer written in TypeScript. Provides fast parsing and advanced transformation capabilities.
112 lines • 5.82 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ValuesQueryParser = void 0;
const Lexeme_1 = require("../models/Lexeme");
const SelectQuery_1 = require("../models/SelectQuery");
const ValueComponent_1 = require("../models/ValueComponent");
const SqlTokenizer_1 = require("./SqlTokenizer");
const ValueParser_1 = require("./ValueParser");
const LexemeCommentUtils_1 = require("./utils/LexemeCommentUtils");
class ValuesQueryParser {
static parse(query) {
const tokenizer = new SqlTokenizer_1.SqlTokenizer(query); // Initialize tokenizer
const lexemes = tokenizer.readLexmes(); // Get tokens
// Parse
const result = this.parseFromLexeme(lexemes, 0);
// Error if there are remaining tokens
if (result.newIndex < lexemes.length) {
throw new Error(`Syntax error: Unexpected token "${lexemes[result.newIndex].value}" at position ${result.newIndex}. The VALUES clause is complete but there are additional tokens.`);
}
return result.value;
}
static parseFromLexeme(lexemes, index) {
let idx = index;
if (idx >= lexemes.length) {
throw new Error(`Syntax error at position ${idx}: Expected 'VALUES' keyword but input ended early.`);
}
const valuesLexeme = lexemes[idx];
if (valuesLexeme.value.toLowerCase() !== 'values') {
throw new Error(`Syntax error at position ${idx}: Expected 'VALUES' keyword but found "${valuesLexeme.value}". VALUES clauses must start with the VALUES keyword.`);
}
const valuesComments = (0, LexemeCommentUtils_1.extractLexemeComments)(valuesLexeme);
idx++;
if (idx >= lexemes.length) {
throw new Error(`Syntax error: Unexpected end of input after 'VALUES' keyword. The VALUES clause requires at least one tuple expression.`);
}
const tuples = [];
const firstTuple = this.parseTuple(lexemes, idx);
tuples.push(firstTuple.value);
idx = firstTuple.newIndex;
if (valuesComments.after.length > 0) {
firstTuple.value.addPositionedComments('before', valuesComments.after);
}
while (idx < lexemes.length && (lexemes[idx].type & Lexeme_1.TokenType.Comma)) {
idx++;
const tuple = this.parseTuple(lexemes, idx);
tuples.push(tuple.value);
idx = tuple.newIndex;
}
const query = new SelectQuery_1.ValuesQuery(tuples);
if (valuesComments.before.length > 0) {
query.headerComments = valuesComments.before;
}
return { value: query, newIndex: idx };
}
static parseTuple(lexemes, index) {
let idx = index;
if (idx >= lexemes.length || lexemes[idx].type !== Lexeme_1.TokenType.OpenParen) {
throw new Error(`Syntax error at position ${idx}: Expected opening parenthesis but found "${idx < lexemes.length ? lexemes[idx].value : "end of input"}". Tuple expressions in VALUES clause must be enclosed in parentheses.`);
}
const openingComments = (0, LexemeCommentUtils_1.extractLexemeComments)(lexemes[idx]);
idx++;
const values = [];
if (idx >= lexemes.length) {
throw new Error(`Syntax error: Unexpected end of input after opening parenthesis in tuple expression.`);
}
if (lexemes[idx].type & Lexeme_1.TokenType.CloseParen) {
const tuple = new ValueComponent_1.TupleExpression([]);
const closingComments = (0, LexemeCommentUtils_1.extractLexemeComments)(lexemes[idx]);
idx++;
if (openingComments.before.length > 0) {
tuple.addPositionedComments('before', openingComments.before);
}
if (closingComments.after.length > 0) {
tuple.addPositionedComments('after', closingComments.after);
}
return { value: tuple, newIndex: idx };
}
const firstValue = ValueParser_1.ValueParser.parseFromLexeme(lexemes, idx);
values.push(firstValue.value);
idx = firstValue.newIndex;
while (idx < lexemes.length && (lexemes[idx].type & Lexeme_1.TokenType.Comma)) {
idx++;
if (idx >= lexemes.length) {
throw new Error(`Syntax error: Unexpected end of input after comma in tuple expression.`);
}
const value = ValueParser_1.ValueParser.parseFromLexeme(lexemes, idx);
values.push(value.value);
idx = value.newIndex;
}
if (idx >= lexemes.length || lexemes[idx].type !== Lexeme_1.TokenType.CloseParen) {
throw new Error(`Syntax error at position ${idx}: Expected closing parenthesis but found "${idx < lexemes.length ? lexemes[idx].value : "end of input"}". Tuple expressions in VALUES clause must be enclosed in parentheses.`);
}
const closingComments = (0, LexemeCommentUtils_1.extractLexemeComments)(lexemes[idx]);
idx++;
const tuple = new ValueComponent_1.TupleExpression(values);
if (openingComments.before.length > 0) {
tuple.addPositionedComments('before', openingComments.before);
}
if (openingComments.after.length > 0 && values.length > 0) {
values[0].addPositionedComments('before', openingComments.after);
}
if (closingComments.before.length > 0 && values.length > 0) {
values[values.length - 1].addPositionedComments('after', closingComments.before);
}
if (closingComments.after.length > 0) {
tuple.addPositionedComments('after', closingComments.after);
}
return { value: tuple, newIndex: idx };
}
}
exports.ValuesQueryParser = ValuesQueryParser;
//# sourceMappingURL=ValuesQueryParser.js.map