kaqing 2.0.49__py3-none-any.whl → 2.0.51__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,12 +1,11 @@
1
- from adam.commands.cql.cql_table_completer import CqlTableCompleter
2
1
  from adam.commands.cql.cql_utils import table_names
3
2
  from adam.repl_state import ReplState
3
+ from adam.sql.sql_completer import SqlCompleter
4
4
 
5
5
  def cql_completions(state: ReplState) -> dict[str, any]:
6
- completer = CqlTableCompleter(table_names(state))
7
6
  return {
8
7
  'describe': {
9
8
  'keyspaces': None,
10
- 'table': completer.completions(),
9
+ 'table': {t: None for t in table_names(state)},
11
10
  'tables': None},
12
- } | completer.completions()
11
+ } | SqlCompleter.completions(lambda: table_names(state))
@@ -1,16 +1,8 @@
1
- from adam.table_completer import NestedDict, TableCompleter
1
+ from adam.sql.term_completer import TermCompleter
2
2
 
3
- class CqlTableCompleter(TableCompleter):
4
- def __init__(self, tables: list[str], nested_dict: NestedDict = {}, ignore_case: bool = True):
5
- self._tables = tables
6
- self.ignore_case = ignore_case
7
- self.append_nested_dict(nested_dict)
3
+ class CqlTableNameCompleter(TermCompleter):
4
+ def __init__(self, tables: list[str], ignore_case: bool = True):
5
+ super().__init__(tables, ignore_case=ignore_case)
8
6
 
9
7
  def __repr__(self) -> str:
10
- return "CqlTableCompleter(%r)" % (len(self._tables))
11
-
12
- def nested(self, data: NestedDict) -> 'TableCompleter':
13
- return CqlTableCompleter(self._tables).append_nested_dict(data)
14
-
15
- def tables(self) -> list[str]:
16
- return self._tables
8
+ return "CqlTableCompleter(%r)" % (len(self.words))
@@ -55,6 +55,7 @@ class Cqlsh(Command):
55
55
 
56
56
  def completion(self, state: ReplState) -> dict[str, any]:
57
57
  if state.device != state.C:
58
+ # conflicts with psql completions
58
59
  return {}
59
60
 
60
61
  if state.sts or state.pod:
@@ -59,8 +59,7 @@ class MedusaRestore(Command):
59
59
  now_dtformat = datetime.now().strftime("%Y-%m-%d.%H.%M.%S")
60
60
  rtname = 'medusa-' + now_dtformat + '-restore-from-' + bkname
61
61
  try:
62
- print('SEAN doing')
63
- # CustomResources.create_medusa_restorejob(rtname, bkname, dc, ns)
62
+ CustomResources.create_medusa_restorejob(rtname, bkname, dc, ns)
64
63
  except Exception as e:
65
64
  log2("Exception: MedusaRestore failed: %s\n" % e)
66
65
 
@@ -1,7 +1,7 @@
1
1
  import click
2
2
 
3
3
  from adam.commands.command import Command
4
- from adam.commands.postgres.pg_completions import pg_completions
4
+ from adam.commands.postgres.psql_completions import psql_completions
5
5
  from adam.commands.postgres.postgres_utils import pg_table_names
6
6
  from .postgres_ls import PostgresLs
7
7
  from .postgres_preview import PostgresPreview
@@ -66,13 +66,14 @@ class Postgres(Command):
66
66
 
67
67
  def completion(self, state: ReplState):
68
68
  if state.device != state.P:
69
+ # conflicts with cql completions
69
70
  return {}
70
71
 
71
72
  leaf = {}
72
73
  session = PostgresSession(state.namespace, state.pg_path)
73
74
  if session.db:
74
75
  if pg_table_names(state.namespace, state.pg_path):
75
- leaf = pg_completions(state.namespace, state.pg_path)
76
+ leaf = psql_completions(state.namespace, state.pg_path)
76
77
  elif state.pg_path:
