2020-05-04 13:00:41 -07:00
|
|
|
library utils.find_referenced_tables;
|
|
|
|
|
2019-06-28 14:41:27 -07:00
|
|
|
import 'package:sqlparser/sqlparser.dart';
|
|
|
|
|
|
|
|
/// An AST-visitor that walks sql statements and finds all tables referenced in
|
|
|
|
/// them.
|
2019-12-26 03:35:29 -08:00
|
|
|
class ReferencedTablesVisitor extends RecursiveVisitor<void, void> {
|
2019-12-21 05:42:36 -08:00
|
|
|
/// All tables that have been referenced anywhere in this query.
|
2019-06-28 14:41:27 -07:00
|
|
|
final Set<Table> foundTables = {};
|
|
|
|
|
|
|
|
@override
|
2019-12-26 03:35:29 -08:00
|
|
|
void visitReference(Reference e, void arg) {
|
2019-06-30 03:01:46 -07:00
|
|
|
final column = e.resolved;
|
2019-06-28 14:41:27 -07:00
|
|
|
if (column is TableColumn) {
|
|
|
|
foundTables.add(column.table);
|
|
|
|
}
|
|
|
|
|
2019-12-26 03:35:29 -08:00
|
|
|
visitChildren(e, arg);
|
2019-06-28 14:41:27 -07:00
|
|
|
}
|
|
|
|
|
2020-02-10 09:41:02 -08:00
|
|
|
Table /*?*/ _toTableOrNull(ResolvesToResultSet resultSet) {
|
|
|
|
var resolved = resultSet.resultSet;
|
|
|
|
|
|
|
|
while (resolved != null && resolved is TableAlias) {
|
|
|
|
resolved = (resolved as TableAlias).delegate;
|
|
|
|
}
|
|
|
|
|
|
|
|
return resolved is Table ? resolved : null;
|
|
|
|
}
|
|
|
|
|
2019-06-28 14:41:27 -07:00
|
|
|
@override
|
2019-12-26 03:35:29 -08:00
|
|
|
void visitQueryable(Queryable e, void arg) {
|
2019-06-28 14:41:27 -07:00
|
|
|
if (e is TableReference) {
|
2020-02-10 09:41:02 -08:00
|
|
|
final resolved = _toTableOrNull(e.resultSet);
|
|
|
|
if (resolved != null) {
|
2019-11-24 05:46:20 -08:00
|
|
|
foundTables.add(resolved);
|
2019-06-30 03:01:46 -07:00
|
|
|
}
|
2019-06-28 14:41:27 -07:00
|
|
|
}
|
|
|
|
|
2019-12-26 03:35:29 -08:00
|
|
|
visitChildren(e, arg);
|
2019-06-28 14:41:27 -07:00
|
|
|
}
|
|
|
|
}
|
2019-06-30 10:34:54 -07:00
|
|
|
|
2020-05-04 13:00:41 -07:00
|
|
|
enum UpdateKind { insert, update, delete }
|
|
|
|
|
|
|
|
/// A write to a table as found while analyzing a statement.
|
|
|
|
class TableWrite {
|
|
|
|
/// The table that a statement might write to when run.
|
2020-03-04 12:28:08 -08:00
|
|
|
final Table table;
|
2020-05-04 13:00:41 -07:00
|
|
|
|
|
|
|
/// What kind of update was found (e.g. insert, update or delete).
|
2020-03-04 12:28:08 -08:00
|
|
|
final UpdateKind kind;
|
|
|
|
|
2020-05-04 13:00:41 -07:00
|
|
|
TableWrite(this.table, this.kind);
|
|
|
|
|
|
|
|
@override
|
|
|
|
int get hashCode => 37 * table.hashCode + kind.hashCode;
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool operator ==(dynamic other) {
|
|
|
|
return other is TableWrite && other.table == table && other.kind == kind;
|
|
|
|
}
|
2020-03-04 12:28:08 -08:00
|
|
|
}
|
|
|
|
|
2019-06-30 10:34:54 -07:00
|
|
|
/// Finds all tables that could be affected when executing a query. In
|
|
|
|
/// contrast to [ReferencedTablesVisitor], which finds all references, this
|
|
|
|
/// visitor only collects tables a query writes to.
|
2019-12-21 05:42:36 -08:00
|
|
|
class UpdatedTablesVisitor extends ReferencedTablesVisitor {
|
|
|
|
/// All tables that can potentially be updated by this query.
|
|
|
|
///
|
|
|
|
/// Note that this is a subset of [foundTables], since an updating tables
|
|
|
|
/// could reference tables it's not updating (e.g. with `INSERT INTO foo
|
|
|
|
/// SELECT * FROM bar`).
|
2020-05-04 13:00:41 -07:00
|
|
|
final Set<TableWrite> writtenTables = {};
|
2019-06-30 10:34:54 -07:00
|
|
|
|
2020-03-04 12:28:08 -08:00
|
|
|
void _addIfResolved(ResolvesToResultSet r, UpdateKind kind) {
|
2020-02-10 09:41:02 -08:00
|
|
|
final resolved = _toTableOrNull(r);
|
|
|
|
if (resolved != null) {
|
2020-05-04 13:00:41 -07:00
|
|
|
writtenTables.add(TableWrite(resolved, kind));
|
2019-06-30 10:34:54 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2019-12-26 03:35:29 -08:00
|
|
|
void visitDeleteStatement(DeleteStatement e, void arg) {
|
2020-03-04 12:28:08 -08:00
|
|
|
_addIfResolved(e.from, UpdateKind.delete);
|
2019-12-26 03:35:29 -08:00
|
|
|
visitChildren(e, arg);
|
2019-06-30 10:34:54 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2019-12-26 03:35:29 -08:00
|
|
|
void visitUpdateStatement(UpdateStatement e, void arg) {
|
2020-03-04 12:28:08 -08:00
|
|
|
_addIfResolved(e.table, UpdateKind.update);
|
2019-12-26 03:35:29 -08:00
|
|
|
visitChildren(e, arg);
|
2019-06-30 10:34:54 -07:00
|
|
|
}
|
2019-08-29 07:27:02 -07:00
|
|
|
|
|
|
|
@override
|
2019-12-26 03:35:29 -08:00
|
|
|
void visitInsertStatement(InsertStatement e, void arg) {
|
2020-03-04 12:28:08 -08:00
|
|
|
_addIfResolved(e.table, UpdateKind.insert);
|
2019-12-26 03:35:29 -08:00
|
|
|
visitChildren(e, arg);
|
2019-08-29 07:27:02 -07:00
|
|
|
}
|
2019-06-30 10:34:54 -07:00
|
|
|
}
|
2020-05-04 13:00:41 -07:00
|
|
|
|
|
|
|
/// Finds all writes to a table that occur anywhere inside the [root] node or a
|
|
|
|
/// descendant.
|
|
|
|
///
|
|
|
|
/// The [root] node must have all its references resolved. This means that using
|
|
|
|
/// a node obtained via [SqlEngine.parse] directly won't report meaningful
|
|
|
|
/// results. Instead, use [SqlEngine.analyze] or [SqlEngine.analyzeParsed].
|
|
|
|
///
|
|
|
|
/// If you want to find all referenced tables, use [findReferencedTables]. If
|
|
|
|
/// you want to find writes (including their [UpdateKind]) and referenced
|
|
|
|
/// tables, constrct a [UpdatedTablesVisitor] manually.
|
|
|
|
/// Then, let it [RecursiveVisitor.visit] the [root] node. You can now use
|
|
|
|
/// [UpdatedTablesVisitor.writtenTables] and
|
|
|
|
/// [ReferencedTablesVisitor.foundTables]. This will only walk the ast once,
|
|
|
|
/// whereas calling this and [findReferencedTables] will require two walks.
|
|
|
|
///
|
|
|
|
Set<TableWrite> findWrittenTables(AstNode root) {
|
|
|
|
return (UpdatedTablesVisitor()..visit(root, null)).writtenTables;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Finds all tables referenced in [root] or a descendant.
|
|
|
|
///
|
|
|
|
/// The [root] node must have all its references resolved. This means that using
|
|
|
|
/// a node obtained via [SqlEngine.parse] directly won't report meaningful
|
|
|
|
/// results. Instead, use [SqlEngine.analyze] or [SqlEngine.analyzeParsed].
|
|
|
|
///
|
|
|
|
/// If you want to use both [findWrittenTables] and this on the same ast node,
|
|
|
|
/// follow the advice on [findWrittenTables] to only walk the ast once.
|
|
|
|
Set<Table> findReferencedTables(AstNode root) {
|
|
|
|
return (ReferencedTablesVisitor()..visit(root, null)).foundTables;
|
|
|
|
}
|