1use squawk_syntax::ast;
2
3fn has_foreign_key_constraint(create_table: &ast::CreateTable) -> bool {
4 if let Some(table_arg_list) = create_table.table_arg_list() {
5 for arg in table_arg_list.args() {
6 match arg {
7 ast::TableArg::TableConstraint(ast::TableConstraint::ForeignKeyConstraint(_)) => {
8 return true;
9 }
10 ast::TableArg::Column(column) => {
11 if column
12 .constraints()
13 .any(|c| matches!(c, ast::ColumnConstraint::ReferencesConstraint(_)))
14 {
15 return true;
16 }
17 }
18 _ => (),
19 }
20 }
21 }
22 false
23}
24
25pub fn possibly_slow_stmt(stmt: &ast::Stmt) -> bool {
27 match stmt {
33 ast::Stmt::CreateTable(create_table) => has_foreign_key_constraint(create_table),
35 | ast::Stmt::AlterAggregate(_)
36 | ast::Stmt::AlterCollation(_)
37 | ast::Stmt::AlterConversion(_)
38 | ast::Stmt::AlterDatabase(_)
39 | ast::Stmt::AlterDefaultPrivileges(_)
40 | ast::Stmt::AlterDomain(_)
41 | ast::Stmt::AlterEventTrigger(_)
42 | ast::Stmt::AlterExtension(_)
43 | ast::Stmt::AlterForeignDataWrapper(_)
44 | ast::Stmt::AlterForeignTable(_)
45 | ast::Stmt::AlterFunction(_)
46 | ast::Stmt::AlterGroup(_)
47 | ast::Stmt::AlterIndex(_)
48 | ast::Stmt::AlterLanguage(_)
49 | ast::Stmt::AlterLargeObject(_)
50 | ast::Stmt::AlterMaterializedView(_)
51 | ast::Stmt::AlterOperator(_)
52 | ast::Stmt::AlterOperatorClass(_)
53 | ast::Stmt::AlterOperatorFamily(_)
54 | ast::Stmt::AlterPolicy(_)
55 | ast::Stmt::AlterPropertyGraph(_)
56 | ast::Stmt::AlterProcedure(_)
57 | ast::Stmt::AlterPublication(_)
58 | ast::Stmt::AlterRole(_)
59 | ast::Stmt::AlterRoutine(_)
60 | ast::Stmt::AlterRule(_)
61 | ast::Stmt::AlterSchema(_)
62 | ast::Stmt::AlterSequence(_)
63 | ast::Stmt::AlterServer(_)
64 | ast::Stmt::AlterStatistics(_)
65 | ast::Stmt::AlterSubscription(_)
66 | ast::Stmt::AlterSystem(_)
67 | ast::Stmt::AlterTable(_)
68 | ast::Stmt::AlterTablespace(_)
69 | ast::Stmt::AlterTextSearchConfiguration(_)
70 | ast::Stmt::AlterTextSearchDictionary(_)
71 | ast::Stmt::AlterTextSearchParser(_)
72 | ast::Stmt::AlterTextSearchTemplate(_)
73 | ast::Stmt::AlterTrigger(_)
74 | ast::Stmt::AlterType(_)
75 | ast::Stmt::AlterUser(_)
76 | ast::Stmt::AlterUserMapping(_)
77 | ast::Stmt::AlterView(_)
78 | ast::Stmt::CreateAccessMethod(_)
79 | ast::Stmt::CreateAggregate(_)
80 | ast::Stmt::CreateCast(_)
81 | ast::Stmt::CreateCollation(_)
82 | ast::Stmt::CreateConversion(_)
83 | ast::Stmt::CreateDatabase(_)
84 | ast::Stmt::CreateDomain(_)
85 | ast::Stmt::CreateEventTrigger(_)
86 | ast::Stmt::CreateExtension(_)
87 | ast::Stmt::CreateForeignDataWrapper(_)
88 | ast::Stmt::CreateForeignTable(_)
89 | ast::Stmt::CreateFunction(_)
90 | ast::Stmt::CreateGroup(_)
91 | ast::Stmt::CreateIndex(_)
92 | ast::Stmt::CreateLanguage(_)
93 | ast::Stmt::CreateMaterializedView(_)
94 | ast::Stmt::CreateOperator(_)
95 | ast::Stmt::CreateOperatorClass(_)
96 | ast::Stmt::CreateOperatorFamily(_)
97 | ast::Stmt::CreatePolicy(_)
98 | ast::Stmt::CreatePropertyGraph(_)
99 | ast::Stmt::CreateProcedure(_)
100 | ast::Stmt::CreatePublication(_)
101 | ast::Stmt::CreateRole(_)
102 | ast::Stmt::CreateRule(_)
103 | ast::Stmt::CreateSchema(_)
104 | ast::Stmt::CreateSequence(_)
105 | ast::Stmt::CreateServer(_)
106 | ast::Stmt::CreateStatistics(_)
107 | ast::Stmt::CreateSubscription(_)
108 | ast::Stmt::CreateTableAs(_)
109 | ast::Stmt::SelectInto(_)
110 | ast::Stmt::CreateTablespace(_)
111 | ast::Stmt::CreateTextSearchConfiguration(_)
112 | ast::Stmt::CreateTextSearchDictionary(_)
113 | ast::Stmt::CreateTextSearchParser(_)
114 | ast::Stmt::CreateTextSearchTemplate(_)
115 | ast::Stmt::CreateTransform(_)
116 | ast::Stmt::CreateTrigger(_)
117 | ast::Stmt::CreateType(_)
118 | ast::Stmt::CreateUser(_)
119 | ast::Stmt::CreateUserMapping(_)
120 | ast::Stmt::CreateView(_)
121 | ast::Stmt::DropAccessMethod(_)
122 | ast::Stmt::DropAggregate(_)
123 | ast::Stmt::DropCast(_)
124 | ast::Stmt::DropCollation(_)
125 | ast::Stmt::DropConversion(_)
126 | ast::Stmt::DropDatabase(_)
127 | ast::Stmt::DropDomain(_)
128 | ast::Stmt::DropEventTrigger(_)
129 | ast::Stmt::DropExtension(_)
130 | ast::Stmt::DropForeignDataWrapper(_)
131 | ast::Stmt::DropForeignTable(_)
132 | ast::Stmt::DropFunction(_)
133 | ast::Stmt::DropGroup(_)
134 | ast::Stmt::DropIndex(_)
135 | ast::Stmt::DropLanguage(_)
136 | ast::Stmt::DropMaterializedView(_)
137 | ast::Stmt::DropOperator(_)
138 | ast::Stmt::DropOperatorClass(_)
139 | ast::Stmt::DropOperatorFamily(_)
140 | ast::Stmt::DropOwned(_)
141 | ast::Stmt::DropPolicy(_)
142 | ast::Stmt::DropPropertyGraph(_)
143 | ast::Stmt::DropProcedure(_)
144 | ast::Stmt::DropPublication(_)
145 | ast::Stmt::DropRole(_)
146 | ast::Stmt::DropRoutine(_)
147 | ast::Stmt::DropRule(_)
148 | ast::Stmt::DropSchema(_)
149 | ast::Stmt::DropSequence(_)
150 | ast::Stmt::DropServer(_)
151 | ast::Stmt::DropStatistics(_)
152 | ast::Stmt::DropSubscription(_)
153 | ast::Stmt::DropTable(_)
154 | ast::Stmt::DropTablespace(_)
155 | ast::Stmt::DropTextSearchConfig(_)
156 | ast::Stmt::DropTextSearchDict(_)
157 | ast::Stmt::DropTextSearchParser(_)
158 | ast::Stmt::DropTextSearchTemplate(_)
159 | ast::Stmt::DropTransform(_)
160 | ast::Stmt::DropTrigger(_)
161 | ast::Stmt::DropType(_)
162 | ast::Stmt::DropUser(_)
163 | ast::Stmt::DropUserMapping(_)
164 | ast::Stmt::DropView(_)
165 | ast::Stmt::Cluster(_)
167 | ast::Stmt::CommentOn(_)
168 | ast::Stmt::ImportForeignSchema(_)
169 | ast::Stmt::Load(_)
170 | ast::Stmt::Lock(_)
171 | ast::Stmt::Refresh(_)
172 | ast::Stmt::Reindex(_)
173 | ast::Stmt::Truncate(_)
174 | ast::Stmt::Vacuum(_)
175 => true,
176 ast::Stmt::Analyze(_)
177 | ast::Stmt::Begin(_)
178 | ast::Stmt::Call(_)
179 | ast::Stmt::Checkpoint(_)
180 | ast::Stmt::Close(_)
181 | ast::Stmt::Commit(_)
182 | ast::Stmt::Copy(_)
183 | ast::Stmt::Deallocate(_)
184 | ast::Stmt::Declare(_)
185 | ast::Stmt::Delete(_)
186 | ast::Stmt::Discard(_)
187 | ast::Stmt::Do(_)
188 | ast::Stmt::EmptyStmt(_)
189 | ast::Stmt::Execute(_)
190 | ast::Stmt::Explain(_)
191 | ast::Stmt::Fetch(_)
192 | ast::Stmt::Grant(_)
193 | ast::Stmt::Insert(_)
194 | ast::Stmt::Listen(_)
195 | ast::Stmt::Merge(_)
196 | ast::Stmt::Move(_)
197 | ast::Stmt::Notify(_)
198 | ast::Stmt::ParenSelect(_)
199 | ast::Stmt::Prepare(_)
200 | ast::Stmt::PrepareTransaction(_)
201 | ast::Stmt::Reassign(_)
202 | ast::Stmt::ReleaseSavepoint(_)
203 | ast::Stmt::Reset(_)
204 | ast::Stmt::Revoke(_)
205 | ast::Stmt::Repack(_)
206 | ast::Stmt::Rollback(_)
207 | ast::Stmt::Savepoint(_)
208 | ast::Stmt::SecurityLabel(_)
209 | ast::Stmt::Select(_)
210 | ast::Stmt::Set(_)
211 | ast::Stmt::SetConstraints(_)
212 | ast::Stmt::SetRole(_)
213 | ast::Stmt::SetSessionAuth(_)
214 | ast::Stmt::ResetSessionAuth(_)
215 | ast::Stmt::SetTransaction(_)
216 | ast::Stmt::Show(_)
217 | ast::Stmt::Table(_)
218 | ast::Stmt::Unlisten(_)
219 | ast::Stmt::Update(_)
220 | ast::Stmt::Values(_) => false,
221 }
222}
223
224#[cfg(test)]
225mod tests {
226 use super::*;
227 use squawk_syntax::SourceFile;
228
229 #[test]
230 fn alter_table() {
231 let sql = "ALTER TABLE users ADD COLUMN email TEXT;";
232 let file = SourceFile::parse(sql);
233 let stmts = file.tree().stmts().next().unwrap();
234 assert!(possibly_slow_stmt(&stmts));
235 }
236
237 #[test]
238 fn select_into() {
239 let sql = "select 1 a into t;";
240 let file = SourceFile::parse(sql);
241 let stmts = file.tree().stmts().next().unwrap();
242 assert!(possibly_slow_stmt(&stmts));
243 }
244
245 #[test]
246 fn select() {
247 let sql = "select 1;";
248 let file = SourceFile::parse(sql);
249 let stmts = file.tree().stmts().next().unwrap();
250 assert!(!possibly_slow_stmt(&stmts));
251 }
252
253 #[test]
254 fn create_table_without_foreign_key() {
255 let sql = "create table foo (id integer generated by default as identity primary key);";
256 let file = SourceFile::parse(sql);
257 let stmts = file.tree().stmts().next().unwrap();
258 assert!(!possibly_slow_stmt(&stmts));
259 }
260
261 #[test]
262 fn create_table_with_foreign_key() {
263 let sql = "create table foo (id integer, user_id integer references users(id));";
264 let file = SourceFile::parse(sql);
265 let stmts = file.tree().stmts().next().unwrap();
266 assert!(possibly_slow_stmt(&stmts));
267 }
268
269 #[test]
270 fn create_table_with_table_level_foreign_key() {
271 let sql = "create table foo (id integer, user_id integer, foreign key (user_id) references users(id));";
272 let file = SourceFile::parse(sql);
273 let stmts = file.tree().stmts().next().unwrap();
274 assert!(possibly_slow_stmt(&stmts));
275 }
276}