drift/sqlparser/lib/utils/node_to_text.dart

1421 lines
34 KiB
Dart
Raw Normal View History

/// 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';
class NodeSqlBuilder extends AstVisitor<void, void> {
final StringSink buffer;
/// Whether we need to insert a space before writing the next identifier.
bool needsSpace = false;
NodeSqlBuilder([StringSink? buffer]) : buffer = buffer ?? StringBuffer();
/// Writes a space character if [needsSpace] is set.
///
/// This also resets [needsSpace] to `false`.
void spaceIfNeeded() {
if (needsSpace) {
needsSpace = false;
_space();
}
}
bool isKeyword(String lexeme) {
return isKeywordLexeme(lexeme);
}
@override
2022-03-08 01:58:40 -08:00
void visitAggregateFunctionInvocation(
AggregateFunctionInvocation e, void arg) {
symbol(e.name, spaceBefore: true);
symbol('(');
visit(e.parameters, arg);
symbol(')');
if (e.filter != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.filter);
symbol('(', spaceBefore: true);
2023-01-06 06:10:03 -08:00
keyword(TokenType.where);
2020-12-11 01:53:17 -08:00
visit(e.filter!, arg);
symbol(')', spaceAfter: true);
}
2022-03-08 01:58:40 -08:00
}
@override
void visitWindowFunctionInvocation(WindowFunctionInvocation e, void arg) {
visitAggregateFunctionInvocation(e, arg);
if (e.windowDefinition != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.over);
2020-12-11 01:53:17 -08:00
visit(e.windowDefinition!, arg);
} else if (e.windowName != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.over);
identifier(e.windowName!);
}
}
2021-09-08 13:53:57 -07:00
@override
void visitBeginTransaction(BeginTransactionStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.begin);
2021-09-08 13:53:57 -07:00
switch (e.mode) {
case TransactionMode.none:
break;
case TransactionMode.deferred:
2023-01-06 06:10:03 -08:00
keyword(TokenType.deferred);
2021-09-08 13:53:57 -07:00
break;
case TransactionMode.immediate:
2023-01-06 06:10:03 -08:00
keyword(TokenType.immediate);
2021-09-08 13:53:57 -07:00
break;
case TransactionMode.exclusive:
2023-01-06 06:10:03 -08:00
keyword(TokenType.exclusive);
2021-09-08 13:53:57 -07:00
break;
}
}
@override
void visitBetweenExpression(BetweenExpression e, void arg) {
visit(e.check, arg);
if (e.not) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.between);
visit(e.lower, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.and);
visit(e.upper, arg);
}
@override
void visitBinaryExpression(BinaryExpression e, void arg) {
visit(e.left, arg);
final operatorSymbol = const {
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: '&',
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: '->>',
}[e.operator.type];
if (operatorSymbol != null) {
symbol(operatorSymbol, spaceBefore: true, spaceAfter: true);
} else {
2023-01-06 06:10:03 -08:00
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);
symbol(';');
2021-09-09 09:05:03 -07:00
}
}
@override
void visitBlock(Block block, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.begin);
2021-09-09 09:05:03 -07:00
_writeStatements(block.statements);
2023-01-06 06:10:03 -08:00
keyword(TokenType.end);
}
@override
void visitBooleanLiteral(BooleanLiteral e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(e.value ? TokenType.$true : TokenType.$false);
}
@override
void visitCaseExpression(CaseExpression e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$case);
visitNullable(e.base, arg);
visitList(e.whens, arg);
final elseExpr = e.elseExpr;
if (elseExpr != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$else);
visit(elseExpr, arg);
}
2020-08-22 03:36:36 -07:00
2023-01-06 06:10:03 -08:00
keyword(TokenType.end);
}
@override
void visitCastExpression(CastExpression e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.cast);
symbol('(');
visit(e.operand, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
symbol(e.typeName, spaceBefore: true);
symbol(')', spaceAfter: true);
}
@override
void visitCollateExpression(CollateExpression e, void arg) {
visit(e.inner, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.collate);
identifier(e.collation);
}
@override
void visitColumnConstraint(ColumnConstraint e, void arg) {
if (e.name != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.constraint);
identifier(e.name!);
}
2023-01-05 14:04:36 -08:00
e.when(
primaryKey: (primaryKey) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.primary);
keyword(TokenType.key);
2023-01-05 14:04:36 -08:00
_orderingMode(primaryKey.mode);
_conflictClause(primaryKey.onConflict);
2023-01-06 06:10:03 -08:00
if (primaryKey.autoIncrement) keyword(TokenType.autoincrement);
2023-01-05 14:04:36 -08:00
},
notNull: (notNull) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
keyword(TokenType.$null);
2023-01-05 14:04:36 -08:00
_conflictClause(notNull.onConflict);
},
unique: (unique) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.unique);
2023-01-05 14:04:36 -08:00
_conflictClause(unique.onConflict);
},
check: (check) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.check);
symbol('(', spaceBefore: true);
2023-01-05 14:04:36 -08:00
visit(check.expression, arg);
symbol(')', spaceAfter: true);
2023-01-05 14:04:36 -08:00
},
isDefault: (def) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$default);
2023-01-05 14:04:36 -08:00
final expr = def.expression;
if (expr is Literal) {
visit(expr, arg);
} else {
symbol('(', spaceBefore: true);
visit(expr, arg);
symbol(')', spaceAfter: true);
}
},
collate: (collate) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.collate);
2023-01-05 14:04:36 -08:00
identifier(collate.collation);
},
foreignKey: (foreignKey) {
visit(foreignKey.clause, arg);
},
generatedAs: (generatedAs) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.generated);
keyword(TokenType.always);
keyword(TokenType.as);
2023-01-05 14:04:36 -08:00
symbol('(', spaceBefore: true);
visit(generatedAs.expression, arg);
symbol(')', spaceAfter: true);
2023-01-05 14:04:36 -08:00
if (generatedAs.stored) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.stored);
2023-01-05 14:04:36 -08:00
} else {
2023-01-06 06:10:03 -08:00
keyword(TokenType.virtual);
2023-01-05 14:04:36 -08:00
}
},
mappedBy: (mappedBy) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.mapped);
keyword(TokenType.by);
2023-01-05 14:04:36 -08:00
dartCode(mappedBy.mapper.dartCode);
},
);
}
@override
void visitColumnDefinition(ColumnDefinition e, void arg) {
identifier(e.columnName);
if (e.typeName != null) {
symbol(e.typeName!, spaceAfter: true, spaceBefore: true);
}
visitList(e.constraints, arg);
}
2021-09-08 13:53:57 -07:00
@override
void visitCommitStatement(CommitStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.commit);
2021-09-08 13:53:57 -07:00
}
@override
void visitCommonTableExpression(CommonTableExpression e, void arg) {
identifier(e.cteTableName);
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);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
switch (e.materializationHint) {
case MaterializationHint.notMaterialized:
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
keyword(TokenType.materialized);
break;
case MaterializationHint.materialized:
2023-01-06 06:10:03 -08:00
keyword(TokenType.materialized);
break;
case null:
break;
}
symbol('(', spaceBefore: true);
visit(e.as, arg);
symbol(')', spaceAfter: true);
}
@override
void visitCompoundSelectPart(CompoundSelectPart e, void arg) {
switch (e.mode) {
case CompoundSelectMode.union:
2023-01-06 06:10:03 -08:00
keyword(TokenType.union);
break;
case CompoundSelectMode.unionAll:
2023-01-06 06:10:03 -08:00
keyword(TokenType.union);
keyword(TokenType.all);
break;
case CompoundSelectMode.intersect:
2023-01-06 06:10:03 -08:00
keyword(TokenType.intersect);
break;
case CompoundSelectMode.except:
2023-01-06 06:10:03 -08:00
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) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.create);
if (e.unique) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.unique);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.$index);
_ifNotExists(e.ifNotExists);
identifier(e.indexName);
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
visit(e.on, arg);
symbol('(', spaceBefore: true);
_join(e.columns, ',');
symbol(')', spaceAfter: true);
_where(e.where);
}
@override
void visitCreateTableStatement(CreateTableStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.create);
keyword(TokenType.table);
_ifNotExists(e.ifNotExists);
identifier(e.tableName);
symbol('(');
_join([...e.columns, ...e.tableConstraints], ',');
symbol(')');
if (e.withoutRowId) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.without);
keyword(TokenType.rowid);
}
2021-08-28 07:28:29 -07:00
if (e.isStrict) {
if (e.withoutRowId) symbol(',');
2021-08-28 07:28:29 -07:00
2023-01-06 06:10:03 -08:00
keyword(TokenType.strict);
2021-08-28 07:28:29 -07:00
}
}
@override
void visitCreateTriggerStatement(CreateTriggerStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.create);
keyword(TokenType.trigger);
_ifNotExists(e.ifNotExists);
identifier(e.triggerName);
switch (e.mode) {
case TriggerMode.before:
2023-01-06 06:10:03 -08:00
keyword(TokenType.before);
break;
case TriggerMode.after:
2023-01-06 06:10:03 -08:00
keyword(TokenType.after);
break;
case TriggerMode.insteadOf:
2023-01-06 06:10:03 -08:00
keyword(TokenType.instead);
keyword(TokenType.of);
break;
default:
// Can happen if e.mode == null
break;
}
visit(e.target, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
visit(e.onTable, arg);
if (e.when != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.when);
2020-12-11 01:53:17 -08:00
visit(e.when!, arg);
}
visit(e.action, arg);
}
@override
void visitCreateViewStatement(CreateViewStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.create);
keyword(TokenType.view);
_ifNotExists(e.ifNotExists);
identifier(e.viewName);
if (e.columns != null) {
symbol('(', spaceBefore: true);
symbol(e.columns!.join(','));
symbol(')', spaceAfter: true);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
visit(e.query, arg);
}
@override
void visitCreateVirtualTableStatement(
CreateVirtualTableStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.create);
keyword(TokenType.virtual);
keyword(TokenType.table);
_ifNotExists(e.ifNotExists);
identifier(e.tableName);
2023-01-06 06:10:03 -08:00
keyword(TokenType.using);
identifier(e.moduleName);
symbol('(${e.argumentContent.join(', ')})');
}
@override
void visitDriftSpecificNode(DriftSpecificNode e, void arg) {
if (e is DartPlaceholder) {
2022-10-23 09:13:40 -07:00
e.when(
2023-01-06 06:10:03 -08:00
isLimit: (_) => keyword(TokenType.limit),
2022-10-23 09:13:40 -07:00
isOrderBy: (_) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.order);
keyword(TokenType.by);
2022-10-23 09:13:40 -07:00
},
);
symbol(r'$', spaceBefore: true);
symbol(e.name, spaceAfter: true);
} else if (e is DeclaredStatement) {
identifier(e.identifier.name);
if (e.parameters.isNotEmpty) {
symbol('(');
_join(e.parameters, ',');
symbol(')');
}
2022-12-26 10:17:17 -08:00
visitNullable(e.as, arg);
symbol(':', spaceAfter: true);
visit(e.statement, arg);
symbol(';');
} else if (e is DriftFile) {
for (final stmt in e.statements) {
visit(stmt, arg);
buffer.write('\n');
needsSpace = false;
}
} else if (e is ImportStatement) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.import);
_stringLiteral(e.importedFile);
symbol(';', spaceAfter: true);
} else if (e is StatementParameter) {
if (e is VariableTypeHint) {
2023-01-06 06:10:03 -08:00
if (e.isRequired) keyword(TokenType.required);
visit(e.variable, arg);
final typeName = e.typeName;
if (typeName != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
symbol(typeName, spaceBefore: true, spaceAfter: true);
}
if (e.orNull) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.or);
keyword(TokenType.$null);
}
} else if (e is DartPlaceholderDefaultValue) {
symbol('\$${e.variableName}', spaceAfter: true);
symbol('=', spaceBefore: true, spaceAfter: true);
visit(e.defaultValue, arg);
} else {
throw AssertionError('Unknown StatementParameter: $e');
}
} else if (e is DriftTableName) {
2023-01-06 06:10:03 -08:00
keyword(e.useExistingDartClass ? TokenType.$with : TokenType.as);
identifier(e.overriddenDataClassName);
} else if (e is NestedStarResultColumn) {
identifier(e.tableName);
symbol('.**', spaceAfter: true);
2022-01-23 08:08:32 -08:00
} else if (e is NestedQueryColumn) {
2022-12-27 08:04:38 -08:00
symbol('LIST(', spaceBefore: true);
2022-01-23 08:08:32 -08:00
visit(e.select, arg);
symbol(')', spaceAfter: true);
2022-12-27 08:04:38 -08:00
if (e.as != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
2022-12-27 08:04:38 -08:00
identifier(e.as!);
}
2021-09-09 09:05:03 -07:00
} else if (e is TransactionBlock) {
visit(e.begin, arg);
_writeStatements(e.innerStatements);
visit(e.commit, arg);
}
}
@override
void visitDefaultValues(DefaultValues e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$default);
keyword(TokenType.$values);
}
@override
void visitDeferrableClause(DeferrableClause e, void arg) {
if (e.not) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.deferrable);
switch (e.declaredInitially) {
case InitialDeferrableMode.deferred:
2023-01-06 06:10:03 -08:00
keyword(TokenType.initially);
keyword(TokenType.deferred);
break;
case InitialDeferrableMode.immediate:
2023-01-06 06:10:03 -08:00
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);
2023-01-06 06:10:03 -08:00
keyword(TokenType.delete);
_from(e.from);
_where(e.where);
visitNullable(e.returning, arg);
}
@override
void visitDeleteTriggerTarget(DeleteTarget e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.delete);
}
@override
void visitDoNothing(DoNothing e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.nothing);
}
@override
void visitDoUpdate(DoUpdate e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.update);
keyword(TokenType.set);
_join(e.set, ',');
_where(e.where);
}
@override
void visitExists(ExistsExpression e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.exists);
symbol('(', spaceBefore: true);
visit(e.select, null);
symbol(')', spaceAfter: true);
}
@override
void visitExpressionFunctionParameters(ExprFunctionParameters e, void arg) {
if (e.distinct) {
2023-01-06 06:10:03 -08:00
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);
2023-01-05 14:04:36 -08:00
visitNullable(e.mappedBy, arg);
2021-09-08 13:53:57 -07:00
if (e.as != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
identifier(e.as!);
2021-09-08 13:53:57 -07:00
}
}
@override
void visitForeignKeyClause(ForeignKeyClause e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.references);
visit(e.foreignTable, arg);
if (e.columnNames.isNotEmpty) {
symbol('(');
_join(e.columnNames, ',');
symbol(')');
}
void referenceAction(ReferenceAction action) {
switch (action) {
case ReferenceAction.setNull:
2023-01-06 06:10:03 -08:00
keyword(TokenType.set);
keyword(TokenType.$null);
break;
case ReferenceAction.setDefault:
2023-01-06 06:10:03 -08:00
keyword(TokenType.set);
keyword(TokenType.$default);
break;
case ReferenceAction.cascade:
2023-01-06 06:10:03 -08:00
keyword(TokenType.cascade);
break;
case ReferenceAction.restrict:
2023-01-06 06:10:03 -08:00
keyword(TokenType.restrict);
break;
case ReferenceAction.noAction:
2023-01-06 06:10:03 -08:00
keyword(TokenType.no);
keyword(TokenType.action);
break;
}
}
if (e.onUpdate != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
keyword(TokenType.update);
2020-12-11 01:53:17 -08:00
referenceAction(e.onUpdate!);
}
if (e.onDelete != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
keyword(TokenType.delete);
2020-12-11 01:53:17 -08:00
referenceAction(e.onDelete!);
}
visitNullable(e.deferrable, arg);
}
@override
void visitFrameSpec(FrameSpec e, void arg) {
void frameBoundary(FrameBoundary boundary) {
void precedingOrFollowing(bool preceding) {
if (boundary.isUnbounded) {
2023-01-06 06:10:03 -08:00
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-22 03:36:36 -07:00
2023-01-06 06:10:03 -08:00
keyword(preceding ? TokenType.preceding : TokenType.following);
}
if (boundary.isCurrentRow) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.current);
keyword(TokenType.row);
} else if (boundary.preceding) {
precedingOrFollowing(true);
} else {
precedingOrFollowing(false);
}
}
2023-01-06 06:10:03 -08:00
keyword(const {
FrameType.range: TokenType.range,
FrameType.rows: TokenType.rows,
FrameType.groups: TokenType.groups,
2020-12-11 01:53:17 -08:00
}[e.type!]!);
2023-01-06 06:10:03 -08:00
keyword(TokenType.between);
2020-12-11 01:53:17 -08:00
frameBoundary(e.start);
2023-01-06 06:10:03 -08:00
keyword(TokenType.and);
2020-12-11 01:53:17 -08:00
frameBoundary(e.end);
if (e.excludeMode != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.exclude);
2020-12-11 01:53:17 -08:00
switch (e.excludeMode!) {
case ExcludeMode.noOthers:
2023-01-06 06:10:03 -08:00
keyword(TokenType.no);
keyword(TokenType.others);
break;
case ExcludeMode.currentRow:
2023-01-06 06:10:03 -08:00
keyword(TokenType.current);
keyword(TokenType.row);
break;
case ExcludeMode.group:
2023-01-06 06:10:03 -08:00
keyword(TokenType.group);
break;
case ExcludeMode.ties:
2023-01-06 06:10:03 -08:00
keyword(TokenType.ties);
break;
}
}
}
@override
void visitFunction(FunctionExpression e, void arg) {
identifier(e.name);
symbol('(');
visit(e.parameters, arg);
symbol(')', spaceAfter: true);
}
@override
void visitGroupBy(GroupBy e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.group);
keyword(TokenType.by);
_join(e.by, ',');
if (e.having != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.having);
2020-12-11 01:53:17 -08:00
visit(e.having!, arg);
}
}
2021-09-08 13:53:57 -07:00
@override
void visitIndexedColumn(IndexedColumn e, void arg) {
visit(e.expression, arg);
_orderingMode(e.ordering);
}
@override
void visitInExpression(InExpression e, void arg) {
visit(e.left, arg);
if (e.not) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.$in);
visit(e.inside, arg);
}
@override
void visitInsertStatement(InsertStatement e, void arg) {
visitNullable(e.withClause, arg);
final mode = e.mode;
if (mode == InsertMode.insert) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.insert);
} else if (mode == InsertMode.replace) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.replace);
} else {
2023-01-06 06:10:03 -08:00
keyword(TokenType.insert);
keyword(TokenType.or);
2023-01-06 06:10:03 -08: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]!);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.into);
visit(e.table, arg);
if (e.targetColumns.isNotEmpty) {
symbol('(', spaceBefore: true);
_join(e.targetColumns, ',');
symbol(')', spaceAfter: true);
}
visit(e.source, arg);
2021-05-06 13:35:57 -07:00
visitNullable(e.upsert, arg);
visitNullable(e.returning, arg);
}
@override
void visitInsertTriggerTarget(InsertTarget e, void arg) {
2023-01-06 06:10:03 -08:00
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);
2023-01-06 06:10:03 -08:00
keyword(TokenType.$is);
// Avoid writing `DISTINCT FROM`, but be aware that it effectively negates
// the generated `IS` again.
final negated = e.negated ^ e.distinctFromSyntax;
if (negated) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
}
visit(e.right, arg);
}
@override
void visitIsNullExpression(IsNullExpression e, void arg) {
visit(e.operand, arg);
if (e.negated) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.notNull);
2020-08-22 03:36:36 -07:00
} else {
2023-01-06 06:10:03 -08:00
keyword(TokenType.isNull);
}
}
@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) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
2022-05-26 14:23:02 -07:00
visit(constraint.expression, arg);
} else if (constraint is UsingConstraint) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.using);
2022-05-26 14:23:02 -07:00
symbol('(${constraint.columnNames.join(', ')})');
}
}
@override
void visitJoinOperator(JoinOperator e, void arg) {
if (e.operator == JoinOperatorKind.comma) {
symbol(',');
} else {
if (e.natural) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.natural);
}
switch (e.operator) {
2022-05-26 14:23:02 -07:00
case JoinOperatorKind.none:
break;
2022-05-26 14:23:02 -07:00
case JoinOperatorKind.comma:
throw AssertionError("Can't happen");
2022-05-26 14:23:02 -07:00
case JoinOperatorKind.left:
2023-01-06 06:10:03 -08:00
keyword(TokenType.left);
break;
2022-05-26 14:23:02 -07:00
case JoinOperatorKind.right:
2023-01-06 06:10:03 -08:00
keyword(TokenType.right);
break;
2022-05-26 14:23:02 -07:00
case JoinOperatorKind.full:
2023-01-06 06:10:03 -08:00
keyword(TokenType.full);
2022-05-26 14:23:02 -07:00
break;
case JoinOperatorKind.inner:
2023-01-06 06:10:03 -08:00
keyword(TokenType.inner);
break;
2022-05-26 14:23:02 -07:00
case JoinOperatorKind.cross:
2023-01-06 06:10:03 -08:00
keyword(TokenType.cross);
break;
}
2022-05-26 14:23:02 -07:00
if (e.outer) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.outer);
2022-05-26 14:23:02 -07:00
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.join);
}
}
@override
void visitJoinClause(JoinClause e, void arg) {
visit(e.primary, arg);
visitList(e.joins, arg);
}
@override
void visitLimit(Limit e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.limit);
visit(e.count, arg);
if (e.offset != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.offset);
2020-12-11 01:53:17 -08:00
visit(e.offset!, arg);
}
}
@override
void visitNamedVariable(ColonNamedVariable e, void arg) {
2020-08-22 03:36:36 -07:00
// Note: The name already starts with the colon
symbol(e.name, spaceBefore: true, spaceAfter: true);
}
@override
void visitNullLiteral(NullLiteral e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$null);
}
@override
void visitNumberedVariable(NumberedVariable e, void arg) {
symbol('?', spaceBefore: true, spaceAfter: e.explicitIndex == null);
if (e.explicitIndex != null) {
symbol(e.explicitIndex.toString(), spaceAfter: true);
}
}
@override
void visitNumericLiteral(NumericLiteral e, void arg) {
symbol(e.value.toString(), spaceBefore: true, spaceAfter: true);
}
@override
void visitOrderBy(OrderBy e, void arg) {
2023-01-06 06:10:03 -08:00
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) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.nulls);
2023-01-06 06:10:03 -08:00
keyword(const {
OrderingBehaviorForNulls.first: TokenType.first,
OrderingBehaviorForNulls.last: TokenType.last,
2020-12-11 01:53:17 -08:00
}[e.nulls!]!);
}
}
@override
void visitParentheses(Parentheses e, void arg) {
symbol('(');
visit(e.expression, arg);
symbol(')');
}
2021-09-08 13:53:57 -07:00
@override
void visitRaiseExpression(RaiseExpression e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.raise);
symbol('(', spaceBefore: true);
2023-01-06 06:10:03 -08:00
keyword(const {
2021-09-08 13:53:57 -07:00
RaiseKind.ignore: TokenType.ignore,
RaiseKind.rollback: TokenType.rollback,
RaiseKind.abort: TokenType.abort,
RaiseKind.fail: TokenType.fail,
}[e.raiseKind]!);
if (e.errorMessage != null) {
symbol(',', spaceAfter: true);
2021-09-08 13:53:57 -07:00
_stringLiteral(e.errorMessage!);
}
symbol(')', spaceAfter: true);
2021-09-08 13:53:57 -07:00
}
@override
void visitReference(Reference e, void arg) {
2021-07-22 12:32:53 -07:00
var didWriteSpaceBefore = false;
if (e.schemaName != null) {
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) {
identifier(e.entityName!,
2021-07-22 12:32:53 -07:00
spaceAfter: false, spaceBefore: !didWriteSpaceBefore);
symbol('.');
2021-07-22 12:32:53 -07:00
didWriteSpaceBefore = true;
}
identifier(e.columnName,
2021-07-22 12:32:53 -07:00
spaceAfter: true, spaceBefore: !didWriteSpaceBefore);
}
2021-03-13 11:41:14 -08:00
@override
void visitReturning(Returning e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.returning);
2021-03-13 11:41:14 -08:00
_join(e.columns, ',');
}
@override
void visitSelectInsertSource(SelectInsertSource e, void arg) {
visit(e.stmt, arg);
}
@override
void visitSelectStatement(SelectStatement e, void arg) {
visitNullable(e.withClause, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.select);
if (e.distinct) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.distinct);
}
_join(e.columns, ',');
_from(e.from);
_where(e.where);
visitNullable(e.groupBy, arg);
if (e.windowDeclarations.isNotEmpty) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.window);
var isFirst = true;
for (final declaration in e.windowDeclarations) {
if (!isFirst) {
symbol(',', spaceAfter: true);
}
identifier(declaration.name);
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
2020-08-22 03:36:36 -07:00
visit(declaration.definition, arg);
isFirst = false;
}
}
visitNullable(e.orderBy, arg);
visitNullable(e.limit, arg);
}
@override
void visitSelectStatementAsSource(SelectStatementAsSource e, void arg) {
symbol('(', spaceBefore: true);
visit(e.statement, arg);
symbol(')', spaceAfter: true);
if (e.as != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
identifier(e.as!);
}
}
@override
void visitSetComponent(SetComponent e, void arg) {
visit(e.column, arg);
symbol('=', spaceBefore: true, spaceAfter: true);
visit(e.expression, arg);
}
@override
void visitStarFunctionParameter(StarFunctionParameter e, void arg) {
symbol('*', spaceAfter: true);
}
2021-09-08 13:53:57 -07:00
@override
void visitStarResultColumn(StarResultColumn e, void arg) {
if (e.tableName != null) {
identifier(e.tableName!);
symbol('.');
2021-09-08 13:53:57 -07:00
}
symbol('*', spaceAfter: true, spaceBefore: e.tableName == null);
2021-09-08 13:53:57 -07:00
}
@override
void visitStringComparison(StringComparisonExpression e, void arg) {
visit(e.left, arg);
if (e.not) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.not);
}
2023-01-06 06:10:03 -08:00
keyword(e.operator.type);
visit(e.right, arg);
if (e.escape != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.escape);
2020-12-11 01:53:17 -08:00
visit(e.escape!, arg);
}
}
@override
void visitStringLiteral(StringLiteral e, void arg) {
_stringLiteral(e.value);
}
@override
void visitSubQuery(SubQuery e, void arg) {
symbol('(', spaceBefore: true);
visit(e.select, arg);
symbol(')', spaceAfter: true);
}
@override
void visitTableConstraint(TableConstraint e, void arg) {
if (e.name != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.constraint);
identifier(e.name!);
}
if (e is KeyClause) {
if (e.isPrimaryKey) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.primary);
keyword(TokenType.key);
} else {
2023-01-06 06:10:03 -08:00
keyword(TokenType.unique);
}
symbol('(');
_join(e.columns, ',');
symbol(')');
_conflictClause(e.onConflict);
} else if (e is CheckTable) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.check);
symbol('(');
visit(e.expression, arg);
symbol(')');
} else if (e is ForeignKeyTableConstraint) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.foreign);
keyword(TokenType.key);
symbol('(');
_join(e.columns, ',');
symbol(')');
visit(e.clause, arg);
}
}
@override
void visitTableReference(TableReference e, void arg) {
2021-07-22 12:32:53 -07:00
if (e.schemaName != null) {
identifier(e.schemaName!, spaceAfter: false);
symbol('.');
2021-07-22 12:32:53 -07:00
}
identifier(e.tableName, spaceBefore: e.schemaName == null);
2021-07-22 12:32:53 -07:00
if (e.as != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
identifier(e.as!);
}
}
@override
void visitTableValuedFunction(TableValuedFunction e, void arg) {
identifier(e.name);
symbol('(');
visit(e.parameters, arg);
symbol(')');
if (e.as != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.as);
identifier(e.as!);
}
}
@override
void visitTimeConstantLiteral(TimeConstantLiteral e, void arg) {
switch (e.kind) {
case TimeConstantKind.currentTime:
2023-01-06 06:10:03 -08:00
keyword(TokenType.currentTime);
break;
case TimeConstantKind.currentDate:
2023-01-06 06:10:03 -08:00
keyword(TokenType.currentDate);
break;
case TimeConstantKind.currentTimestamp:
2023-01-06 06:10:03 -08:00
keyword(TokenType.currentTimestamp);
break;
}
}
@override
void visitTuple(Tuple e, void arg) {
symbol('(', spaceBefore: true);
_join(e.expressions, ',');
symbol(')', spaceAfter: true);
}
@override
void visitUnaryExpression(UnaryExpression e, void arg) {
switch (e.operator.type) {
case TokenType.minus:
symbol('-', spaceBefore: true);
break;
case TokenType.plus:
symbol('+', spaceBefore: true);
break;
case TokenType.tilde:
symbol('~', spaceBefore: true);
break;
case TokenType.not:
2023-01-06 06:10:03 -08:00
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);
2023-01-06 06:10:03 -08:00
keyword(TokenType.update);
if (e.or != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.or);
2023-01-06 06:10:03 -08:00
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!]!);
}
visit(e.table, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.set);
_join(e.set, ',');
_from(e.from);
_where(e.where);
visitNullable(e.returning, arg);
}
@override
void visitUpdateTriggerTarget(UpdateTarget e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.update);
if (e.columnNames.isNotEmpty) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.of);
_join(e.columnNames, ',');
}
}
@override
void visitUpsertClause(UpsertClause e, void arg) {
_join(e.entries, '');
}
@override
void visitUpsertClauseEntry(UpsertClauseEntry e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
keyword(TokenType.conflict);
if (e.onColumns != null) {
symbol('(', spaceBefore: true);
2020-12-11 01:53:17 -08:00
_join(e.onColumns!, ',');
symbol(')', spaceAfter: true);
_where(e.where);
}
2023-01-06 06:10:03 -08:00
keyword(TokenType.$do);
visit(e.action, arg);
}
@override
void visitValuesSelectStatement(ValuesSelectStatement e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$values);
_join(e.values, ',');
}
@override
void visitValuesSource(ValuesSource e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$values);
_join(e.values, ',');
}
@override
void visitWhen(WhenComponent e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.when);
visit(e.when, arg);
2023-01-06 06:10:03 -08:00
keyword(TokenType.then);
visit(e.then, arg);
}
@override
void visitWindowDefinition(WindowDefinition e, void arg) {
symbol('(', spaceBefore: true);
if (e.baseWindowName != null) {
identifier(e.baseWindowName!);
}
if (e.partitionBy.isNotEmpty) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.partition);
keyword(TokenType.by);
_join(e.partitionBy, ',');
}
visitNullable(e.orderBy, arg);
visitNullable(e.frameSpec, arg);
symbol(')', spaceAfter: true);
}
@override
void visitWithClause(WithClause e, void arg) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$with);
if (e.recursive) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.recursive);
}
_join(e.ctes, ',');
}
2021-09-08 13:53:57 -07:00
void _conflictClause(ConflictClause? clause) {
if (clause != null) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.on);
keyword(TokenType.conflict);
2021-09-08 13:53:57 -07:00
2023-01-06 06:10:03 -08:00
keyword(const {
2021-09-08 13:53:57 -07:00
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) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.from);
2021-09-08 13:53:57 -07:00
visit(from, null);
}
}
/// 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}) {
if (isKeyword(identifier) || _notAKeywordRegex.hasMatch(identifier)) {
2021-09-08 13:53:57 -07:00
identifier = '"$identifier"';
}
symbol(identifier, spaceBefore: spaceBefore, spaceAfter: spaceAfter);
2021-09-08 13:53:57 -07:00
}
2023-01-05 14:04:36 -08:00
void dartCode(String code,
{bool spaceBefore = true, bool spaceAfter = true}) {
symbol('`$code`', spaceBefore: spaceBefore, spaceAfter: spaceAfter);
}
2021-09-08 13:53:57 -07:00
void _ifNotExists(bool ifNotExists) {
if (ifNotExists) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.$if);
keyword(TokenType.not);
keyword(TokenType.exists);
2021-09-08 13:53:57 -07:00
}
}
void _join(Iterable<AstNode> nodes, String separatingSymbol) {
var isFirst = true;
for (final node in nodes) {
if (!isFirst) {
symbol(separatingSymbol, spaceAfter: true);
2021-09-08 13:53:57 -07:00
}
visit(node, null);
isFirst = false;
}
}
2023-01-06 06:10:03 -08:00
void keyword(TokenType type) {
symbol(reverseKeywords[type]!, spaceAfter: true, spaceBefore: true);
2021-09-08 13:53:57 -07:00
}
void _orderingMode(OrderingMode? mode) {
if (mode != null) {
2023-01-06 06:10:03 -08:00
keyword(const {
2021-09-08 13:53:57 -07:00
OrderingMode.ascending: TokenType.asc,
OrderingMode.descending: TokenType.desc,
}[mode]!);
}
}
void _space() => buffer.writeCharCode($space);
void _stringLiteral(String content) {
final escapedChars = content.replaceAll("'", "''");
symbol("'$escapedChars'", spaceBefore: true, spaceAfter: true);
2021-09-08 13:53:57 -07: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) {
2023-01-06 06:10:03 -08:00
keyword(TokenType.where);
2021-09-08 13:53:57 -07:00
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() {
final builder = NodeSqlBuilder(null);
2021-09-08 13:53:57 -07:00
builder.visit(this, null);
return builder.buffer.toString();
}
}
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]');