77
78
  leaf = {
78
79
  '\h': None,
@@ -0,0 +1,10 @@
1
+ from adam.commands.postgres.postgres_utils import pg_table_names
2
+ from adam.sql.sql_completer import SqlCompleter
3
+
4
+ def psql_completions(ns: str, pg_path: str):
5
+ return {
6
+ '\h': None,
7
+ '\d': None,
8
+ '\dt': None,
9
+ '\du': None
10
+ } | SqlCompleter.completions(lambda: pg_table_names(ns, pg_path))
@@ -0,0 +1,11 @@
1
+ from adam.commands.postgres.postgres_utils import pg_table_names
2
+ from adam.sql.term_completer import TermCompleter
3
+
4
+ class PsqlTableNameCompleter(TermCompleter):
5
+ def __init__(self, namespace: str, pg_path: str, ignore_case: bool = True):
6
+ super().__init__(pg_table_names(namespace, pg_path), ignore_case=ignore_case)
7
+ self.namespace = namespace
8
+ self.pg_path = pg_path
9
+
10
+ def __repr__(self) -> str:
11
+ return "PsqlTableCompleter(%r, pg_path=%r)" % (self.namespace, self.pg_path)
@@ -1,10 +1,10 @@
1
1
  import functools
2
2
 
3
3
  from adam.commands.command import Command
4
- from adam.commands.cql.cql_table_completer import CqlTableCompleter
4
+ from adam.commands.cql.cql_table_completer import CqlTableNameCompleter
5
5
  from adam.commands.cql.cql_utils import run_cql, table_names, tables
6
- from adam.commands.postgres.pg_table_completer import PgTableCompleter
7
6
  from adam.commands.postgres.postgres_session import PostgresSession
7
+ from adam.commands.postgres.psql_table_completer import PsqlTableNameCompleter
8
8
  from adam.config import Config
9
9
  from adam.repl_state import ReplState, RequiredState
10
10
  from adam.utils import lines_to_tabular, log, log2
@@ -73,9 +73,9 @@ class PreviewTable(Command):
73
73
 
74
74
  def completion(self, state: ReplState):
75
75
  if state.device == ReplState.P:
76
- return {PreviewTable.COMMAND: PgTableCompleter(state.namespace, state.pg_path)}
76
+ return {PreviewTable.COMMAND: PsqlTableNameCompleter(state.namespace, state.pg_path)}
77
77
  elif state.sts:
78
- return {PreviewTable.COMMAND: CqlTableCompleter(table_names(state))}
78
+ return {PreviewTable.COMMAND: CqlTableNameCompleter(table_names(state))}
79
79
 
80
80
  return {}
81
81
 
adam/sql/__init__.py ADDED
File without changes
@@ -0,0 +1,403 @@
1
+ from typing import Callable, Iterable
2
+ from prompt_toolkit.completion import CompleteEvent, Completer, Completion
3
+ from prompt_toolkit.document import Document
4
+ import sqlparse
5
+ from sqlparse.sql import Statement, Token
6
+ from sqlparse import tokens as T
7
+
8
+ from adam.sql.term_completer import TermCompleter
9
+
10
+ class SqlCompleter(Completer):
11
+ def __init__(self, tables: Callable[[], list[str]], dml: str = None, debug = False):
12
+ super().__init__()
13
+ self.dml = dml
14
+ self.tables = tables
15
+ self.debug = debug
16
+
17
+ def get_completions(
18
+ self, document: Document, complete_event: CompleteEvent
19
+ ) -> Iterable[Completion]:
20
+ text = document.text_before_cursor.lstrip()
21
+ if self.dml:
22
+ state = f'{self.dml}_'
23
+ text = f'{self.dml} {text}'
24
+
25
+ completer = None
26
+ stmts = sqlparse.parse(text)
27
+ if not stmts:
28
+ completer = TermCompleter(['select', 'insert', 'delete', 'update'])
29
+ else:
30
+ statement: Statement = stmts[0]
31
+ state = self.traverse_tokens(text, statement.tokens)
32
+ if self.debug:
33
+ print('\n =>', state)
34
+ if state == 'dml_incomplete':
35
+ completer = TermCompleter(['select', 'insert', 'delete', 'update'])
36
+
37
+ elif state == 'select_':
38
+ completer = TermCompleter(['*'])
39
+ elif state == 'select_a':
40
+ completer = TermCompleter(['from'])
41
+ elif state == 'select_a,':
42
+ completer = TermCompleter(['*'])
43
+ elif state == 'select_a_':
44
+ completer = TermCompleter(['from'])
45
+ elif state == "select_a_from_":
46
+ completer = TermCompleter(self.tables())
47
+ elif state == "select_a_from_x_":
48
+ completer = TermCompleter(['where', 'group', 'limit'])
49
+ elif state == "select_a_from_x,":
50
+ completer = TermCompleter(self.tables())
51
+ elif state == "select_a_from_x_where_":
52
+ completer = TermCompleter(['id'])
53
+ elif state == "select_a_from_x_where_id":
54
+ completer = TermCompleter(['=', '<', '<=', '>', '>=', '<>', 'like'])
55
+ elif state == "select_a_from_x_where_id=":
56
+ completer = TermCompleter(["'"])
57
+ elif state == "select_a_from_x_where_id=v_":
58
+ completer = TermCompleter(['and', 'or', 'group', 'limit'])
59
+ elif state == "select_a_from_x_where_id=v_limit_":
60
+ completer = TermCompleter(['1'])
61
+ elif state == "select_a_from_x_group_":
62
+ completer = TermCompleter(['by'])
63
+ elif state == "select_a_from_x_group_by_":
64
+ completer = TermCompleter(['id'])
65
+ elif state == "select_a_from_x_group_by_a,":
66
+ completer = TermCompleter(['id'])
67
+ elif state == "select_a_from_x_group_by_a_":
68
+ completer = TermCompleter(['limit'])
69
+ elif state == "select_a_from_x_group_by_a_limit_":
70
+ completer = TermCompleter(['1'])
71
+
72
+ elif state == "insert_":
73
+ completer = TermCompleter(['into'])
74
+ elif state == "insert_into_":
75
+ completer = TermCompleter(self.tables())
76
+ elif state == "insert_into_x_":
77
+ completer = TermCompleter(['values'])
78
+ elif state == "insert_into_x(":
79
+ completer = TermCompleter(['id'])
80
+ elif state == "insert_into_x(a,":
81
+ completer = TermCompleter(['id'])
82
+ elif state == "insert_into_x(a)_":
83
+ completer = TermCompleter(['values('])
84
+ elif state == "insert_into_x_values":
85
+ completer = TermCompleter(['('])
86
+ elif state == "insert_into_x_values(":
87
+ completer = TermCompleter(["'"])
88
+
89
+ elif state == "update_":
90
+ completer = TermCompleter(self.tables())
91
+ elif state == "update_x_":
92
+ completer = TermCompleter(['set'])
93
+ elif state in ["update_x_set_", "update_x_set_a=v,"]:
94
+ completer = TermCompleter(['id'])
95
+ elif state == "update_x_set_a":
96
+ completer = TermCompleter(['='])
97
+ elif state == "update_x_set_a=":
98
+ completer = TermCompleter(["'"])
99
+ elif state == "update_x_set_a=v_":
100
+ completer = TermCompleter(['where'])
101
+ elif state == "update_x_set_a=v_where_":
102
+ completer = TermCompleter(['id'])
103
+ elif state == "update_x_set_a=v_where_id":
104
+ completer = TermCompleter(['='])
105
+ elif state == "update_x_set_a=v_where_id=v_":
106
+ completer = TermCompleter(['and', 'or'])
107
+
108
+ elif state == "delete_":
109
+ completer = TermCompleter(['from'])
110
+ elif state == "delete_from_":
111
+ completer = TermCompleter(self.tables())
112
+ elif state == "delete_from_x_":
113
+ completer = TermCompleter(['where'])
114
+ elif state == "delete_from_x_where_":
115
+ completer = TermCompleter(['id'])
116
+ elif state == "delete_from_x_where_id":
117
+ completer = TermCompleter(['='])
118
+ elif state == "delete_from_x_where_id=":
119
+ completer = TermCompleter(["'"])
120
+ elif state == "delete_from_x_where_id=v_":
121
+ completer = TermCompleter(['and', 'or'])
122
+
123
+ if completer:
124
+ for c in completer.get_completions(document, complete_event):
125
+ yield c
126
+
127
+ def traverse_tokens(self, text: str, tokens: list[Token], state: str = None, indent=0):
128
+ # state: str = None
129
+ for token in tokens:
130
+ if self.debug:
131
+ if token.ttype == T.Whitespace:
132
+ print('_ ', end='')
133
+ elif token.ttype in [T.DML, T.Wildcard, T.Punctuation]:
134
+ print(f'{token.value} ', end='')
135
+ elif token.ttype:
136
+ tks = str(token.ttype).split('.')
137
+ typ = tks[len(tks) - 1]
138
+ if ' ' in token.value:
139
+ print(f'"{token.value}:{typ}" ', end='')
140
+ else:
141
+ print(f'{token.value}:{typ} ', end='')
142
+ # print(" " * indent + f"Token: {token.value}, Type: {token.ttype}@{token.ttype.__class__}")
143
+ node: str = None
144
+ if token.is_group:
145
+ state = self.traverse_tokens(text, token.tokens, state, indent + 1)
146
+ else:
147
+ if not state:
148
+ if token.ttype == T.Keyword.DML and token.value.lower() == 'select':
149
+ state = 'select'
150
+ if token.ttype == T.Keyword.DML and token.value.lower() == 'insert':
151
+ state = 'insert'
152
+ if token.ttype == T.Keyword.DML and token.value.lower() == 'update':
153
+ state = 'update'
154
+ if token.ttype == T.Keyword.DML and token.value.lower() == 'delete':
155
+ state = 'delete'
156
+ elif token.ttype == T.Name:
157
+ state = 'dml_incomplete'
158
+
159
+ elif state == 'select':
160
+ if token.ttype == T.Text.Whitespace:
161
+ state = 'select_'
162
+ elif state == 'select_':
163
+ if token.ttype == T.Name or token.ttype == T.Wildcard:
164
+ state = 'select_a'
165
+ elif state == 'select_a':
166
+ if token.ttype == T.Text.Whitespace:
167
+ state = 'select_a_'
168
+ elif token.ttype == T.Punctuation and token.value == ',':
169
+ state = 'select_a,'
170
+ elif state == 'select_a,':
171
+ if token.ttype == T.Name or token.ttype == T.Wildcard:
172
+ state = 'select_a'
173
+ elif state == 'select_a_':
174
+ if token.ttype == T.Keyword and token.value.lower() == 'from':
175
+ state = 'select_a_from'
176
+ elif state == 'select_a_from':
177
+ if token.ttype == T.Text.Whitespace:
178
+ state = 'select_a_from_'
179
+ elif state == 'select_a_from_':
180
+ if token.ttype == T.Name:
181
+ state = 'select_a_from_x'
182
+ elif state == 'select_a_from_x':
183
+ if token.ttype == T.Text.Whitespace:
184
+ state = 'select_a_from_x_'
185
+ elif token.ttype == T.Punctuation and token.value == ',':
186
+ state = 'select_a_from_x,'
187
+ elif state == 'select_a_from_x,':
188
+ if token.ttype == T.Name:
189
+ state = 'select_a_from_x'
190
+ elif state == 'select_a_from_x_':
191
+ if token.ttype == T.Keyword and token.value.lower() == 'where':
192
+ state = 'select_a_from_x_where'
193
+ elif token.ttype == T.Keyword and token.value.lower() == 'limit':
194
+ state = 'select_a_from_x_where_id=v_limit'
195
+ elif token.ttype == T.Keyword and token.value.lower() == 'group':
196
+ state = 'select_a_from_x_group'
197
+ elif token.ttype == T.Keyword and token.value.lower() == 'group by':
198
+ state = 'select_a_from_x_group_by'
199
+ elif state == 'select_a_from_x_where':
200
+ if token.ttype == T.Text.Whitespace:
201
+ state = 'select_a_from_x_where_'
202
+ elif state == 'select_a_from_x_where_':
203
+ if token.ttype == T.Name:
204
+ state = 'select_a_from_x_where_id'
205
+ elif state == 'select_a_from_x_where_id':
206
+ if token.ttype == T.Operator.Comparison:
207
+ state = 'select_a_from_x_where_id='
208
+ elif state == 'select_a_from_x_where_id=':
209
+ if token.ttype in [T.Literal.String.Single, T.Name]:
210
+ state = 'select_a_from_x_where_id=v'
211
+ elif state == 'select_a_from_x_where_id=v':
212
+ if token.ttype == T.Text.Whitespace:
213
+ state = 'select_a_from_x_where_id=v_'
214
+ elif state == 'select_a_from_x_where_id=v_':
215
+ if token.ttype == T.Keyword and token.value.lower() in ['and', 'or']:
216
+ state = 'select_a_from_x_where'
217
+ elif token.ttype == T.Keyword and token.value.lower() == 'group':
218
+ state = 'select_a_from_x_group'
219
+ elif token.ttype == T.Keyword and token.value.lower() == 'limit':
220
+ state = 'select_a_from_x_where_id=v_limit'
221
+ elif state == 'select_a_from_x_group':
222
+ if token.ttype == T.Text.Whitespace:
223
+ state = 'select_a_from_x_group_'
224
+ elif state == 'select_a_from_x_group_':
225
+ if token.ttype == T.Keyword and token.value.lower() == 'by':
226
+ state = 'select_a_from_x_group_by'
227
+ elif state == 'select_a_from_x_group_by':
228
+ if token.ttype == T.Text.Whitespace:
229
+ state = 'select_a_from_x_group_by_'
230
+ elif state == 'select_a_from_x_group_by_':
231
+ if token.ttype == T.Name:
232
+ state = 'select_a_from_x_group_by_a'
233
+ elif state == 'select_a_from_x_group_by_a':
234
+ if token.ttype == T.Text.Whitespace:
235
+ state = 'select_a_from_x_group_by_a_'
236
+ elif token.ttype == T.Punctuation and token.value == ',':
237
+ state = 'select_a_from_x_group_by_a,'
238
+ elif state == 'select_a_from_x_group_by_a,':
239
+ if token.ttype == T.Name:
240
+ state = 'select_a_from_x_group_by_a'
241
+ elif state == 'select_a_from_x_group_by_a_':
242
+ if token.ttype == T.Keyword and token.value.lower() == 'limit':
243
+ state = 'select_a_from_x_where_id=v_limit'
244
+ elif state == 'select_a_from_x_where_id=v_limit':
245
+ if token.ttype == T.Text.Whitespace:
246
+ state = 'select_a_from_x_where_id=v_limit_'
247
+
248
+ elif state == 'insert':
249
+ if token.ttype == T.Text.Whitespace:
250
+ state = 'insert_'
251
+ elif state == 'insert_':
252
+ if token.ttype == T.Keyword and token.value.lower() == 'into':
253
+ state = 'insert_into'
254
+ elif state == 'insert_into':
255
+ if token.ttype == T.Text.Whitespace:
256
+ state = 'insert_into_'
257
+ elif state == 'insert_into_':
258
+ if token.ttype == T.Name:
259
+ state = 'insert_into_x'
260
+ elif state == 'insert_into_x':
261
+ if token.ttype == T.Text.Whitespace:
262
+ state = 'insert_into_x_'
263
+ elif token.ttype == T.Punctuation and token.value == '(':
264
+ state = 'insert_into_x('
265
+ elif state == 'insert_into_x_':
266
+ if token.ttype == T.Punctuation and token.value == '(':
267
+ state = 'insert_into_x('
268
+ elif token.ttype == T.Keyword and token.value.lower() == 'values':
269
+ state = 'insert_into_x_values'
270
+ elif state == 'insert_into_x(':
271
+ if token.ttype == T.Name:
272
+ state = 'insert_into_x(a'
273
+ elif state == 'insert_into_x(a':
274
+ if token.ttype == T.Punctuation and token.value == ',':
275
+ state = 'insert_into_x(a,'
276
+ elif token.ttype == T.Punctuation and token.value == ')':
277
+ state = 'insert_into_x(a)'
278
+ elif state == 'insert_into_x(a,':
279
+ if token.ttype == T.Name:
280
+ state = 'insert_into_x(a'
281
+ elif state == 'insert_into_x(a)':
282
+ if token.ttype == T.Text.Whitespace:
283
+ state = 'insert_into_x(a)_'
284
+ elif state == 'insert_into_x(a)_':
285
+ if token.ttype == T.Keyword and token.value.lower() == 'values':
286
+ state = 'insert_into_x_values'
287
+ elif state == 'insert_into_x_values':
288
+ if token.ttype == T.Punctuation and token.value == '(':
289
+ state = 'insert_into_x_values('
290
+ elif state == 'insert_into_x_values(':
291
+ if token.ttype in [T.Literal.String.Single, T.Name]:
292
+ state = 'insert_into_x_values(v'
293
+ elif state == 'insert_into_x_values(v':
294
+ if token.ttype == T.Punctuation and token.value == ',':
295
+ state = 'insert_into_x_values(v,'
296
+ elif token.ttype == T.Punctuation and token.value == ')':
297
+ state = 'insert_into_x_values(v)'
298
+ elif state == 'insert_into_x_values(v,':
299
+ if token.ttype in [T.Literal.String.Single, T.Name]:
300
+ state = 'insert_into_x_values(v'
301
+
302
+ elif state == 'update':
303
+ if token.ttype == T.Text.Whitespace:
304
+ state = 'update_'
305
+ elif state == 'update_':
306
+ if token.ttype == T.Name:
307
+ state = 'update_x'
308
+ elif state == 'update_x':
309
+ if token.ttype == T.Text.Whitespace:
310
+ state = 'update_x_'
311
+ elif state == 'update_x_':
312
+ if token.ttype == T.Keyword and token.value.lower() == 'set':
313
+ state = 'update_x_set'
314
+ elif state == 'update_x_set':
315
+ if token.ttype == T.Text.Whitespace:
316
+ state = 'update_x_set_'
317
+ elif state == 'update_x_set_':
318
+ if token.ttype == T.Name:
319
+ state = 'update_x_set_a'
320
+ elif state == 'update_x_set_a':
321
+ if token.ttype == T.Operator.Comparison:
322
+ state = 'update_x_set_a='
323
+ elif state == 'update_x_set_a=':
324
+ if token.ttype in [T.Literal.String.Single, T.Name]:
325
+ state = 'update_x_set_a=v'
326
+ elif state == 'update_x_set_a=v':
327
+ if token.ttype == T.Punctuation and token.value == ',':
328
+ state = 'update_x_set_a=v,'
329
+ elif token.ttype == T.Text.Whitespace:
330
+ state = 'update_x_set_a=v_'
331
+ elif state == 'update_x_set_a=v,':
332
+ if token.ttype == T.Name:
333
+ state = 'update_x_set_a'
334
+ elif state == 'update_x_set_a=v_':
335
+ if token.ttype == T.Punctuation and token.value == ',':
336
+ state = 'update_x_set_a=v,'
337
+ elif token.ttype == T.Keyword and token.value.lower() == 'where':
338
+ state = 'update_x_set_a=v_where'
339
+ elif state == 'update_x_set_a=v_where':
340
+ if token.ttype == T.Text.Whitespace:
341
+ state = 'update_x_set_a=v_where_'
342
+ elif state == 'update_x_set_a=v_where_':
343
+ if token.ttype == T.Name:
344
+ state = 'update_x_set_a=v_where_id'
345
+ elif state == 'update_x_set_a=v_where_id':
346
+ if token.ttype == T.Operator.Comparison:
347
+ state = 'update_x_set_a=v_where_id='
348
+ elif state == 'update_x_set_a=v_where_id=':
349
+ if token.ttype in [T.Literal.String.Single, T.Name]:
350
+ state = 'update_x_set_a=v_where_id=v'
351
+ elif state == 'update_x_set_a=v_where_id=v':
352
+ if token.ttype == T.Text.Whitespace:
353
+ state = 'update_x_set_a=v_where_id=v_'
354
+ elif state == 'update_x_set_a=v_where_id=v_':
355
+ if token.ttype == T.Keyword and token.value.lower() in ['and', 'or']:
356
+ state = 'update_x_set_a=v_where'
357
+
358
+ elif state == 'delete':
359
+ if token.ttype == T.Text.Whitespace:
360
+ state = 'delete_'
361
+ elif state == 'delete_':
362
+ if token.ttype == T.Keyword and token.value.lower() == 'from':
363
+ state = 'delete_from'
364
+ elif state == 'delete_from':
365
+ if token.ttype == T.Text.Whitespace:
366
+ state = 'delete_from_'
367
+ elif state == 'delete_from_':
368
+ if token.ttype == T.Name:
369
+ state = 'delete_from_x'
370
+ elif state == 'delete_from_x':
371
+ if token.ttype == T.Text.Whitespace:
372
+ state = 'delete_from_x_'
373
+ elif state == 'delete_from_x_':
374
+ if token.ttype == T.Keyword and token.value.lower() == 'where':
375
+ state = 'delete_from_x_where'
376
+ elif state == 'delete_from_x_where':
377
+ if token.ttype == T.Text.Whitespace:
378
+ state = 'delete_from_x_where_'
379
+ elif state == 'delete_from_x_where_':
380
+ if token.ttype == T.Name:
381
+ state = 'delete_from_x_where_id'
382
+ elif state == 'delete_from_x_where_id':
383
+ if token.ttype == T.Operator.Comparison:
384
+ state = 'delete_from_x_where_id='
385
+ elif state == 'delete_from_x_where_id=':
386
+ if token.ttype in [T.Literal.String.Single, T.Name]:
387
+ state = 'delete_from_x_where_id=v'
388
+ elif state == 'delete_from_x_where_id=v':
389
+ if token.ttype == T.Text.Whitespace:
390
+ state = 'delete_from_x_where_id=v_'
391
+ elif state == 'delete_from_x_where_id=v_':
392
+ if token.ttype == T.Keyword and token.value.lower() in ['and', 'or']:
393
+ state = 'delete_from_x_where'
394
+
395
+ return state
396
+
397
+ def completions(table_names: Callable[[], list[str]]):
398
+ return {
399
+ 'delete': SqlCompleter(table_names, 'delete'),
400
+ 'insert': SqlCompleter(table_names, 'insert'),
401
+ 'select': SqlCompleter(table_names, 'select'),
402
+ 'update': SqlCompleter(table_names, 'update'),
403
+ }
adam/sql/sql_utils.py ADDED
@@ -0,0 +1,5 @@
1
+ import re
2
+
3
+ def safe_terms(text: str):
4
+ tokens = re.findall(r'"[^"]+"|\b\w+\b|\S', text)
5
+ return tokens, len(tokens) > 1 or text.startswith(' ') or text.endswith(' ')
@@ -0,0 +1,66 @@
1
+ from typing import Callable, Iterable, List, Mapping, Optional, Pattern, Union
2
+
3
+ from prompt_toolkit.completion import CompleteEvent, Completer, Completion, WordCompleter
4
+ from prompt_toolkit.document import Document
5
+ from prompt_toolkit.formatted_text import AnyFormattedText
6
+
7
+ __all__ = [
8
+ "TermCompleter",
9
+ ]
10
+
11
+ class TermCompleter(WordCompleter):
12
+ def __init__(
13
+ self,
14
+ words: Union[List[str], Callable[[], List[str]]],
15
+ ignore_case: bool = False,
16
+ display_dict: Optional[Mapping[str, AnyFormattedText]] = None,
17
+ meta_dict: Optional[Mapping[str, AnyFormattedText]] = None,
18
+ WORD: bool = False,
19
+ sentence: bool = False,
20
+ match_middle: bool = False,
21
+ pattern: Optional[Pattern[str]] = None,
22
+ ) -> None:
23
+ super().__init__(words, ignore_case, display_dict, meta_dict, WORD, sentence, match_middle, pattern)
24
+
25
+ def get_completions(
26
+ self, document: Document, complete_event: CompleteEvent
27
+ ) -> Iterable[Completion]:
28
+ # Get list of words.
29
+ words = self.words
30
+ if callable(words):
31
+ words = words()
32
+
33
+ # Get word/text before cursor.
34
+ if self.sentence:
35
+ word_before_cursor = document.text_before_cursor
36
+ else:
37
+ word_before_cursor = document.get_word_before_cursor(
38
+ WORD=self.WORD, pattern=self.pattern
39
+ )
40
+
41
+ if self.ignore_case:
42
+ word_before_cursor = word_before_cursor.lower()
43
+
44
+ def word_matches(word: str) -> bool:
45
+ """True when the word before the cursor matches."""
46
+ if self.ignore_case:
47
+ word = word.lower()
48
+
49
+ if word_before_cursor in ['(', ',', '=']:
50
+ return True
51
+
52
+ if self.match_middle:
53
+ return word_before_cursor in word
54
+ else:
55
+ return word.startswith(word_before_cursor)
56
+
57
+ for a in words:
58
+ if word_matches(a):
59
+ display = self.display_dict.get(a, a)
60
+ display_meta = self.meta_dict.get(a, "")
61
+ yield Completion(
62
+ a,
63
+ -len(word_before_cursor),
64
+ display=display,
65
+ display_meta=display_meta,
66
+ )
adam/version.py CHANGED
@@ -1,5 +1,5 @@
1
1
  #!/usr/bin/env python
2
2
  # -*- coding: utf-8 -*-
3
3
 
4
- __version__ = "2.0.49" #: the working version
4
+ __version__ = "2.0.51" #: the working version
5
5
  __release__ = "1.0.0" #: the release version
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: kaqing
3
- Version: 2.0.49
3
+ Version: 2.0.51
4
4
  Summary: UNKNOWN
5
5
  Home-page: UNKNOWN
6
6
  License: UNKNOWN
@@ -13,9 +13,8 @@ adam/repl.py,sha256=X9ga1iZFIXADQxBXvztFyVNBrfBRxiFLxTUYkm11img,7328
13
13
  adam/repl_commands.py,sha256=WA90Rl27Juctzr3U3kfCDk5N-oYMKlfWbZeafUgk7k0,4723
14
14
  adam/repl_session.py,sha256=uIogcvWBh7wd8QQ-p_JgLsyJ8YJgINw5vOd6JIsd7Vo,472
15
15
  adam/repl_state.py,sha256=591d7gV6uQSFtm7IWdlIYAHjfAzs9bdvIkwlIAeKddE,7540
16
- adam/table_completer.py,sha256=6lOSusLCbmILJfERRNxOIgVp-8GrmJf4rQ_pchZbceg,3569
17
16
  adam/utils.py,sha256=2DoWsrcaioFFH0-RjT30qelVRPUJqCGTfz_ucfE7F8g,7406
18
- adam/version.py,sha256=yZ-JQgru1WpJO6wYfsHtZ05JFJ6_oRFLc_ltDuD5KEI,139
17
+ adam/version.py,sha256=DIVzASNDynfH_uX-JULcKL8jQRSiIpR4Iw9R6p2ZFXM,139
19
18
  adam/checks/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
20
19
  adam/checks/check.py,sha256=Qopr3huYcMu2bzQgb99dEUYjFzkjKHRI76S6KA9b9Rk,702
21
20
  adam/checks/check_context.py,sha256=FEHkQ32jY1EDopQ2uYWqy9v7aEEX1orLpJWhopwAlh4,402
@@ -70,7 +69,7 @@ adam/commands/nodetool.py,sha256=HV9yDzMhRjS4lw6UfV7Hc1pcmeSx5a1jU6cAEKSZ1Bg,233
70
69
  adam/commands/nodetool_commands.py,sha256=5IgWC3rmeDD1cgwqQjiiWzi-wJpJ3n_8pAzz_9phXuk,2635
71
70
  adam/commands/param_get.py,sha256=kPAAppK2T0tEFRnSIVFLDPIIGHhgLA7drJhn8TRyvvE,1305
72
71
  adam/commands/param_set.py,sha256=QDIuqfU80aWCB16OK49yf7XRaRTWwiLkwMsJuVikq9I,1271
73
- adam/commands/preview_table.py,sha256=T70GQULpoxBQWppGlMKpqjJiBTmCVlArTSuPf1hHh9c,3046
72
+ adam/commands/preview_table.py,sha256=5im47Uk6RclgkZD-Ci0lBs1eosnfvBtKt2lgsb_bohs,3068
74
73
  adam/commands/pwd.py,sha256=VlgFjxFl66I7Df1YI6cuiEeY6Q13lEavMKfrzHLESKo,2340
75
74
  adam/commands/report.py,sha256=Ky45LIzSlB_X4V12JZWjU3SA2u4_FKRencRTq7psOWU,1944
76
75
  adam/commands/restart.py,sha256=Hik1t5rjH2ATZv4tzwrGB3e44b3dNuATgY327_Nb8Bs,2044
@@ -78,10 +77,10 @@ adam/commands/rollout.py,sha256=52_4ijna3v-8Oug12et43DRHFDNhiN34p6xLTQmhdbQ,2959
78
77
  adam/commands/shell.py,sha256=wY_PIx7Lt6vuxhFArlfxdEnBbrouCJ3yNHhFn17DEqw,848
79
78
  adam/commands/watch.py,sha256=mmBFpB8T1V7zrNs5b2YNyDDztMym_ILPDdkrbdAXTas,2438
80
79
  adam/commands/cql/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
81
- adam/commands/cql/cql_completions.py,sha256=stgjuB7iAXKY2F6vM5bkOrNuJF9zmTziNEd9dyA5fpE,442
82
- adam/commands/cql/cql_table_completer.py,sha256=6kvKatvKDW4qL2auZDjH050CKlV3lSmTSV4lvcEnpMo,604
80
+ adam/commands/cql/cql_completions.py,sha256=dXe51NTWEJis76587IWSn9Av-cjC0J6KMaxlBKfF4wM,411
81
+ adam/commands/cql/cql_table_completer.py,sha256=Tth6lmZ1eCEbJeAVZojTx594ttQeeVf-OjhhkSLyRnI,312
83
82
  adam/commands/cql/cql_utils.py,sha256=q5hzAUVh7h8iVVH2s0M4E76zWqaMUDXULMn412mfHII,3893
84
- adam/commands/cql/cqlsh.py,sha256=mIoF8S33DAOFhkJrITRR4rkxFGVznD5hDb_Y-LsZA2w,2615
83
+ adam/commands/cql/cqlsh.py,sha256=qEQufaDVi9FXkvruum6OHQDfLX01DVWVDnWsAjyCZYQ,2661
85
84
  adam/commands/deploy/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
86
85
  adam/commands/deploy/code_start.py,sha256=-iH8HThTNM83IfBxT_LqTByuHVatV9d-Il4OYOfrwLI,1370
87
86
  adam/commands/deploy/code_stop.py,sha256=ch7ZMgosvTHsGaIcDwQY5XYh_5HYrUjBkZFOI-d2gOU,1696
@@ -104,17 +103,17 @@ adam/commands/describe/describe_tables.py,sha256=vIs4lKz4nHBJumOlR4wBhR1QxXTFqzX
104
103
  adam/commands/medusa/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
105
104
  adam/commands/medusa/medusa.py,sha256=KNFjStvilIuOJt3wTtcWmKvdm8FdCnrDY2ltEWbratk,1402
106
105
  adam/commands/medusa/medusa_backup.py,sha256=j4DTVWFT-4rzs4gG_pBvjE-JuPsVCJIsnyQjIzJ4EbA,1801
107
- adam/commands/medusa/medusa_restore.py,sha256=MU47bmozrjfGJ6GVkj_OVgLH6Uz_fGh03MdLR4ZDKHE,3308
106
+ adam/commands/medusa/medusa_restore.py,sha256=GSjwsZYCqRA-XmJ1f6VELzX9cKXh5NDwxyBoGEiqUXY,3274
108
107
  adam/commands/medusa/medusa_show_backupjobs.py,sha256=QekHpKezVJdgfa9hOxfgyx-y4D08tmHzyu_AAa8QPR0,1756
109
108
  adam/commands/medusa/medusa_show_restorejobs.py,sha256=wgPonSmC6buDIp3k3WUY-Yu2MyP1xyE3Q_XhvAwpnx4,1651
110
109
  adam/commands/postgres/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
111
- adam/commands/postgres/pg_completions.py,sha256=GteenLhrBlRXMKQLSqM-6SmNZEFXY8wZ4CQWkz091OI,260
112
- adam/commands/postgres/pg_table_completer.py,sha256=f5YLFcmC2e1_sb_Kl5T5OZ6b8I-kjE5Lf4_BWH392qM,784
113
- adam/commands/postgres/postgres.py,sha256=A2mGzvyuCLFFTSKL74_RUcnJ0fI8wkkUukwmx2sF0as,3091
110
+ adam/commands/postgres/postgres.py,sha256=TiJwXGa4faPZ7DI8698SwgjnUXsIaVgG-R_-eXVPhwk,3142
114
111
  adam/commands/postgres/postgres_ls.py,sha256=HwZTgwGKXUqHX33S8aQPF6FqCrLqtoz4cLyJV2SpoE0,1186
115
112
  adam/commands/postgres/postgres_preview.py,sha256=MLzdEc4mvNj6V1Q8jO5OPznXyYELJHgd35_eQgLlNIU,1274
116
113
  adam/commands/postgres/postgres_session.py,sha256=RcqkCgtA78M-9LKTwG6pp8n9JwjmURXgf1FknIIPl9g,9305
117
114
  adam/commands/postgres/postgres_utils.py,sha256=mC5PZXrfxzqZt6n_HYASwe6Y-OyAK3QwEtDeVb_TKrA,877
115
+ adam/commands/postgres/psql_completions.py,sha256=IHcSy-fh028_rYw2gcXaPuf_4TidP5a2hPrl63N643M,322
116
+ adam/commands/postgres/psql_table_completer.py,sha256=WI9VvnI2w6U8HZOdPPiacpf4ShPjHuunHnOtrx7LYG8,509
118
117
  adam/commands/reaper/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
119
118
  adam/commands/reaper/reaper.py,sha256=83R0ZRitEwaYKKssfKxn3zAzLnWIP9QKd1mA6awceS8,1908
120
119
  adam/commands/reaper/reaper_forward.py,sha256=mUp409MzT91cVXGxoPfBGceaR3qZ0rVdWKGdyzPNzSA,3177
@@ -164,6 +163,10 @@ adam/k8s_utils/service_accounts.py,sha256=v2oQSqCrNvt2uRnKlNwR3fjtpUG7oF5nqgzEB7
164
163
  adam/k8s_utils/services.py,sha256=EOJJGACVbbRvu5T3rMKqIJqgYic1_MSJ17EA0TJ6UOk,3156
165
164
  adam/k8s_utils/statefulsets.py,sha256=hiBOmJZ3KTI6_naAFzNoW1NoYnnBG35BZ7RMdPhNC6o,4664
166
165
  adam/k8s_utils/volumes.py,sha256=RIBmlOSWM3V3QVXLCFT0owVOyh4rGG1ETp521a-6ndo,1137
166
+ adam/sql/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
167
+ adam/sql/sql_completer.py,sha256=RhfE8_ecn84HtiCFPMDvKrkPySFXVIJ2ez9bN8-wdP4,20902
168
+ adam/sql/sql_utils.py,sha256=MLoxB9h22WZiANu0SZzmBJcY2gtb6f_D4jnoJEjLgPc,171
169
+ adam/sql/term_completer.py,sha256=O-sUxFVc1DJmPQCdT-XPK2SSkUe4AL1dw8ZPXXicATY,2273
167
170
  adam/sso/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
168
171
  adam/sso/authenticator.py,sha256=BCm16L9zf5aLU47-sTCnudn2zLPwd8M2wwRminJfsqw,615
169
172
  adam/sso/authn_ad.py,sha256=fDW8UR3WWykny5Awa5dQjjBUSFzIDz4aMn-lwXoABl8,5857
@@ -174,8 +177,8 @@ adam/sso/idp.py,sha256=fvcwUw_URTgsO6ySaqTIw0zQT2qRO1IPSGhf6rPtybo,5804
174
177
  adam/sso/idp_login.py,sha256=QAtCUeDTVWliJy40RK_oac8Vgybr13xH8wzeBoxPaa8,1754
175
178
  adam/sso/idp_session.py,sha256=9BUHNRf70u4rVKrVY1HKPOEmOviXvkjam8WJxmXSKIM,1735
176
179
  adam/sso/sso_config.py,sha256=5N8WZgIJQBtHUy585XLRWKjpU87_v6QluyNK9E27D5s,2459
177
- kaqing-2.0.49.dist-info/METADATA,sha256=wtMivpLn3FjaPqvRL1TCEzPNjhNlB9SwDJNUH3Wb0CI,132
178
- kaqing-2.0.49.dist-info/WHEEL,sha256=tZoeGjtWxWRfdplE7E3d45VPlLNQnvbKiYnx7gwAy8A,92
179
- kaqing-2.0.49.dist-info/entry_points.txt,sha256=SkzhuQJUWsXOzHeZ5TgQ2c3_g53UGK23zzJU_JTZOZI,39
180
- kaqing-2.0.49.dist-info/top_level.txt,sha256=8_2PZkwBb-xDcnc8a2rAbQeJhXKXskc7zTP7pSPa1fw,5
181
- kaqing-2.0.49.dist-info/RECORD,,
180
+ kaqing-2.0.51.dist-info/METADATA,sha256=yrXdUA18-Fl_t3WHsjhq9WePo338WX7YSh0VD0M6fgM,132
181
+ kaqing-2.0.51.dist-info/WHEEL,sha256=tZoeGjtWxWRfdplE7E3d45VPlLNQnvbKiYnx7gwAy8A,92
182
+ kaqing-2.0.51.dist-info/entry_points.txt,sha256=SkzhuQJUWsXOzHeZ5TgQ2c3_g53UGK23zzJU_JTZOZI,39
183
+ kaqing-2.0.51.dist-info/top_level.txt,sha256=8_2PZkwBb-xDcnc8a2rAbQeJhXKXskc7zTP7pSPa1fw,5
184
+ kaqing-2.0.51.dist-info/RECORD,,
@@ -1,9 +0,0 @@
1
- from adam.commands.postgres.pg_table_completer import PgTableCompleter
2
-
3
- def pg_completions(ns: str, pg_path: str):
4
- return {
5
- '\h': None,
6
- '\d': None,
7
- '\dt': None,
8
- '\du': None
9
- } | PgTableCompleter(ns, pg_path).completions()
@@ -1,18 +0,0 @@
1
- from adam.commands.postgres.postgres_utils import pg_table_names
2
- from adam.table_completer import NestedDict, TableCompleter
3
-
4
- class PgTableCompleter(TableCompleter):
5
- def __init__(self, namespace: str, pg_path: str, nested_dict: NestedDict = {}, ignore_case: bool = True):
6
- self.namespace = namespace
7
- self.pg_path = pg_path
8
- self.ignore_case = ignore_case
9
- self.append_nested_dict(nested_dict)
10
-
11
- def __repr__(self) -> str:
12
- return "PgTableCompleter(%r, pg_path=%r)" % (self.namespace, self.pg_path)
13
-
14
- def nested(self, data: NestedDict) -> 'TableCompleter':
15
- return PgTableCompleter(self.namespace, self.pg_path).append_nested_dict(data)
16
-
17
- def tables(self) -> list[str]:
18
- return pg_table_names(self.namespace, self.pg_path)
adam/table_completer.py DELETED
@@ -1,93 +0,0 @@
1
- from abc import abstractmethod
2
- from typing import Any, Dict, Iterable, Mapping, Optional, Set, Union
3
- from prompt_toolkit.completion import CompleteEvent, Completer, Completion, NestedCompleter, WordCompleter
4
- from prompt_toolkit.document import Document
5
-
6
- NestedDict = Mapping[str, Union[Any, Set[str], None, Completer]]
7
-
8
- class TableCompleter(Completer):
9
- def __init__(self, nested_dict: NestedDict = {}, ignore_case: bool = True):
10
- self.ignore_case = ignore_case
11
- self.append_nested_dict(nested_dict)
12
-
13
- def append_nested_dict(self, data: NestedDict) -> "TableCompleter":
14
- options: Dict[str, Optional[Completer]] = {}
15
- for key, value in data.items():
16
- if isinstance(value, Completer):
17
- options[key] = value
18
- elif isinstance(value, dict):
19
- options[key] = NestedCompleter.from_nested_dict(value)
20
- elif isinstance(value, set):
21
- options[key] = NestedCompleter.from_nested_dict({item: None for item in value})
22
- else:
23
- assert value is None
24
- options[key] = None
25
-
26
- self.options = options
27
-
28
- return self
29
-
30
- def get_completions(
31
- self, document: Document, complete_event: CompleteEvent
32
- ) -> Iterable[Completion]:
33
- text = document.text_before_cursor.lstrip()
34
- stripped_len = len(document.text_before_cursor) - len(text)
35
-
36
- if " " in text:
37
- second_term = None
38
- tokens = text.split()
39
- if len(tokens) > 1:
40
- second_term = tokens[1]
41
-
42
- yielded = False
43
- if second_term:
44
- completer = self.options.get(second_term)
45
-
46
- if completer is not None:
47
- first_term = tokens[0]
48
- remaining_text = text[len(first_term) :].lstrip()
49
- move_cursor = len(text) - len(remaining_text) + stripped_len
50
-
51
- remaining_text = remaining_text[len(second_term) :].lstrip()
52
- move_cursor = len(text) - len(remaining_text) + stripped_len
53
-
54
- new_document = Document(
55
- remaining_text,
56
- cursor_position=document.cursor_position - move_cursor,
57
- )
58
-
59
- for c in completer.get_completions(new_document, complete_event):
60
- yield c
61
- yielded = True
62
-
63
- if not yielded:
64
- completer = WordCompleter(
65
- list(self.options.keys()), ignore_case=self.ignore_case
66
- )
67
- for c in completer.get_completions(document, complete_event):
68
- yield c
69
- else:
70
- completer = WordCompleter(
71
- self.tables(), ignore_case=self.ignore_case
72
- )
73
- for c in completer.get_completions(document, complete_event):
74
- yield c
75
-
76
- @abstractmethod
77
- def nested(self, data: NestedDict) -> 'TableCompleter':
78
- pass
79
-
80
- @abstractmethod
81
- def tables(self) -> list[str]:
82
- pass
83
-
84
- def completions(self):
85
- return {
86
- 'delete': {'from': self.nested({'where': {'id': {'=': {"'id'": None}}}})},
87
- 'insert': {'into': self.nested({'values(': None})},
88
- 'select': {'*': {'from': self.nested({
89
- 'limit': {'1': {'where': {'id': {'=': {"'id'": None}}}}},
90
- 'where': {'id': {'=': {"'id'": {'limit': {'1': None}}}}}
91
- })}},
92
- 'update': self.nested({'set': {'column': {'=': None}}}),
93
- }