2020-08-21 07:44:17 -07:00
|
|
|
/// Library to convert AST nodes back to text.
|
|
|
|
///
|
|
|
|
/// See the [NodeToText] extension for details.
|
|
|
|
library utils.node_to_text;
|
|
|
|
|
|
|
|
import 'package:charcode/charcode.dart';
|
|
|
|
import 'package:sqlparser/sqlparser.dart';
|
|
|
|
|
2021-01-23 12:03:03 -08:00
|
|
|
class NodeSqlBuilder extends AstVisitor<void, void> {
|
|
|
|
final StringSink buffer;
|
2021-01-24 03:27:53 -08:00
|
|
|
|
|
|
|
/// Whether we need to insert a space before writing the next identifier.
|
|
|
|
bool needsSpace = false;
|
2020-08-21 07:44:17 -07:00
|
|
|
|
2021-01-23 12:03:03 -08:00
|
|
|
NodeSqlBuilder([StringSink? buffer]) : buffer = buffer ?? StringBuffer();
|
|
|
|
|
2021-01-24 03:27:53 -08:00
|
|
|
/// Writes a space character if [needsSpace] is set.
|
|
|
|
///
|
|
|
|
/// This also resets [needsSpace] to `false`.
|
|
|
|
void spaceIfNeeded() {
|
|
|
|
if (needsSpace) {
|
|
|
|
needsSpace = false;
|
|
|
|
_space();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
2022-03-08 01:58:40 -08:00
|
|
|
void visitAggregateFunctionInvocation(
|
|
|
|
AggregateFunctionInvocation e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(e.name);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.parameters, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.filter != null) {
|
|
|
|
_keyword(TokenType.filter);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.where);
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(e.filter!, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
2022-03-08 01:58:40 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitWindowFunctionInvocation(WindowFunctionInvocation e, void arg) {
|
|
|
|
visitAggregateFunctionInvocation(e, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.windowDefinition != null) {
|
|
|
|
_keyword(TokenType.over);
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(e.windowDefinition!, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
} else if (e.windowName != null) {
|
|
|
|
_keyword(TokenType.over);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.windowName!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-08 13:53:57 -07:00
|
|
|
@override
|
|
|
|
void visitBeginTransaction(BeginTransactionStatement e, void arg) {
|
|
|
|
_keyword(TokenType.begin);
|
|
|
|
|
|
|
|
switch (e.mode) {
|
|
|
|
case TransactionMode.none:
|
|
|
|
break;
|
|
|
|
case TransactionMode.deferred:
|
|
|
|
_keyword(TokenType.deferred);
|
|
|
|
break;
|
|
|
|
case TransactionMode.immediate:
|
|
|
|
_keyword(TokenType.immediate);
|
|
|
|
break;
|
|
|
|
case TransactionMode.exclusive:
|
|
|
|
_keyword(TokenType.exclusive);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitBetweenExpression(BetweenExpression e, void arg) {
|
|
|
|
visit(e.check, arg);
|
|
|
|
|
|
|
|
if (e.not) {
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(TokenType.between);
|
|
|
|
visit(e.lower, arg);
|
|
|
|
_keyword(TokenType.and);
|
|
|
|
visit(e.upper, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitBinaryExpression(BinaryExpression e, void arg) {
|
|
|
|
visit(e.left, arg);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
final operatorSymbol = const {
|
2020-08-21 07:44:17 -07:00
|
|
|
TokenType.doublePipe: '||',
|
|
|
|
TokenType.star: '*',
|
|
|
|
TokenType.slash: '/',
|
|
|
|
TokenType.percent: '%',
|
|
|
|
TokenType.plus: '+',
|
|
|
|
TokenType.minus: '-',
|
|
|
|
TokenType.shiftLeft: '<<',
|
|
|
|
TokenType.shiftRight: '>>',
|
2020-08-22 03:36:36 -07:00
|
|
|
TokenType.ampersand: '&',
|
2020-08-21 07:44:17 -07:00
|
|
|
TokenType.pipe: '|',
|
|
|
|
TokenType.less: '<',
|
|
|
|
TokenType.lessEqual: '<=',
|
|
|
|
TokenType.more: '>',
|
|
|
|
TokenType.moreEqual: '>=',
|
|
|
|
TokenType.equal: '=',
|
|
|
|
TokenType.doubleEqual: '==',
|
|
|
|
TokenType.exclamationEqual: '!=',
|
|
|
|
TokenType.lessMore: '<>',
|
2022-02-23 13:06:42 -08:00
|
|
|
TokenType.dashRangle: '->',
|
|
|
|
TokenType.dashRangleRangle: '->>',
|
2020-08-21 07:44:17 -07:00
|
|
|
}[e.operator.type];
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
if (operatorSymbol != null) {
|
|
|
|
symbol(operatorSymbol, spaceBefore: true, spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
} else {
|
|
|
|
_keyword(e.operator.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.right, arg);
|
|
|
|
}
|
|
|
|
|
2021-09-09 09:05:03 -07:00
|
|
|
void _writeStatements(Iterable<Statement> statements) {
|
|
|
|
for (final stmt in statements) {
|
|
|
|
visit(stmt, null);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(';');
|
2021-09-09 09:05:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitBlock(Block block, void arg) {
|
|
|
|
_keyword(TokenType.begin);
|
2021-09-09 09:05:03 -07:00
|
|
|
_writeStatements(block.statements);
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.end);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitBooleanLiteral(BooleanLiteral e, void arg) {
|
|
|
|
_keyword(e.value ? TokenType.$true : TokenType.$false);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCaseExpression(CaseExpression e, void arg) {
|
|
|
|
_keyword(TokenType.$case);
|
|
|
|
visitNullable(e.base, arg);
|
|
|
|
visitList(e.whens, arg);
|
|
|
|
|
|
|
|
final elseExpr = e.elseExpr;
|
|
|
|
if (elseExpr != null) {
|
|
|
|
_keyword(TokenType.$else);
|
|
|
|
visit(elseExpr, arg);
|
|
|
|
}
|
2020-08-22 03:36:36 -07:00
|
|
|
|
|
|
|
_keyword(TokenType.end);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCastExpression(CastExpression e, void arg) {
|
|
|
|
_keyword(TokenType.cast);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.operand, arg);
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(e.typeName, spaceBefore: true);
|
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCollateExpression(CollateExpression e, void arg) {
|
|
|
|
visit(e.inner, arg);
|
|
|
|
_keyword(TokenType.collate);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.collation);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitColumnConstraint(ColumnConstraint e, void arg) {
|
|
|
|
if (e.name != null) {
|
|
|
|
_keyword(TokenType.constraint);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.name!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
2021-11-07 13:10:08 -08:00
|
|
|
e.when(primaryKey: (primaryKey) {
|
|
|
|
_keyword(TokenType.primary);
|
|
|
|
_keyword(TokenType.key);
|
|
|
|
_orderingMode(primaryKey.mode);
|
|
|
|
_conflictClause(primaryKey.onConflict);
|
|
|
|
if (primaryKey.autoIncrement) _keyword(TokenType.autoincrement);
|
|
|
|
}, notNull: (notNull) {
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
_keyword(TokenType.$null);
|
|
|
|
_conflictClause(notNull.onConflict);
|
|
|
|
}, unique: (unique) {
|
|
|
|
_keyword(TokenType.unique);
|
|
|
|
_conflictClause(unique.onConflict);
|
|
|
|
}, check: (check) {
|
|
|
|
_keyword(TokenType.check);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2021-11-07 13:10:08 -08:00
|
|
|
visit(check.expression, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2021-11-07 13:10:08 -08:00
|
|
|
}, isDefault: (def) {
|
|
|
|
_keyword(TokenType.$default);
|
|
|
|
final expr = def.expression;
|
|
|
|
if (expr is Literal) {
|
|
|
|
visit(expr, arg);
|
|
|
|
} else {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2021-11-07 13:10:08 -08:00
|
|
|
visit(expr, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2021-11-07 13:10:08 -08:00
|
|
|
}
|
|
|
|
}, collate: (collate) {
|
|
|
|
_keyword(TokenType.collate);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(collate.collation);
|
2021-11-07 13:10:08 -08:00
|
|
|
}, foreignKey: (foreignKey) {
|
|
|
|
visit(foreignKey.clause, arg);
|
|
|
|
}, generatedAs: (generatedAs) {
|
|
|
|
_keyword(TokenType.generated);
|
|
|
|
_keyword(TokenType.always);
|
|
|
|
_keyword(TokenType.as);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2021-11-07 13:10:08 -08:00
|
|
|
visit(generatedAs.expression, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2021-11-07 13:10:08 -08:00
|
|
|
|
|
|
|
if (generatedAs.stored) {
|
|
|
|
_keyword(TokenType.stored);
|
|
|
|
} else {
|
|
|
|
_keyword(TokenType.virtual);
|
|
|
|
}
|
|
|
|
});
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitColumnDefinition(ColumnDefinition e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.columnName);
|
2020-08-21 07:44:17 -07:00
|
|
|
if (e.typeName != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(e.typeName!, spaceAfter: true, spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visitList(e.constraints, arg);
|
|
|
|
}
|
|
|
|
|
2021-09-08 13:53:57 -07:00
|
|
|
@override
|
|
|
|
void visitCommitStatement(CommitStatement e, void arg) {
|
|
|
|
_keyword(TokenType.commit);
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitCommonTableExpression(CommonTableExpression e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.cteTableName);
|
2020-08-21 07:44:17 -07:00
|
|
|
if (e.columnNames != null) {
|
2022-10-29 08:30:18 -07:00
|
|
|
symbol('(', spaceBefore: true);
|
|
|
|
|
|
|
|
var first = true;
|
|
|
|
for (final columnName in e.columnNames!) {
|
|
|
|
if (!first) {
|
|
|
|
symbol(',', spaceAfter: true);
|
|
|
|
}
|
|
|
|
|
|
|
|
identifier(columnName, spaceBefore: !first, spaceAfter: false);
|
|
|
|
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(TokenType.as);
|
2021-03-13 05:00:06 -08:00
|
|
|
switch (e.materializationHint) {
|
|
|
|
case MaterializationHint.notMaterialized:
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
_keyword(TokenType.materialized);
|
|
|
|
break;
|
|
|
|
case MaterializationHint.materialized:
|
|
|
|
_keyword(TokenType.materialized);
|
|
|
|
break;
|
|
|
|
case null:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.as, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCompoundSelectPart(CompoundSelectPart e, void arg) {
|
|
|
|
switch (e.mode) {
|
|
|
|
case CompoundSelectMode.union:
|
|
|
|
_keyword(TokenType.union);
|
|
|
|
break;
|
|
|
|
case CompoundSelectMode.unionAll:
|
|
|
|
_keyword(TokenType.union);
|
|
|
|
_keyword(TokenType.all);
|
|
|
|
break;
|
|
|
|
case CompoundSelectMode.intersect:
|
|
|
|
_keyword(TokenType.intersect);
|
|
|
|
break;
|
|
|
|
case CompoundSelectMode.except:
|
|
|
|
_keyword(TokenType.except);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.select, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCompoundSelectStatement(CompoundSelectStatement e, void arg) {
|
|
|
|
visitNullable(e.withClause, arg);
|
|
|
|
visit(e.base, arg);
|
|
|
|
visitList(e.additional, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCreateIndexStatement(CreateIndexStatement e, void arg) {
|
|
|
|
_keyword(TokenType.create);
|
|
|
|
if (e.unique) {
|
|
|
|
_keyword(TokenType.unique);
|
|
|
|
}
|
|
|
|
_keyword(TokenType.$index);
|
|
|
|
_ifNotExists(e.ifNotExists);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.indexName);
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.on);
|
|
|
|
visit(e.on, arg);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
_join(e.columns, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
_where(e.where);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCreateTableStatement(CreateTableStatement e, void arg) {
|
|
|
|
_keyword(TokenType.create);
|
|
|
|
_keyword(TokenType.table);
|
|
|
|
_ifNotExists(e.ifNotExists);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.tableName);
|
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
_join([...e.columns, ...e.tableConstraints], ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.withoutRowId) {
|
|
|
|
_keyword(TokenType.without);
|
|
|
|
_keyword(TokenType.rowid);
|
|
|
|
}
|
2021-08-28 07:28:29 -07:00
|
|
|
|
|
|
|
if (e.isStrict) {
|
2021-11-11 05:40:41 -08:00
|
|
|
if (e.withoutRowId) symbol(',');
|
2021-08-28 07:28:29 -07:00
|
|
|
|
|
|
|
_keyword(TokenType.strict);
|
|
|
|
}
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCreateTriggerStatement(CreateTriggerStatement e, void arg) {
|
|
|
|
_keyword(TokenType.create);
|
|
|
|
_keyword(TokenType.trigger);
|
|
|
|
_ifNotExists(e.ifNotExists);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.triggerName);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
switch (e.mode) {
|
|
|
|
case TriggerMode.before:
|
|
|
|
_keyword(TokenType.before);
|
|
|
|
break;
|
|
|
|
case TriggerMode.after:
|
|
|
|
_keyword(TokenType.after);
|
|
|
|
break;
|
|
|
|
case TriggerMode.insteadOf:
|
|
|
|
_keyword(TokenType.instead);
|
|
|
|
_keyword(TokenType.of);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Can happen if e.mode == null
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.target, arg);
|
|
|
|
|
|
|
|
_keyword(TokenType.on);
|
|
|
|
visit(e.onTable, arg);
|
|
|
|
|
|
|
|
if (e.when != null) {
|
|
|
|
_keyword(TokenType.when);
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(e.when!, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.action, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCreateViewStatement(CreateViewStatement e, void arg) {
|
|
|
|
_keyword(TokenType.create);
|
|
|
|
_keyword(TokenType.view);
|
|
|
|
_ifNotExists(e.ifNotExists);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.viewName);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.columns != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
|
|
|
symbol(e.columns!.join(','));
|
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(TokenType.as);
|
|
|
|
visit(e.query, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitCreateVirtualTableStatement(
|
|
|
|
CreateVirtualTableStatement e, void arg) {
|
|
|
|
_keyword(TokenType.create);
|
|
|
|
_keyword(TokenType.virtual);
|
|
|
|
_keyword(TokenType.table);
|
|
|
|
_ifNotExists(e.ifNotExists);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.tableName);
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.using);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.moduleName);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(${e.argumentContent.join(', ')})');
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2022-02-26 12:38:28 -08:00
|
|
|
void visitDriftSpecificNode(DriftSpecificNode e, void arg) {
|
2021-09-08 14:20:45 -07:00
|
|
|
if (e is DartPlaceholder) {
|
2022-10-23 09:13:40 -07:00
|
|
|
e.when(
|
|
|
|
isLimit: (_) => _keyword(TokenType.limit),
|
|
|
|
isOrderBy: (_) {
|
|
|
|
_keyword(TokenType.order);
|
|
|
|
_keyword(TokenType.by);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(r'$', spaceBefore: true);
|
|
|
|
symbol(e.name, spaceAfter: true);
|
2021-09-08 14:20:45 -07:00
|
|
|
} else if (e is DeclaredStatement) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.identifier.name);
|
2021-09-08 14:20:45 -07:00
|
|
|
|
|
|
|
if (e.parameters.isNotEmpty) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2021-09-08 14:20:45 -07:00
|
|
|
_join(e.parameters, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2021-09-08 14:20:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (e.as != null) {
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.as!);
|
2021-09-08 14:20:45 -07:00
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(':', spaceAfter: true);
|
2021-09-08 14:20:45 -07:00
|
|
|
visit(e.statement, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(';');
|
2022-02-26 12:38:28 -08:00
|
|
|
} else if (e is DriftFile) {
|
2021-09-08 14:20:45 -07:00
|
|
|
for (final stmt in e.statements) {
|
|
|
|
visit(stmt, arg);
|
|
|
|
buffer.write('\n');
|
|
|
|
needsSpace = false;
|
|
|
|
}
|
|
|
|
} else if (e is ImportStatement) {
|
|
|
|
_keyword(TokenType.import);
|
|
|
|
_stringLiteral(e.importedFile);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(';', spaceAfter: true);
|
2021-09-08 14:20:45 -07:00
|
|
|
} else if (e is StatementParameter) {
|
|
|
|
if (e is VariableTypeHint) {
|
|
|
|
if (e.isRequired) _keyword(TokenType.required);
|
|
|
|
|
|
|
|
visit(e.variable, arg);
|
|
|
|
final typeName = e.typeName;
|
|
|
|
if (typeName != null) {
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(typeName, spaceBefore: true, spaceAfter: true);
|
2021-09-08 14:20:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (e.orNull) {
|
|
|
|
_keyword(TokenType.or);
|
|
|
|
_keyword(TokenType.$null);
|
|
|
|
}
|
|
|
|
} else if (e is DartPlaceholderDefaultValue) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('\$${e.variableName}', spaceAfter: true);
|
|
|
|
symbol('=', spaceBefore: true, spaceAfter: true);
|
2021-09-08 14:20:45 -07:00
|
|
|
visit(e.defaultValue, arg);
|
|
|
|
} else {
|
|
|
|
throw AssertionError('Unknown StatementParameter: $e');
|
|
|
|
}
|
2022-02-26 12:38:28 -08:00
|
|
|
} else if (e is DriftTableName) {
|
2021-09-08 14:20:45 -07:00
|
|
|
_keyword(e.useExistingDartClass ? TokenType.$with : TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.overriddenDataClassName);
|
2021-09-08 14:20:45 -07:00
|
|
|
} else if (e is NestedStarResultColumn) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.tableName);
|
|
|
|
symbol('.**', spaceAfter: true);
|
2022-01-23 08:08:32 -08:00
|
|
|
} else if (e is NestedQueryColumn) {
|
|
|
|
symbol('LIST(');
|
|
|
|
visit(e.select, arg);
|
|
|
|
symbol(')', spaceAfter: true);
|
2021-09-09 09:05:03 -07:00
|
|
|
} else if (e is TransactionBlock) {
|
|
|
|
visit(e.begin, arg);
|
|
|
|
_writeStatements(e.innerStatements);
|
|
|
|
visit(e.commit, arg);
|
2021-09-08 14:20:45 -07:00
|
|
|
}
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitDefaultValues(DefaultValues e, void arg) {
|
|
|
|
_keyword(TokenType.$default);
|
|
|
|
_keyword(TokenType.$values);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitDeferrableClause(DeferrableClause e, void arg) {
|
|
|
|
if (e.not) {
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
}
|
|
|
|
_keyword(TokenType.deferrable);
|
|
|
|
|
|
|
|
switch (e.declaredInitially) {
|
|
|
|
case InitialDeferrableMode.deferred:
|
|
|
|
_keyword(TokenType.initially);
|
|
|
|
_keyword(TokenType.deferred);
|
|
|
|
break;
|
|
|
|
case InitialDeferrableMode.immediate:
|
|
|
|
_keyword(TokenType.initially);
|
|
|
|
_keyword(TokenType.immediate);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// declaredInitially == null, don't do anything
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitDeleteStatement(DeleteStatement e, void arg) {
|
|
|
|
visitNullable(e.withClause, arg);
|
|
|
|
|
|
|
|
_keyword(TokenType.delete);
|
2021-03-13 03:02:35 -08:00
|
|
|
_from(e.from);
|
2020-08-21 07:44:17 -07:00
|
|
|
_where(e.where);
|
2021-03-30 14:24:28 -07:00
|
|
|
visitNullable(e.returning, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitDeleteTriggerTarget(DeleteTarget e, void arg) {
|
|
|
|
_keyword(TokenType.delete);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitDoNothing(DoNothing e, void arg) {
|
|
|
|
_keyword(TokenType.nothing);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitDoUpdate(DoUpdate e, void arg) {
|
|
|
|
_keyword(TokenType.update);
|
|
|
|
_keyword(TokenType.set);
|
|
|
|
_join(e.set, ',');
|
|
|
|
_where(e.where);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitExists(ExistsExpression e, void arg) {
|
|
|
|
_keyword(TokenType.exists);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.select, null);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitExpressionFunctionParameters(ExprFunctionParameters e, void arg) {
|
|
|
|
if (e.distinct) {
|
|
|
|
_keyword(TokenType.distinct);
|
|
|
|
}
|
|
|
|
_join(e.parameters, ',');
|
|
|
|
}
|
|
|
|
|
2021-09-08 13:53:57 -07:00
|
|
|
@override
|
|
|
|
void visitExpressionResultColumn(ExpressionResultColumn e, void arg) {
|
|
|
|
visit(e.expression, arg);
|
|
|
|
if (e.as != null) {
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.as!);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitForeignKeyClause(ForeignKeyClause e, void arg) {
|
|
|
|
_keyword(TokenType.references);
|
|
|
|
visit(e.foreignTable, arg);
|
|
|
|
|
|
|
|
if (e.columnNames.isNotEmpty) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
_join(e.columnNames, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void referenceAction(ReferenceAction action) {
|
|
|
|
switch (action) {
|
|
|
|
case ReferenceAction.setNull:
|
|
|
|
_keyword(TokenType.set);
|
|
|
|
_keyword(TokenType.$null);
|
|
|
|
break;
|
|
|
|
case ReferenceAction.setDefault:
|
|
|
|
_keyword(TokenType.set);
|
|
|
|
_keyword(TokenType.$default);
|
|
|
|
break;
|
|
|
|
case ReferenceAction.cascade:
|
|
|
|
_keyword(TokenType.cascade);
|
|
|
|
break;
|
|
|
|
case ReferenceAction.restrict:
|
|
|
|
_keyword(TokenType.restrict);
|
|
|
|
break;
|
|
|
|
case ReferenceAction.noAction:
|
|
|
|
_keyword(TokenType.no);
|
|
|
|
_keyword(TokenType.action);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (e.onUpdate != null) {
|
|
|
|
_keyword(TokenType.on);
|
|
|
|
_keyword(TokenType.update);
|
2020-12-11 01:53:17 -08:00
|
|
|
referenceAction(e.onUpdate!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
if (e.onDelete != null) {
|
|
|
|
_keyword(TokenType.on);
|
|
|
|
_keyword(TokenType.delete);
|
2020-12-11 01:53:17 -08:00
|
|
|
referenceAction(e.onDelete!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visitNullable(e.deferrable, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitFrameSpec(FrameSpec e, void arg) {
|
|
|
|
void frameBoundary(FrameBoundary boundary) {
|
|
|
|
void precedingOrFollowing(bool preceding) {
|
|
|
|
if (boundary.isUnbounded) {
|
|
|
|
_keyword(TokenType.unbounded);
|
2020-08-22 03:36:36 -07:00
|
|
|
} else {
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(boundary.offset!, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
2020-08-22 03:36:36 -07:00
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(preceding ? TokenType.preceding : TokenType.following);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (boundary.isCurrentRow) {
|
|
|
|
_keyword(TokenType.current);
|
|
|
|
_keyword(TokenType.row);
|
|
|
|
} else if (boundary.preceding) {
|
|
|
|
precedingOrFollowing(true);
|
|
|
|
} else {
|
|
|
|
precedingOrFollowing(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(const {
|
|
|
|
FrameType.range: TokenType.range,
|
|
|
|
FrameType.rows: TokenType.rows,
|
|
|
|
FrameType.groups: TokenType.groups,
|
2020-12-11 01:53:17 -08:00
|
|
|
}[e.type!]!);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
2020-12-11 01:53:17 -08:00
|
|
|
_keyword(TokenType.between);
|
|
|
|
frameBoundary(e.start);
|
|
|
|
_keyword(TokenType.and);
|
|
|
|
frameBoundary(e.end);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.excludeMode != null) {
|
|
|
|
_keyword(TokenType.exclude);
|
2020-12-11 01:53:17 -08:00
|
|
|
switch (e.excludeMode!) {
|
2020-08-21 07:44:17 -07:00
|
|
|
case ExcludeMode.noOthers:
|
|
|
|
_keyword(TokenType.no);
|
|
|
|
_keyword(TokenType.others);
|
|
|
|
break;
|
|
|
|
case ExcludeMode.currentRow:
|
|
|
|
_keyword(TokenType.current);
|
|
|
|
_keyword(TokenType.row);
|
|
|
|
break;
|
|
|
|
case ExcludeMode.group:
|
|
|
|
_keyword(TokenType.group);
|
|
|
|
break;
|
|
|
|
case ExcludeMode.ties:
|
|
|
|
_keyword(TokenType.ties);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitFunction(FunctionExpression e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.name);
|
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.parameters, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitGroupBy(GroupBy e, void arg) {
|
|
|
|
_keyword(TokenType.group);
|
|
|
|
_keyword(TokenType.by);
|
|
|
|
|
|
|
|
_join(e.by, ',');
|
|
|
|
|
|
|
|
if (e.having != null) {
|
|
|
|
_keyword(TokenType.having);
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(e.having!, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-08 13:53:57 -07:00
|
|
|
@override
|
|
|
|
void visitIndexedColumn(IndexedColumn e, void arg) {
|
|
|
|
visit(e.expression, arg);
|
|
|
|
_orderingMode(e.ordering);
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitInExpression(InExpression e, void arg) {
|
|
|
|
visit(e.left, arg);
|
2021-01-23 12:03:03 -08:00
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
if (e.not) {
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
}
|
2021-01-23 12:03:03 -08:00
|
|
|
_keyword(TokenType.$in);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
visit(e.inside, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitInsertStatement(InsertStatement e, void arg) {
|
|
|
|
visitNullable(e.withClause, arg);
|
|
|
|
|
|
|
|
final mode = e.mode;
|
|
|
|
if (mode == InsertMode.insert) {
|
|
|
|
_keyword(TokenType.insert);
|
|
|
|
} else if (mode == InsertMode.replace) {
|
|
|
|
_keyword(TokenType.replace);
|
|
|
|
} else {
|
|
|
|
_keyword(TokenType.insert);
|
2020-08-22 03:36:36 -07:00
|
|
|
_keyword(TokenType.or);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
_keyword(const {
|
|
|
|
InsertMode.insertOrReplace: TokenType.replace,
|
|
|
|
InsertMode.insertOrRollback: TokenType.rollback,
|
|
|
|
InsertMode.insertOrAbort: TokenType.abort,
|
|
|
|
InsertMode.insertOrFail: TokenType.fail,
|
|
|
|
InsertMode.insertOrIgnore: TokenType.ignore,
|
2020-12-11 01:53:17 -08:00
|
|
|
}[mode]!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(TokenType.into);
|
2021-03-13 07:07:12 -08:00
|
|
|
visit(e.table, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.targetColumns.isNotEmpty) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
_join(e.targetColumns, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.source, arg);
|
2021-05-06 13:35:57 -07:00
|
|
|
visitNullable(e.upsert, arg);
|
2021-03-30 14:24:28 -07:00
|
|
|
visitNullable(e.returning, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitInsertTriggerTarget(InsertTarget e, void arg) {
|
|
|
|
_keyword(TokenType.insert);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitInvalidStatement(InvalidStatement e, void arg) {
|
|
|
|
throw UnsupportedError(
|
|
|
|
'InvalidStatement does not have a textual representation');
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitIsExpression(IsExpression e, void arg) {
|
|
|
|
visit(e.left, arg);
|
|
|
|
_keyword(TokenType.$is);
|
|
|
|
|
2022-05-26 13:24:41 -07:00
|
|
|
// Avoid writing `DISTINCT FROM`, but be aware that it effectively negates
|
|
|
|
// the generated `IS` again.
|
|
|
|
final negated = e.negated ^ e.distinctFromSyntax;
|
|
|
|
|
|
|
|
if (negated) {
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.not);
|
|
|
|
}
|
|
|
|
visit(e.right, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitIsNullExpression(IsNullExpression e, void arg) {
|
|
|
|
visit(e.operand, arg);
|
|
|
|
|
|
|
|
if (e.negated) {
|
2020-08-22 03:36:36 -07:00
|
|
|
_keyword(TokenType.notNull);
|
|
|
|
} else {
|
|
|
|
_keyword(TokenType.isNull);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitJoin(Join e, void arg) {
|
2022-05-26 14:23:02 -07:00
|
|
|
visit(e.operator, null);
|
|
|
|
visit(e.query, null);
|
|
|
|
|
|
|
|
final constraint = e.constraint;
|
|
|
|
if (constraint is OnConstraint) {
|
|
|
|
_keyword(TokenType.on);
|
|
|
|
visit(constraint.expression, arg);
|
|
|
|
} else if (constraint is UsingConstraint) {
|
|
|
|
_keyword(TokenType.using);
|
|
|
|
symbol('(${constraint.columnNames.join(', ')})');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitJoinOperator(JoinOperator e, void arg) {
|
|
|
|
if (e.operator == JoinOperatorKind.comma) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(',');
|
2020-08-21 07:44:17 -07:00
|
|
|
} else {
|
|
|
|
if (e.natural) {
|
|
|
|
_keyword(TokenType.natural);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (e.operator) {
|
2022-05-26 14:23:02 -07:00
|
|
|
case JoinOperatorKind.none:
|
2020-08-21 07:44:17 -07:00
|
|
|
break;
|
2022-05-26 14:23:02 -07:00
|
|
|
case JoinOperatorKind.comma:
|
2020-08-21 07:44:17 -07:00
|
|
|
throw AssertionError("Can't happen");
|
2022-05-26 14:23:02 -07:00
|
|
|
case JoinOperatorKind.left:
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.left);
|
|
|
|
break;
|
2022-05-26 14:23:02 -07:00
|
|
|
case JoinOperatorKind.right:
|
|
|
|
_keyword(TokenType.right);
|
2020-08-21 07:44:17 -07:00
|
|
|
break;
|
2022-05-26 14:23:02 -07:00
|
|
|
case JoinOperatorKind.full:
|
|
|
|
_keyword(TokenType.full);
|
|
|
|
break;
|
|
|
|
case JoinOperatorKind.inner:
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.inner);
|
|
|
|
break;
|
2022-05-26 14:23:02 -07:00
|
|
|
case JoinOperatorKind.cross:
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.cross);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-05-26 14:23:02 -07:00
|
|
|
if (e.outer) {
|
|
|
|
_keyword(TokenType.outer);
|
|
|
|
}
|
|
|
|
_keyword(TokenType.join);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitJoinClause(JoinClause e, void arg) {
|
|
|
|
visit(e.primary, arg);
|
|
|
|
visitList(e.joins, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitLimit(Limit e, void arg) {
|
|
|
|
_keyword(TokenType.limit);
|
|
|
|
visit(e.count, arg);
|
|
|
|
|
|
|
|
if (e.offset != null) {
|
|
|
|
_keyword(TokenType.offset);
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(e.offset!, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitNamedVariable(ColonNamedVariable e, void arg) {
|
2020-08-22 03:36:36 -07:00
|
|
|
// Note: The name already starts with the colon
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(e.name, spaceBefore: true, spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitNullLiteral(NullLiteral e, void arg) {
|
|
|
|
_keyword(TokenType.$null);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitNumberedVariable(NumberedVariable e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('?', spaceBefore: true, spaceAfter: e.explicitIndex == null);
|
2020-08-21 07:44:17 -07:00
|
|
|
if (e.explicitIndex != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(e.explicitIndex.toString(), spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitNumericLiteral(NumericLiteral e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(e.value.toString(), spaceBefore: true, spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitOrderBy(OrderBy e, void arg) {
|
|
|
|
_keyword(TokenType.order);
|
|
|
|
_keyword(TokenType.by);
|
|
|
|
_join(e.terms, ',');
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitOrderingTerm(OrderingTerm e, void arg) {
|
|
|
|
visit(e.expression, arg);
|
|
|
|
_orderingMode(e.orderingMode);
|
|
|
|
|
|
|
|
if (e.nulls != null) {
|
|
|
|
_keyword(TokenType.nulls);
|
|
|
|
|
|
|
|
_keyword(const {
|
|
|
|
OrderingBehaviorForNulls.first: TokenType.first,
|
|
|
|
OrderingBehaviorForNulls.last: TokenType.last,
|
2020-12-11 01:53:17 -08:00
|
|
|
}[e.nulls!]!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitParentheses(Parentheses e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.expression, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
2021-09-08 13:53:57 -07:00
|
|
|
@override
|
|
|
|
void visitRaiseExpression(RaiseExpression e, void arg) {
|
|
|
|
_keyword(TokenType.raise);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2021-09-08 13:53:57 -07:00
|
|
|
_keyword(const {
|
|
|
|
RaiseKind.ignore: TokenType.ignore,
|
|
|
|
RaiseKind.rollback: TokenType.rollback,
|
|
|
|
RaiseKind.abort: TokenType.abort,
|
|
|
|
RaiseKind.fail: TokenType.fail,
|
|
|
|
}[e.raiseKind]!);
|
|
|
|
|
|
|
|
if (e.errorMessage != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(',', spaceAfter: true);
|
2021-09-08 13:53:57 -07:00
|
|
|
_stringLiteral(e.errorMessage!);
|
|
|
|
}
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitReference(Reference e, void arg) {
|
2021-07-22 12:32:53 -07:00
|
|
|
var didWriteSpaceBefore = false;
|
|
|
|
|
|
|
|
if (e.schemaName != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.schemaName!, spaceAfter: false);
|
|
|
|
symbol('.');
|
2021-07-22 12:32:53 -07:00
|
|
|
didWriteSpaceBefore = true;
|
|
|
|
}
|
2021-02-11 09:38:41 -08:00
|
|
|
if (e.entityName != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.entityName!,
|
2021-07-22 12:32:53 -07:00
|
|
|
spaceAfter: false, spaceBefore: !didWriteSpaceBefore);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('.');
|
2021-07-22 12:32:53 -07:00
|
|
|
didWriteSpaceBefore = true;
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.columnName,
|
2021-07-22 12:32:53 -07:00
|
|
|
spaceAfter: true, spaceBefore: !didWriteSpaceBefore);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
2021-03-13 11:41:14 -08:00
|
|
|
@override
|
|
|
|
void visitReturning(Returning e, void arg) {
|
|
|
|
_keyword(TokenType.returning);
|
|
|
|
_join(e.columns, ',');
|
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitSelectInsertSource(SelectInsertSource e, void arg) {
|
|
|
|
visit(e.stmt, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitSelectStatement(SelectStatement e, void arg) {
|
|
|
|
visitNullable(e.withClause, arg);
|
|
|
|
_keyword(TokenType.select);
|
|
|
|
if (e.distinct) {
|
|
|
|
_keyword(TokenType.distinct);
|
|
|
|
}
|
|
|
|
|
2022-01-28 12:55:03 -08:00
|
|
|
_join(e.columns, ',');
|
2020-08-21 07:44:17 -07:00
|
|
|
|
2021-03-13 03:02:35 -08:00
|
|
|
_from(e.from);
|
2020-08-21 07:44:17 -07:00
|
|
|
_where(e.where);
|
|
|
|
visitNullable(e.groupBy, arg);
|
|
|
|
if (e.windowDeclarations.isNotEmpty) {
|
|
|
|
_keyword(TokenType.window);
|
|
|
|
|
|
|
|
var isFirst = true;
|
|
|
|
for (final declaration in e.windowDeclarations) {
|
|
|
|
if (!isFirst) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(',', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(declaration.name);
|
2020-08-22 03:36:36 -07:00
|
|
|
_keyword(TokenType.as);
|
|
|
|
|
|
|
|
visit(declaration.definition, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
isFirst = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
visitNullable(e.orderBy, arg);
|
|
|
|
visitNullable(e.limit, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitSelectStatementAsSource(SelectStatementAsSource e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.statement, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.as != null) {
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.as!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitSetComponent(SetComponent e, void arg) {
|
|
|
|
visit(e.column, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('=', spaceBefore: true, spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.expression, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitStarFunctionParameter(StarFunctionParameter e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('*', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
2021-09-08 13:53:57 -07:00
|
|
|
@override
|
|
|
|
void visitStarResultColumn(StarResultColumn e, void arg) {
|
|
|
|
if (e.tableName != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.tableName!);
|
|
|
|
symbol('.');
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('*', spaceAfter: true, spaceBefore: e.tableName == null);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
@override
|
|
|
|
void visitStringComparison(StringComparisonExpression e, void arg) {
|
|
|
|
visit(e.left, arg);
|
|
|
|
if (e.not) {
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(e.operator.type);
|
|
|
|
visit(e.right, arg);
|
|
|
|
|
|
|
|
if (e.escape != null) {
|
|
|
|
_keyword(TokenType.escape);
|
2020-12-11 01:53:17 -08:00
|
|
|
visit(e.escape!, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitStringLiteral(StringLiteral e, void arg) {
|
|
|
|
_stringLiteral(e.value);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitSubQuery(SubQuery e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.select, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitTableConstraint(TableConstraint e, void arg) {
|
|
|
|
if (e.name != null) {
|
|
|
|
_keyword(TokenType.constraint);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.name!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (e is KeyClause) {
|
|
|
|
if (e.isPrimaryKey) {
|
|
|
|
_keyword(TokenType.primary);
|
|
|
|
_keyword(TokenType.key);
|
|
|
|
} else {
|
|
|
|
_keyword(TokenType.unique);
|
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2021-01-18 05:44:05 -08:00
|
|
|
_join(e.columns, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
_conflictClause(e.onConflict);
|
|
|
|
} else if (e is CheckTable) {
|
|
|
|
_keyword(TokenType.check);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.expression, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
} else if (e is ForeignKeyTableConstraint) {
|
|
|
|
_keyword(TokenType.foreign);
|
|
|
|
_keyword(TokenType.key);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
_join(e.columns, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.clause, arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitTableReference(TableReference e, void arg) {
|
2021-07-22 12:32:53 -07:00
|
|
|
if (e.schemaName != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.schemaName!, spaceAfter: false);
|
|
|
|
symbol('.');
|
2021-07-22 12:32:53 -07:00
|
|
|
}
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.tableName, spaceBefore: e.schemaName == null);
|
2021-07-22 12:32:53 -07:00
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
if (e.as != null) {
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.as!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitTableValuedFunction(TableValuedFunction e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.name);
|
|
|
|
symbol('(');
|
2020-08-21 07:44:17 -07:00
|
|
|
visit(e.parameters, arg);
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')');
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.as != null) {
|
|
|
|
_keyword(TokenType.as);
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.as!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitTimeConstantLiteral(TimeConstantLiteral e, void arg) {
|
|
|
|
switch (e.kind) {
|
|
|
|
case TimeConstantKind.currentTime:
|
|
|
|
_keyword(TokenType.currentTime);
|
|
|
|
break;
|
|
|
|
case TimeConstantKind.currentDate:
|
|
|
|
_keyword(TokenType.currentDate);
|
|
|
|
break;
|
|
|
|
case TimeConstantKind.currentTimestamp:
|
|
|
|
_keyword(TokenType.currentTimestamp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitTuple(Tuple e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
_join(e.expressions, ',');
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitUnaryExpression(UnaryExpression e, void arg) {
|
|
|
|
switch (e.operator.type) {
|
|
|
|
case TokenType.minus:
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('-', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
break;
|
|
|
|
case TokenType.plus:
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('+', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
break;
|
|
|
|
case TokenType.tilde:
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('~', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
break;
|
|
|
|
case TokenType.not:
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw AssertionError('Unknown unary operator: ${e.operator}');
|
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.inner, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitUpdateStatement(UpdateStatement e, void arg) {
|
|
|
|
visitNullable(e.withClause, arg);
|
|
|
|
_keyword(TokenType.update);
|
|
|
|
|
|
|
|
if (e.or != null) {
|
|
|
|
_keyword(TokenType.or);
|
|
|
|
|
|
|
|
_keyword(const {
|
|
|
|
FailureMode.rollback: TokenType.rollback,
|
|
|
|
FailureMode.abort: TokenType.abort,
|
|
|
|
FailureMode.replace: TokenType.replace,
|
|
|
|
FailureMode.fail: TokenType.fail,
|
|
|
|
FailureMode.ignore: TokenType.ignore,
|
2020-12-11 01:53:17 -08:00
|
|
|
}[e.or!]!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visit(e.table, arg);
|
|
|
|
_keyword(TokenType.set);
|
|
|
|
_join(e.set, ',');
|
2021-03-13 03:02:35 -08:00
|
|
|
_from(e.from);
|
2020-08-21 07:44:17 -07:00
|
|
|
_where(e.where);
|
2021-03-30 14:24:28 -07:00
|
|
|
visitNullable(e.returning, arg);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitUpdateTriggerTarget(UpdateTarget e, void arg) {
|
|
|
|
_keyword(TokenType.update);
|
|
|
|
if (e.columnNames.isNotEmpty) {
|
|
|
|
_keyword(TokenType.of);
|
|
|
|
_join(e.columnNames, ',');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitUpsertClause(UpsertClause e, void arg) {
|
2021-03-13 07:07:12 -08:00
|
|
|
_join(e.entries, '');
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitUpsertClauseEntry(UpsertClauseEntry e, void arg) {
|
2020-08-21 07:44:17 -07:00
|
|
|
_keyword(TokenType.on);
|
|
|
|
_keyword(TokenType.conflict);
|
|
|
|
|
|
|
|
if (e.onColumns != null) {
|
2022-02-21 09:43:48 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-12-11 01:53:17 -08:00
|
|
|
_join(e.onColumns!, ',');
|
2022-02-21 09:43:48 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
|
|
|
|
2020-08-21 07:44:17 -07:00
|
|
|
_where(e.where);
|
|
|
|
}
|
|
|
|
|
|
|
|
_keyword(TokenType.$do);
|
|
|
|
visit(e.action, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitValuesSelectStatement(ValuesSelectStatement e, void arg) {
|
|
|
|
_keyword(TokenType.$values);
|
|
|
|
_join(e.values, ',');
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitValuesSource(ValuesSource e, void arg) {
|
|
|
|
_keyword(TokenType.$values);
|
|
|
|
_join(e.values, ',');
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitWhen(WhenComponent e, void arg) {
|
|
|
|
_keyword(TokenType.when);
|
|
|
|
visit(e.when, arg);
|
|
|
|
_keyword(TokenType.then);
|
|
|
|
visit(e.then, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitWindowDefinition(WindowDefinition e, void arg) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol('(', spaceBefore: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
|
|
|
|
if (e.baseWindowName != null) {
|
2021-11-11 05:40:41 -08:00
|
|
|
identifier(e.baseWindowName!);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (e.partitionBy.isNotEmpty) {
|
|
|
|
_keyword(TokenType.partition);
|
|
|
|
_keyword(TokenType.by);
|
|
|
|
_join(e.partitionBy, ',');
|
|
|
|
}
|
|
|
|
|
|
|
|
visitNullable(e.orderBy, arg);
|
|
|
|
visitNullable(e.frameSpec, arg);
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(')', spaceAfter: true);
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void visitWithClause(WithClause e, void arg) {
|
|
|
|
_keyword(TokenType.$with);
|
|
|
|
if (e.recursive) {
|
|
|
|
_keyword(TokenType.recursive);
|
|
|
|
}
|
|
|
|
|
|
|
|
_join(e.ctes, ',');
|
|
|
|
}
|
2021-09-08 13:53:57 -07:00
|
|
|
|
|
|
|
void _conflictClause(ConflictClause? clause) {
|
|
|
|
if (clause != null) {
|
|
|
|
_keyword(TokenType.on);
|
|
|
|
_keyword(TokenType.conflict);
|
|
|
|
|
|
|
|
_keyword(const {
|
|
|
|
ConflictClause.rollback: TokenType.rollback,
|
|
|
|
ConflictClause.abort: TokenType.abort,
|
|
|
|
ConflictClause.fail: TokenType.fail,
|
|
|
|
ConflictClause.ignore: TokenType.ignore,
|
|
|
|
ConflictClause.replace: TokenType.replace,
|
|
|
|
}[clause]!);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _from(Queryable? from) {
|
|
|
|
if (from != null) {
|
|
|
|
_keyword(TokenType.from);
|
|
|
|
visit(from, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
/// Writes an identifier, escaping it if necessary.
|
|
|
|
void identifier(String identifier,
|
2021-09-08 13:53:57 -07:00
|
|
|
{bool spaceBefore = true, bool spaceAfter = true}) {
|
2022-01-28 12:55:03 -08:00
|
|
|
if (isKeywordLexeme(identifier) || _notAKeywordRegex.hasMatch(identifier)) {
|
2021-09-08 13:53:57 -07:00
|
|
|
identifier = '"$identifier"';
|
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(identifier, spaceBefore: spaceBefore, spaceAfter: spaceAfter);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void _ifNotExists(bool ifNotExists) {
|
|
|
|
if (ifNotExists) {
|
|
|
|
_keyword(TokenType.$if);
|
|
|
|
_keyword(TokenType.not);
|
|
|
|
_keyword(TokenType.exists);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _join(Iterable<AstNode> nodes, String separatingSymbol) {
|
|
|
|
var isFirst = true;
|
|
|
|
|
|
|
|
for (final node in nodes) {
|
|
|
|
if (!isFirst) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(separatingSymbol, spaceAfter: true);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visit(node, null);
|
|
|
|
isFirst = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _keyword(TokenType type) {
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol(reverseKeywords[type]!, spaceAfter: true, spaceBefore: true);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void _orderingMode(OrderingMode? mode) {
|
|
|
|
if (mode != null) {
|
|
|
|
_keyword(const {
|
|
|
|
OrderingMode.ascending: TokenType.asc,
|
|
|
|
OrderingMode.descending: TokenType.desc,
|
|
|
|
}[mode]!);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _space() => buffer.writeCharCode($space);
|
|
|
|
|
|
|
|
void _stringLiteral(String content) {
|
|
|
|
final escapedChars = content.replaceAll("'", "''");
|
2021-11-11 05:40:41 -08:00
|
|
|
symbol("'$escapedChars'", spaceBefore: true, spaceAfter: true);
|
2021-09-08 13:53:57 -07:00
|
|
|
}
|
|
|
|
|
2021-11-11 05:40:41 -08:00
|
|
|
/// Writes the [lexeme], unchanged.
|
|
|
|
void symbol(String lexeme,
|
2021-09-08 13:53:57 -07:00
|
|
|
{bool spaceBefore = false, bool spaceAfter = false}) {
|
|
|
|
if (needsSpace && spaceBefore) {
|
|
|
|
_space();
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer.write(lexeme);
|
|
|
|
needsSpace = spaceAfter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _where(Expression? where) {
|
|
|
|
if (where != null) {
|
|
|
|
_keyword(TokenType.where);
|
|
|
|
visit(where, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Defines the [toSql] extension method that turns ast nodes into a compatible
|
|
|
|
/// textual representation.
|
|
|
|
///
|
|
|
|
/// Parsing the output of [toSql] will result in an equal AST.
|
|
|
|
extension NodeToText on AstNode {
|
|
|
|
/// Obtains a textual representation for AST nodes.
|
|
|
|
///
|
|
|
|
/// Parsing the output of [toSql] will result in an equal AST. Since only the
|
|
|
|
/// AST is used, the output will not contain comments. It's possible for the
|
|
|
|
/// output to have more than just whitespace changes if there are multiple
|
|
|
|
/// ways to represent an equivalent node (e.g. the no-op `FOR EACH ROW` on
|
|
|
|
/// triggers).
|
|
|
|
String toSql() {
|
2021-11-11 05:40:41 -08:00
|
|
|
final builder = NodeSqlBuilder(null);
|
2021-09-08 13:53:57 -07:00
|
|
|
builder.visit(this, null);
|
|
|
|
return builder.buffer.toString();
|
|
|
|
}
|
2020-08-21 07:44:17 -07:00
|
|
|
}
|
2022-01-28 12:55:03 -08:00
|
|
|
|
2022-01-29 12:50:33 -08:00
|
|
|
// Allow 0, 1, 2, 3 in https://github.com/sqlite/sqlite/blob/665b6b6b35f10a46bb72377ade7c2e5d8ea42cb3/src/tokenize.c#L62-L80
|
|
|
|
final _notAKeywordRegex = RegExp('[^A-Za-z_0-9]');
|