2 * psql - the PostgreSQL interactive terminal
4 * Copyright (c) 2000-2021, PostgreSQL Global Development Group
6 * src/bin/psql/tab-complete.c
9 /*----------------------------------------------------------------------
10 * This file implements a somewhat more sophisticated readline "TAB
11 * completion" in psql. It is not intended to be AI, to replace
12 * learning SQL, or to relieve you from thinking about what you're
13 * doing. Also it does not always give you all the syntactically legal
14 * completions, only those that are the most common or the ones that
15 * the programmer felt most like implementing.
17 * CAVEAT: Tab completion causes queries to be sent to the backend.
18 * The number of tuples returned gets limited, in most default
19 * installations to 1000, but if you still don't like this prospect,
20 * you can turn off tab completion in your ~/.inputrc (or else
21 * ${INPUTRC}) file so:
24 * set disable-completion on
27 * See `man 3 readline' or `info readline' for the full details.
30 * - Quotes, parentheses, and other funny characters are not handled
31 * all that gracefully.
32 *----------------------------------------------------------------------
35 #include "postgres_fe.h"
38 #include "tab-complete.h"
40 /* If we don't have this, we might as well forget about the whole thing: */
46 #include "catalog/pg_am_d.h"
47 #include "catalog/pg_class_d.h"
50 #include "pqexpbuffer.h"
52 #include "stringutils.h"
55 * Ancient versions of libedit provide filename_completion_function()
56 * instead of rl_filename_completion_function(). Likewise for
57 * [rl_]completion_matches().
59 #ifndef HAVE_RL_FILENAME_COMPLETION_FUNCTION
60 #define rl_filename_completion_function filename_completion_function
63 #ifndef HAVE_RL_COMPLETION_MATCHES
64 #define rl_completion_matches completion_matches
68 * Currently we assume that rl_filename_dequoting_function exists if
69 * rl_filename_quoting_function does. If that proves not to be the case,
70 * we'd need to test for the former, or possibly both, in configure.
72 #ifdef HAVE_RL_FILENAME_QUOTING_FUNCTION
73 #define USE_FILENAME_QUOTING_FUNCTIONS 1
76 /* word break characters */
77 #define WORD_BREAKS "\t\n@$><=;|&{() "
80 * Since readline doesn't let us pass any state through to the tab completion
81 * callback, we have to use this global variable to let get_previous_words()
82 * get at the previous lines of the current command. Ick.
84 PQExpBuffer tab_completion_query_buf
= NULL
;
87 * In some situations, the query to find out what names are available to
88 * complete with must vary depending on server version. We handle this by
89 * storing a list of queries, each tagged with the minimum server version
90 * it will work for. Each list must be stored in descending server version
91 * order, so that the first satisfactory query is the one to use.
93 * When the query string is otherwise constant, an array of VersionedQuery
94 * suffices. Terminate the array with an entry having min_server_version = 0.
95 * That entry's query string can be a query that works in all supported older
96 * server versions, or NULL to give up and do no completion.
98 typedef struct VersionedQuery
100 int min_server_version
;
105 * This struct is used to define "schema queries", which are custom-built
106 * to obtain possibly-schema-qualified names of database objects. There is
107 * enough similarity in the structure that we don't want to repeat it each
108 * time. So we put the components of each query into this struct and
109 * assemble them with the common boilerplate in _complete_from_query().
111 * As with VersionedQuery, we can use an array of these if the query details
112 * must vary across versions.
114 typedef struct SchemaQuery
117 * If not zero, minimum server version this struct applies to. If not
118 * zero, there should be a following struct with a smaller minimum server
119 * version; use catname == NULL in the last entry if we should do nothing.
121 int min_server_version
;
124 * Name of catalog or catalogs to be queried, with alias, eg.
125 * "pg_catalog.pg_class c". Note that "pg_namespace n" will be added.
130 * Selection condition --- only rows meeting this condition are candidates
131 * to display. If catname mentions multiple tables, include the necessary
132 * join condition here. For example, this might look like "c.relkind = "
133 * CppAsString2(RELKIND_RELATION). Write NULL (not an empty string) if
136 const char *selcondition
;
139 * Visibility condition --- which rows are visible without schema
140 * qualification? For example, "pg_catalog.pg_table_is_visible(c.oid)".
142 const char *viscondition
;
145 * Namespace --- name of field to join to pg_namespace.oid. For example,
148 const char *namespace;
151 * Result --- the appropriately-quoted name to return, in the case of an
152 * unqualified name. For example, "pg_catalog.quote_ident(c.relname)".
157 * In some cases a different result must be used for qualified names.
158 * Enter that here, or write NULL if result can be used.
160 const char *qualresult
;
164 /* Store maximum number of records we want from database queries
165 * (implemented via SELECT ... LIMIT xx).
167 static int completion_max_records
;
170 * Communication variables set by psql_completion (mostly in COMPLETE_WITH_FOO
171 * macros) and then used by the completion callback functions. Ugly but there
174 static char completion_last_char
; /* last char of input word */
175 static const char *completion_charp
; /* to pass a string */
176 static const char *const *completion_charpp
; /* to pass a list of strings */
177 static const char *completion_info_charp
; /* to pass a second string */
178 static const char *completion_info_charp2
; /* to pass a third string */
179 static const VersionedQuery
*completion_vquery
; /* to pass a VersionedQuery */
180 static const SchemaQuery
*completion_squery
; /* to pass a SchemaQuery */
181 static bool completion_case_sensitive
; /* completion is case sensitive */
182 static bool completion_force_quote
; /* true to force-quote filenames */
185 * A few macros to ease typing. You can use these to complete the given
187 * 1) The results from a query you pass it. (Perhaps one of those below?)
188 * We support both simple and versioned queries.
189 * 2) The results from a schema query you pass it.
190 * We support both simple and versioned schema queries.
191 * 3) The items from a null-pointer-terminated list (with or without
192 * case-sensitive comparison); if the list is constant you can build it
193 * with COMPLETE_WITH() or COMPLETE_WITH_CS().
194 * 4) The list of attributes of the given table (possibly schema-qualified).
195 * 5) The list of arguments to the given function (possibly schema-qualified).
197 #define COMPLETE_WITH_QUERY(query) \
199 completion_charp = query; \
200 matches = rl_completion_matches(text, complete_from_query); \
203 #define COMPLETE_WITH_VERSIONED_QUERY(query) \
205 completion_vquery = query; \
206 matches = rl_completion_matches(text, complete_from_versioned_query); \
209 #define COMPLETE_WITH_SCHEMA_QUERY(query, addon) \
211 completion_squery = &(query); \
212 completion_charp = addon; \
213 matches = rl_completion_matches(text, complete_from_schema_query); \
216 #define COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(query, addon) \
218 completion_squery = query; \
219 completion_vquery = addon; \
220 matches = rl_completion_matches(text, complete_from_versioned_schema_query); \
224 * Caution: COMPLETE_WITH_CONST is not for general-purpose use; you probably
225 * want COMPLETE_WITH() with one element, instead.
227 #define COMPLETE_WITH_CONST(cs, con) \
229 completion_case_sensitive = (cs); \
230 completion_charp = (con); \
231 matches = rl_completion_matches(text, complete_from_const); \
234 #define COMPLETE_WITH_LIST_INT(cs, list) \
236 completion_case_sensitive = (cs); \
237 completion_charpp = (list); \
238 matches = rl_completion_matches(text, complete_from_list); \
241 #define COMPLETE_WITH_LIST(list) COMPLETE_WITH_LIST_INT(false, list)
242 #define COMPLETE_WITH_LIST_CS(list) COMPLETE_WITH_LIST_INT(true, list)
244 #define COMPLETE_WITH(...) \
246 static const char *const list[] = { __VA_ARGS__, NULL }; \
247 COMPLETE_WITH_LIST(list); \
250 #define COMPLETE_WITH_CS(...) \
252 static const char *const list[] = { __VA_ARGS__, NULL }; \
253 COMPLETE_WITH_LIST_CS(list); \
256 #define COMPLETE_WITH_ATTR(relation, addon) \
258 char *_completion_schema; \
259 char *_completion_table; \
261 _completion_schema = strtokx(relation, " \t\n\r", ".", "\"", 0, \
262 false, false, pset.encoding); \
263 (void) strtokx(NULL, " \t\n\r", ".", "\"", 0, \
264 false, false, pset.encoding); \
265 _completion_table = strtokx(NULL, " \t\n\r", ".", "\"", 0, \
266 false, false, pset.encoding); \
267 if (_completion_table == NULL) \
269 completion_charp = Query_for_list_of_attributes addon; \
270 completion_info_charp = relation; \
274 completion_charp = Query_for_list_of_attributes_with_schema addon; \
275 completion_info_charp = _completion_table; \
276 completion_info_charp2 = _completion_schema; \
278 matches = rl_completion_matches(text, complete_from_query); \
281 #define COMPLETE_WITH_ENUM_VALUE(type) \
283 char *_completion_schema; \
284 char *_completion_type; \
286 _completion_schema = strtokx(type, " \t\n\r", ".", "\"", 0, \
287 false, false, pset.encoding); \
288 (void) strtokx(NULL, " \t\n\r", ".", "\"", 0, \
289 false, false, pset.encoding); \
290 _completion_type = strtokx(NULL, " \t\n\r", ".", "\"", 0, \
291 false, false, pset.encoding); \
292 if (_completion_type == NULL)\
294 completion_charp = Query_for_list_of_enum_values; \
295 completion_info_charp = type; \
299 completion_charp = Query_for_list_of_enum_values_with_schema; \
300 completion_info_charp = _completion_type; \
301 completion_info_charp2 = _completion_schema; \
303 matches = rl_completion_matches(text, complete_from_query); \
306 #define COMPLETE_WITH_FUNCTION_ARG(function) \
308 char *_completion_schema; \
309 char *_completion_function; \
311 _completion_schema = strtokx(function, " \t\n\r", ".", "\"", 0, \
312 false, false, pset.encoding); \
313 (void) strtokx(NULL, " \t\n\r", ".", "\"", 0, \
314 false, false, pset.encoding); \
315 _completion_function = strtokx(NULL, " \t\n\r", ".", "\"", 0, \
316 false, false, pset.encoding); \
317 if (_completion_function == NULL) \
319 completion_charp = Query_for_list_of_arguments; \
320 completion_info_charp = function; \
324 completion_charp = Query_for_list_of_arguments_with_schema; \
325 completion_info_charp = _completion_function; \
326 completion_info_charp2 = _completion_schema; \
328 matches = rl_completion_matches(text, complete_from_query); \
332 * Assembly instructions for schema queries
334 * Note that toast tables are not included in those queries to avoid
335 * unnecessary bloat in the completions generated.
338 static const SchemaQuery Query_for_list_of_aggregates
[] = {
340 .min_server_version
= 110000,
341 .catname
= "pg_catalog.pg_proc p",
342 .selcondition
= "p.prokind = 'a'",
343 .viscondition
= "pg_catalog.pg_function_is_visible(p.oid)",
344 .namespace = "p.pronamespace",
345 .result
= "pg_catalog.quote_ident(p.proname)",
348 .catname
= "pg_catalog.pg_proc p",
349 .selcondition
= "p.proisagg",
350 .viscondition
= "pg_catalog.pg_function_is_visible(p.oid)",
351 .namespace = "p.pronamespace",
352 .result
= "pg_catalog.quote_ident(p.proname)",
356 static const SchemaQuery Query_for_list_of_datatypes
= {
357 .catname
= "pg_catalog.pg_type t",
358 /* selcondition --- ignore table rowtypes and array types */
359 .selcondition
= "(t.typrelid = 0 "
360 " OR (SELECT c.relkind = " CppAsString2(RELKIND_COMPOSITE_TYPE
)
361 " FROM pg_catalog.pg_class c WHERE c.oid = t.typrelid)) "
362 "AND t.typname !~ '^_'",
363 .viscondition
= "pg_catalog.pg_type_is_visible(t.oid)",
364 .namespace = "t.typnamespace",
365 .result
= "pg_catalog.format_type(t.oid, NULL)",
366 .qualresult
= "pg_catalog.quote_ident(t.typname)",
369 static const SchemaQuery Query_for_list_of_composite_datatypes
= {
370 .catname
= "pg_catalog.pg_type t",
371 /* selcondition --- only get composite types */
372 .selcondition
= "(SELECT c.relkind = " CppAsString2(RELKIND_COMPOSITE_TYPE
)
373 " FROM pg_catalog.pg_class c WHERE c.oid = t.typrelid) "
374 "AND t.typname !~ '^_'",
375 .viscondition
= "pg_catalog.pg_type_is_visible(t.oid)",
376 .namespace = "t.typnamespace",
377 .result
= "pg_catalog.format_type(t.oid, NULL)",
378 .qualresult
= "pg_catalog.quote_ident(t.typname)",
381 static const SchemaQuery Query_for_list_of_domains
= {
382 .catname
= "pg_catalog.pg_type t",
383 .selcondition
= "t.typtype = 'd'",
384 .viscondition
= "pg_catalog.pg_type_is_visible(t.oid)",
385 .namespace = "t.typnamespace",
386 .result
= "pg_catalog.quote_ident(t.typname)",
389 /* Note: this intentionally accepts aggregates as well as plain functions */
390 static const SchemaQuery Query_for_list_of_functions
[] = {
392 .min_server_version
= 110000,
393 .catname
= "pg_catalog.pg_proc p",
394 .selcondition
= "p.prokind != 'p'",
395 .viscondition
= "pg_catalog.pg_function_is_visible(p.oid)",
396 .namespace = "p.pronamespace",
397 .result
= "pg_catalog.quote_ident(p.proname)",
400 .catname
= "pg_catalog.pg_proc p",
401 .viscondition
= "pg_catalog.pg_function_is_visible(p.oid)",
402 .namespace = "p.pronamespace",
403 .result
= "pg_catalog.quote_ident(p.proname)",
407 static const SchemaQuery Query_for_list_of_procedures
[] = {
409 .min_server_version
= 110000,
410 .catname
= "pg_catalog.pg_proc p",
411 .selcondition
= "p.prokind = 'p'",
412 .viscondition
= "pg_catalog.pg_function_is_visible(p.oid)",
413 .namespace = "p.pronamespace",
414 .result
= "pg_catalog.quote_ident(p.proname)",
417 /* not supported in older versions */
422 static const SchemaQuery Query_for_list_of_routines
= {
423 .catname
= "pg_catalog.pg_proc p",
424 .viscondition
= "pg_catalog.pg_function_is_visible(p.oid)",
425 .namespace = "p.pronamespace",
426 .result
= "pg_catalog.quote_ident(p.proname)",
429 static const SchemaQuery Query_for_list_of_sequences
= {
430 .catname
= "pg_catalog.pg_class c",
431 .selcondition
= "c.relkind IN (" CppAsString2(RELKIND_SEQUENCE
) ")",
432 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
433 .namespace = "c.relnamespace",
434 .result
= "pg_catalog.quote_ident(c.relname)",
437 static const SchemaQuery Query_for_list_of_foreign_tables
= {
438 .catname
= "pg_catalog.pg_class c",
439 .selcondition
= "c.relkind IN (" CppAsString2(RELKIND_FOREIGN_TABLE
) ")",
440 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
441 .namespace = "c.relnamespace",
442 .result
= "pg_catalog.quote_ident(c.relname)",
445 static const SchemaQuery Query_for_list_of_tables
= {
446 .catname
= "pg_catalog.pg_class c",
448 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
449 CppAsString2(RELKIND_PARTITIONED_TABLE
) ")",
450 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
451 .namespace = "c.relnamespace",
452 .result
= "pg_catalog.quote_ident(c.relname)",
455 static const SchemaQuery Query_for_list_of_partitioned_tables
= {
456 .catname
= "pg_catalog.pg_class c",
457 .selcondition
= "c.relkind IN (" CppAsString2(RELKIND_PARTITIONED_TABLE
) ")",
458 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
459 .namespace = "c.relnamespace",
460 .result
= "pg_catalog.quote_ident(c.relname)",
463 static const SchemaQuery Query_for_list_of_views
= {
464 .catname
= "pg_catalog.pg_class c",
465 .selcondition
= "c.relkind IN (" CppAsString2(RELKIND_VIEW
) ")",
466 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
467 .namespace = "c.relnamespace",
468 .result
= "pg_catalog.quote_ident(c.relname)",
471 static const SchemaQuery Query_for_list_of_matviews
= {
472 .catname
= "pg_catalog.pg_class c",
473 .selcondition
= "c.relkind IN (" CppAsString2(RELKIND_MATVIEW
) ")",
474 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
475 .namespace = "c.relnamespace",
476 .result
= "pg_catalog.quote_ident(c.relname)",
479 static const SchemaQuery Query_for_list_of_indexes
= {
480 .catname
= "pg_catalog.pg_class c",
482 "c.relkind IN (" CppAsString2(RELKIND_INDEX
) ", "
483 CppAsString2(RELKIND_PARTITIONED_INDEX
) ")",
484 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
485 .namespace = "c.relnamespace",
486 .result
= "pg_catalog.quote_ident(c.relname)",
489 static const SchemaQuery Query_for_list_of_partitioned_indexes
= {
490 .catname
= "pg_catalog.pg_class c",
491 .selcondition
= "c.relkind = " CppAsString2(RELKIND_PARTITIONED_INDEX
),
492 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
493 .namespace = "c.relnamespace",
494 .result
= "pg_catalog.quote_ident(c.relname)",
499 static const SchemaQuery Query_for_list_of_relations
= {
500 .catname
= "pg_catalog.pg_class c",
501 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
502 .namespace = "c.relnamespace",
503 .result
= "pg_catalog.quote_ident(c.relname)",
506 /* partitioned relations */
507 static const SchemaQuery Query_for_list_of_partitioned_relations
= {
508 .catname
= "pg_catalog.pg_class c",
509 .selcondition
= "c.relkind IN (" CppAsString2(RELKIND_PARTITIONED_TABLE
)
510 ", " CppAsString2(RELKIND_PARTITIONED_INDEX
) ")",
511 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
512 .namespace = "c.relnamespace",
513 .result
= "pg_catalog.quote_ident(c.relname)",
516 static const SchemaQuery Query_for_list_of_operator_families
= {
517 .catname
= "pg_catalog.pg_opfamily c",
518 .viscondition
= "pg_catalog.pg_opfamily_is_visible(c.oid)",
519 .namespace = "c.opfnamespace",
520 .result
= "pg_catalog.quote_ident(c.opfname)",
523 /* Relations supporting INSERT, UPDATE or DELETE */
524 static const SchemaQuery Query_for_list_of_updatables
= {
525 .catname
= "pg_catalog.pg_class c",
527 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
528 CppAsString2(RELKIND_FOREIGN_TABLE
) ", "
529 CppAsString2(RELKIND_VIEW
) ", "
530 CppAsString2(RELKIND_PARTITIONED_TABLE
) ")",
531 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
532 .namespace = "c.relnamespace",
533 .result
= "pg_catalog.quote_ident(c.relname)",
536 /* Relations supporting SELECT */
537 static const SchemaQuery Query_for_list_of_selectables
= {
538 .catname
= "pg_catalog.pg_class c",
540 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
541 CppAsString2(RELKIND_SEQUENCE
) ", "
542 CppAsString2(RELKIND_VIEW
) ", "
543 CppAsString2(RELKIND_MATVIEW
) ", "
544 CppAsString2(RELKIND_FOREIGN_TABLE
) ", "
545 CppAsString2(RELKIND_PARTITIONED_TABLE
) ")",
546 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
547 .namespace = "c.relnamespace",
548 .result
= "pg_catalog.quote_ident(c.relname)",
551 /* Relations supporting TRUNCATE */
552 static const SchemaQuery Query_for_list_of_truncatables
= {
553 .catname
= "pg_catalog.pg_class c",
555 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
556 CppAsString2(RELKIND_FOREIGN_TABLE
) ", "
557 CppAsString2(RELKIND_PARTITIONED_TABLE
) ")",
558 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
559 .namespace = "c.relnamespace",
560 .result
= "pg_catalog.quote_ident(c.relname)",
563 /* Relations supporting GRANT are currently same as those supporting SELECT */
564 #define Query_for_list_of_grantables Query_for_list_of_selectables
566 /* Relations supporting ANALYZE */
567 static const SchemaQuery Query_for_list_of_analyzables
= {
568 .catname
= "pg_catalog.pg_class c",
570 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
571 CppAsString2(RELKIND_PARTITIONED_TABLE
) ", "
572 CppAsString2(RELKIND_MATVIEW
) ", "
573 CppAsString2(RELKIND_FOREIGN_TABLE
) ")",
574 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
575 .namespace = "c.relnamespace",
576 .result
= "pg_catalog.quote_ident(c.relname)",
579 /* Relations supporting index creation */
580 static const SchemaQuery Query_for_list_of_indexables
= {
581 .catname
= "pg_catalog.pg_class c",
583 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
584 CppAsString2(RELKIND_PARTITIONED_TABLE
) ", "
585 CppAsString2(RELKIND_MATVIEW
) ")",
586 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
587 .namespace = "c.relnamespace",
588 .result
= "pg_catalog.quote_ident(c.relname)",
592 * Relations supporting VACUUM are currently same as those supporting
595 #define Query_for_list_of_vacuumables Query_for_list_of_indexables
597 /* Relations supporting CLUSTER */
598 static const SchemaQuery Query_for_list_of_clusterables
= {
599 .catname
= "pg_catalog.pg_class c",
601 "c.relkind IN (" CppAsString2(RELKIND_RELATION
) ", "
602 CppAsString2(RELKIND_MATVIEW
) ")",
603 .viscondition
= "pg_catalog.pg_table_is_visible(c.oid)",
604 .namespace = "c.relnamespace",
605 .result
= "pg_catalog.quote_ident(c.relname)",
608 static const SchemaQuery Query_for_list_of_constraints_with_schema
= {
609 .catname
= "pg_catalog.pg_constraint c",
610 .selcondition
= "c.conrelid <> 0",
611 .viscondition
= "true", /* there is no pg_constraint_is_visible */
612 .namespace = "c.connamespace",
613 .result
= "pg_catalog.quote_ident(c.conname)",
616 static const SchemaQuery Query_for_list_of_statistics
= {
617 .catname
= "pg_catalog.pg_statistic_ext s",
618 .viscondition
= "pg_catalog.pg_statistics_obj_is_visible(s.oid)",
619 .namespace = "s.stxnamespace",
620 .result
= "pg_catalog.quote_ident(s.stxname)",
623 static const SchemaQuery Query_for_list_of_collations
= {
624 .catname
= "pg_catalog.pg_collation c",
625 .selcondition
= "c.collencoding IN (-1, pg_catalog.pg_char_to_encoding(pg_catalog.getdatabaseencoding()))",
626 .viscondition
= "pg_catalog.pg_collation_is_visible(c.oid)",
627 .namespace = "c.collnamespace",
628 .result
= "pg_catalog.quote_ident(c.collname)",
633 * Queries to get lists of names of various kinds of things, possibly
634 * restricted to names matching a partially entered name. In these queries,
635 * the first %s will be replaced by the text entered so far (suitably escaped
636 * to become a SQL literal string). %d will be replaced by the length of the
637 * string (in unescaped form). A second and third %s, if present, will be
638 * replaced by a suitably-escaped version of the string provided in
639 * completion_info_charp. A fourth and fifth %s are similarly replaced by
640 * completion_info_charp2.
642 * Beware that the allowed sequences of %s and %d are determined by
643 * _complete_from_query().
646 #define Query_for_list_of_attributes \
647 "SELECT pg_catalog.quote_ident(attname) "\
648 " FROM pg_catalog.pg_attribute a, pg_catalog.pg_class c "\
649 " WHERE c.oid = a.attrelid "\
650 " AND a.attnum > 0 "\
651 " AND NOT a.attisdropped "\
652 " AND substring(pg_catalog.quote_ident(attname),1,%d)='%s' "\
653 " AND (pg_catalog.quote_ident(relname)='%s' "\
654 " OR '\"' || relname || '\"'='%s') "\
655 " AND pg_catalog.pg_table_is_visible(c.oid)"
657 #define Query_for_list_of_attribute_numbers \
659 " FROM pg_catalog.pg_attribute a, pg_catalog.pg_class c "\
660 " WHERE c.oid = a.attrelid "\
661 " AND a.attnum > 0 "\
662 " AND NOT a.attisdropped "\
663 " AND substring(attnum::pg_catalog.text,1,%d)='%s' "\
664 " AND (pg_catalog.quote_ident(relname)='%s' "\
665 " OR '\"' || relname || '\"'='%s') "\
666 " AND pg_catalog.pg_table_is_visible(c.oid)"
668 #define Query_for_list_of_attributes_with_schema \
669 "SELECT pg_catalog.quote_ident(attname) "\
670 " FROM pg_catalog.pg_attribute a, pg_catalog.pg_class c, pg_catalog.pg_namespace n "\
671 " WHERE c.oid = a.attrelid "\
672 " AND n.oid = c.relnamespace "\
673 " AND a.attnum > 0 "\
674 " AND NOT a.attisdropped "\
675 " AND substring(pg_catalog.quote_ident(attname),1,%d)='%s' "\
676 " AND (pg_catalog.quote_ident(relname)='%s' "\
677 " OR '\"' || relname || '\"' ='%s') "\
678 " AND (pg_catalog.quote_ident(nspname)='%s' "\
679 " OR '\"' || nspname || '\"' ='%s') "
681 #define Query_for_list_of_enum_values \
682 "SELECT pg_catalog.quote_literal(enumlabel) "\
683 " FROM pg_catalog.pg_enum e, pg_catalog.pg_type t "\
684 " WHERE t.oid = e.enumtypid "\
685 " AND substring(pg_catalog.quote_literal(enumlabel),1,%d)='%s' "\
686 " AND (pg_catalog.quote_ident(typname)='%s' "\
687 " OR '\"' || typname || '\"'='%s') "\
688 " AND pg_catalog.pg_type_is_visible(t.oid)"
690 #define Query_for_list_of_enum_values_with_schema \
691 "SELECT pg_catalog.quote_literal(enumlabel) "\
692 " FROM pg_catalog.pg_enum e, pg_catalog.pg_type t, pg_catalog.pg_namespace n "\
693 " WHERE t.oid = e.enumtypid "\
694 " AND n.oid = t.typnamespace "\
695 " AND substring(pg_catalog.quote_literal(enumlabel),1,%d)='%s' "\
696 " AND (pg_catalog.quote_ident(typname)='%s' "\
697 " OR '\"' || typname || '\"'='%s') "\
698 " AND (pg_catalog.quote_ident(nspname)='%s' "\
699 " OR '\"' || nspname || '\"' ='%s') "
701 #define Query_for_list_of_template_databases \
702 "SELECT pg_catalog.quote_ident(d.datname) "\
703 " FROM pg_catalog.pg_database d "\
704 " WHERE substring(pg_catalog.quote_ident(d.datname),1,%d)='%s' "\
705 " AND (d.datistemplate OR pg_catalog.pg_has_role(d.datdba, 'USAGE'))"
707 #define Query_for_list_of_databases \
708 "SELECT pg_catalog.quote_ident(datname) FROM pg_catalog.pg_database "\
709 " WHERE substring(pg_catalog.quote_ident(datname),1,%d)='%s'"
711 #define Query_for_list_of_tablespaces \
712 "SELECT pg_catalog.quote_ident(spcname) FROM pg_catalog.pg_tablespace "\
713 " WHERE substring(pg_catalog.quote_ident(spcname),1,%d)='%s'"
715 #define Query_for_list_of_encodings \
716 " SELECT DISTINCT pg_catalog.pg_encoding_to_char(conforencoding) "\
717 " FROM pg_catalog.pg_conversion "\
718 " WHERE substring(pg_catalog.pg_encoding_to_char(conforencoding),1,%d)=UPPER('%s')"
720 #define Query_for_list_of_languages \
721 "SELECT pg_catalog.quote_ident(lanname) "\
722 " FROM pg_catalog.pg_language "\
723 " WHERE lanname != 'internal' "\
724 " AND substring(pg_catalog.quote_ident(lanname),1,%d)='%s'"
726 #define Query_for_list_of_schemas \
727 "SELECT pg_catalog.quote_ident(nspname) FROM pg_catalog.pg_namespace "\
728 " WHERE substring(pg_catalog.quote_ident(nspname),1,%d)='%s'"
730 #define Query_for_list_of_alter_system_set_vars \
732 " (SELECT pg_catalog.lower(name) AS name FROM pg_catalog.pg_settings "\
733 " WHERE context != 'internal' "\
734 " UNION ALL SELECT 'all') ss "\
735 " WHERE substring(name,1,%d)='%s'"
737 #define Query_for_list_of_set_vars \
739 " (SELECT pg_catalog.lower(name) AS name FROM pg_catalog.pg_settings "\
740 " WHERE context IN ('user', 'superuser') "\
741 " UNION ALL SELECT 'constraints' "\
742 " UNION ALL SELECT 'transaction' "\
743 " UNION ALL SELECT 'session' "\
744 " UNION ALL SELECT 'role' "\
745 " UNION ALL SELECT 'tablespace' "\
746 " UNION ALL SELECT 'all') ss "\
747 " WHERE substring(name,1,%d)='%s'"
749 #define Query_for_list_of_show_vars \
751 " (SELECT pg_catalog.lower(name) AS name FROM pg_catalog.pg_settings "\
752 " UNION ALL SELECT 'session authorization' "\
753 " UNION ALL SELECT 'all') ss "\
754 " WHERE substring(name,1,%d)='%s'"
756 #define Query_for_list_of_roles \
757 " SELECT pg_catalog.quote_ident(rolname) "\
758 " FROM pg_catalog.pg_roles "\
759 " WHERE substring(pg_catalog.quote_ident(rolname),1,%d)='%s'"
761 #define Query_for_list_of_grant_roles \
762 " SELECT pg_catalog.quote_ident(rolname) "\
763 " FROM pg_catalog.pg_roles "\
764 " WHERE substring(pg_catalog.quote_ident(rolname),1,%d)='%s'"\
765 " UNION ALL SELECT 'PUBLIC'"\
766 " UNION ALL SELECT 'CURRENT_ROLE'"\
767 " UNION ALL SELECT 'CURRENT_USER'"\
768 " UNION ALL SELECT 'SESSION_USER'"
770 /* the silly-looking length condition is just to eat up the current word */
771 #define Query_for_index_of_table \
772 "SELECT pg_catalog.quote_ident(c2.relname) "\
773 " FROM pg_catalog.pg_class c1, pg_catalog.pg_class c2, pg_catalog.pg_index i"\
774 " WHERE c1.oid=i.indrelid and i.indexrelid=c2.oid"\
775 " and (%d = pg_catalog.length('%s'))"\
776 " and pg_catalog.quote_ident(c1.relname)='%s'"\
777 " and pg_catalog.pg_table_is_visible(c2.oid)"
779 #define Query_for_unique_index_of_table \
780 Query_for_index_of_table \
783 /* the silly-looking length condition is just to eat up the current word */
784 #define Query_for_constraint_of_table \
785 "SELECT pg_catalog.quote_ident(conname) "\
786 " FROM pg_catalog.pg_class c1, pg_catalog.pg_constraint con "\
787 " WHERE c1.oid=conrelid and (%d = pg_catalog.length('%s'))"\
788 " and pg_catalog.quote_ident(c1.relname)='%s'"\
789 " and pg_catalog.pg_table_is_visible(c1.oid)"
791 #define Query_for_all_table_constraints \
792 "SELECT pg_catalog.quote_ident(conname) "\
793 " FROM pg_catalog.pg_constraint c "\
794 " WHERE c.conrelid <> 0 "
796 /* the silly-looking length condition is just to eat up the current word */
797 #define Query_for_constraint_of_type \
798 "SELECT pg_catalog.quote_ident(conname) "\
799 " FROM pg_catalog.pg_type t, pg_catalog.pg_constraint con "\
800 " WHERE t.oid=contypid and (%d = pg_catalog.length('%s'))"\
801 " and pg_catalog.quote_ident(t.typname)='%s'"\
802 " and pg_catalog.pg_type_is_visible(t.oid)"
804 /* the silly-looking length condition is just to eat up the current word */
805 #define Query_for_list_of_tables_for_constraint \
806 "SELECT pg_catalog.quote_ident(relname) "\
807 " FROM pg_catalog.pg_class"\
808 " WHERE (%d = pg_catalog.length('%s'))"\
810 " (SELECT conrelid FROM pg_catalog.pg_constraint "\
811 " WHERE pg_catalog.quote_ident(conname)='%s')"
813 /* the silly-looking length condition is just to eat up the current word */
814 #define Query_for_rule_of_table \
815 "SELECT pg_catalog.quote_ident(rulename) "\
816 " FROM pg_catalog.pg_class c1, pg_catalog.pg_rewrite "\
817 " WHERE c1.oid=ev_class and (%d = pg_catalog.length('%s'))"\
818 " and pg_catalog.quote_ident(c1.relname)='%s'"\
819 " and pg_catalog.pg_table_is_visible(c1.oid)"
821 /* the silly-looking length condition is just to eat up the current word */
822 #define Query_for_list_of_tables_for_rule \
823 "SELECT pg_catalog.quote_ident(relname) "\
824 " FROM pg_catalog.pg_class"\
825 " WHERE (%d = pg_catalog.length('%s'))"\
827 " (SELECT ev_class FROM pg_catalog.pg_rewrite "\
828 " WHERE pg_catalog.quote_ident(rulename)='%s')"
830 /* the silly-looking length condition is just to eat up the current word */
831 #define Query_for_trigger_of_table \
832 "SELECT pg_catalog.quote_ident(tgname) "\
833 " FROM pg_catalog.pg_class c1, pg_catalog.pg_trigger "\
834 " WHERE c1.oid=tgrelid and (%d = pg_catalog.length('%s'))"\
835 " and pg_catalog.quote_ident(c1.relname)='%s'"\
836 " and pg_catalog.pg_table_is_visible(c1.oid)"\
837 " and not tgisinternal"
839 /* the silly-looking length condition is just to eat up the current word */
840 #define Query_for_list_of_tables_for_trigger \
841 "SELECT pg_catalog.quote_ident(relname) "\
842 " FROM pg_catalog.pg_class"\
843 " WHERE (%d = pg_catalog.length('%s'))"\
845 " (SELECT tgrelid FROM pg_catalog.pg_trigger "\
846 " WHERE pg_catalog.quote_ident(tgname)='%s')"
848 #define Query_for_list_of_ts_configurations \
849 "SELECT pg_catalog.quote_ident(cfgname) FROM pg_catalog.pg_ts_config "\
850 " WHERE substring(pg_catalog.quote_ident(cfgname),1,%d)='%s'"
852 #define Query_for_list_of_ts_dictionaries \
853 "SELECT pg_catalog.quote_ident(dictname) FROM pg_catalog.pg_ts_dict "\
854 " WHERE substring(pg_catalog.quote_ident(dictname),1,%d)='%s'"
856 #define Query_for_list_of_ts_parsers \
857 "SELECT pg_catalog.quote_ident(prsname) FROM pg_catalog.pg_ts_parser "\
858 " WHERE substring(pg_catalog.quote_ident(prsname),1,%d)='%s'"
860 #define Query_for_list_of_ts_templates \
861 "SELECT pg_catalog.quote_ident(tmplname) FROM pg_catalog.pg_ts_template "\
862 " WHERE substring(pg_catalog.quote_ident(tmplname),1,%d)='%s'"
864 #define Query_for_list_of_fdws \
865 " SELECT pg_catalog.quote_ident(fdwname) "\
866 " FROM pg_catalog.pg_foreign_data_wrapper "\
867 " WHERE substring(pg_catalog.quote_ident(fdwname),1,%d)='%s'"
869 #define Query_for_list_of_servers \
870 " SELECT pg_catalog.quote_ident(srvname) "\
871 " FROM pg_catalog.pg_foreign_server "\
872 " WHERE substring(pg_catalog.quote_ident(srvname),1,%d)='%s'"
874 #define Query_for_list_of_user_mappings \
875 " SELECT pg_catalog.quote_ident(usename) "\
876 " FROM pg_catalog.pg_user_mappings "\
877 " WHERE substring(pg_catalog.quote_ident(usename),1,%d)='%s'"
879 #define Query_for_list_of_access_methods \
880 " SELECT pg_catalog.quote_ident(amname) "\
881 " FROM pg_catalog.pg_am "\
882 " WHERE substring(pg_catalog.quote_ident(amname),1,%d)='%s'"
884 #define Query_for_list_of_index_access_methods \
885 " SELECT pg_catalog.quote_ident(amname) "\
886 " FROM pg_catalog.pg_am "\
887 " WHERE substring(pg_catalog.quote_ident(amname),1,%d)='%s' AND "\
888 " amtype=" CppAsString2(AMTYPE_INDEX)
890 #define Query_for_list_of_table_access_methods \
891 " SELECT pg_catalog.quote_ident(amname) "\
892 " FROM pg_catalog.pg_am "\
893 " WHERE substring(pg_catalog.quote_ident(amname),1,%d)='%s' AND "\
894 " amtype=" CppAsString2(AMTYPE_TABLE)
896 /* the silly-looking length condition is just to eat up the current word */
897 #define Query_for_list_of_arguments \
898 "SELECT pg_catalog.oidvectortypes(proargtypes)||')' "\
899 " FROM pg_catalog.pg_proc "\
900 " WHERE (%d = pg_catalog.length('%s'))"\
901 " AND (pg_catalog.quote_ident(proname)='%s'"\
902 " OR '\"' || proname || '\"'='%s') "\
903 " AND (pg_catalog.pg_function_is_visible(pg_proc.oid))"
905 /* the silly-looking length condition is just to eat up the current word */
906 #define Query_for_list_of_arguments_with_schema \
907 "SELECT pg_catalog.oidvectortypes(proargtypes)||')' "\
908 " FROM pg_catalog.pg_proc p, pg_catalog.pg_namespace n "\
909 " WHERE (%d = pg_catalog.length('%s'))"\
910 " AND n.oid = p.pronamespace "\
911 " AND (pg_catalog.quote_ident(proname)='%s' "\
912 " OR '\"' || proname || '\"' ='%s') "\
913 " AND (pg_catalog.quote_ident(nspname)='%s' "\
914 " OR '\"' || nspname || '\"' ='%s') "
916 #define Query_for_list_of_extensions \
917 " SELECT pg_catalog.quote_ident(extname) "\
918 " FROM pg_catalog.pg_extension "\
919 " WHERE substring(pg_catalog.quote_ident(extname),1,%d)='%s'"
921 #define Query_for_list_of_available_extensions \
922 " SELECT pg_catalog.quote_ident(name) "\
923 " FROM pg_catalog.pg_available_extensions "\
924 " WHERE substring(pg_catalog.quote_ident(name),1,%d)='%s' AND installed_version IS NULL"
926 /* the silly-looking length condition is just to eat up the current word */
927 #define Query_for_list_of_available_extension_versions \
928 " SELECT pg_catalog.quote_ident(version) "\
929 " FROM pg_catalog.pg_available_extension_versions "\
930 " WHERE (%d = pg_catalog.length('%s'))"\
931 " AND pg_catalog.quote_ident(name)='%s'"
933 /* the silly-looking length condition is just to eat up the current word */
934 #define Query_for_list_of_available_extension_versions_with_TO \
935 " SELECT 'TO ' || pg_catalog.quote_ident(version) "\
936 " FROM pg_catalog.pg_available_extension_versions "\
937 " WHERE (%d = pg_catalog.length('%s'))"\
938 " AND pg_catalog.quote_ident(name)='%s'"
940 #define Query_for_list_of_prepared_statements \
941 " SELECT pg_catalog.quote_ident(name) "\
942 " FROM pg_catalog.pg_prepared_statements "\
943 " WHERE substring(pg_catalog.quote_ident(name),1,%d)='%s'"
945 #define Query_for_list_of_event_triggers \
946 " SELECT pg_catalog.quote_ident(evtname) "\
947 " FROM pg_catalog.pg_event_trigger "\
948 " WHERE substring(pg_catalog.quote_ident(evtname),1,%d)='%s'"
950 #define Query_for_list_of_tablesample_methods \
951 " SELECT pg_catalog.quote_ident(proname) "\
952 " FROM pg_catalog.pg_proc "\
953 " WHERE prorettype = 'pg_catalog.tsm_handler'::pg_catalog.regtype AND "\
954 " proargtypes[0] = 'pg_catalog.internal'::pg_catalog.regtype AND "\
955 " substring(pg_catalog.quote_ident(proname),1,%d)='%s'"
957 #define Query_for_list_of_policies \
958 " SELECT pg_catalog.quote_ident(polname) "\
959 " FROM pg_catalog.pg_policy "\
960 " WHERE substring(pg_catalog.quote_ident(polname),1,%d)='%s'"
962 #define Query_for_list_of_tables_for_policy \
963 "SELECT pg_catalog.quote_ident(relname) "\
964 " FROM pg_catalog.pg_class"\
965 " WHERE (%d = pg_catalog.length('%s'))"\
967 " (SELECT polrelid FROM pg_catalog.pg_policy "\
968 " WHERE pg_catalog.quote_ident(polname)='%s')"
970 #define Query_for_enum \
971 " SELECT name FROM ( "\
972 " SELECT pg_catalog.quote_ident(pg_catalog.unnest(enumvals)) AS name "\
973 " FROM pg_catalog.pg_settings "\
974 " WHERE pg_catalog.lower(name)=pg_catalog.lower('%s') "\
976 " SELECT 'DEFAULT' ) ss "\
977 " WHERE pg_catalog.substring(name,1,%%d)='%%s'"
979 /* the silly-looking length condition is just to eat up the current word */
980 #define Query_for_partition_of_table \
981 "SELECT pg_catalog.quote_ident(c2.relname) "\
982 " FROM pg_catalog.pg_class c1, pg_catalog.pg_class c2, pg_catalog.pg_inherits i"\
983 " WHERE c1.oid=i.inhparent and i.inhrelid=c2.oid"\
984 " and (%d = pg_catalog.length('%s'))"\
985 " and pg_catalog.quote_ident(c1.relname)='%s'"\
986 " and pg_catalog.pg_table_is_visible(c2.oid)"\
987 " and c2.relispartition = 'true'"
989 #define Query_for_list_of_cursors \
990 " SELECT pg_catalog.quote_ident(name) "\
991 " FROM pg_catalog.pg_cursors "\
992 " WHERE substring(pg_catalog.quote_ident(name),1,%d)='%s'"
995 * These object types were introduced later than our support cutoff of
996 * server version 7.4. We use the VersionedQuery infrastructure so that
997 * we don't send certain-to-fail queries to older servers.
1000 static const VersionedQuery Query_for_list_of_publications
[] = {
1002 " SELECT pg_catalog.quote_ident(pubname) "
1003 " FROM pg_catalog.pg_publication "
1004 " WHERE substring(pg_catalog.quote_ident(pubname),1,%d)='%s'"
1009 static const VersionedQuery Query_for_list_of_subscriptions
[] = {
1011 " SELECT pg_catalog.quote_ident(s.subname) "
1012 " FROM pg_catalog.pg_subscription s, pg_catalog.pg_database d "
1013 " WHERE substring(pg_catalog.quote_ident(s.subname),1,%d)='%s' "
1014 " AND d.datname = pg_catalog.current_database() "
1015 " AND s.subdbid = d.oid"
1021 * This is a list of all "things" in Pgsql, which can show up after CREATE or
1022 * DROP; and there is also a query to get a list of them.
1028 const char *query
; /* simple query, or NULL */
1029 const VersionedQuery
*vquery
; /* versioned query, or NULL */
1030 const SchemaQuery
*squery
; /* schema query, or NULL */
1031 const bits32 flags
; /* visibility flags, see below */
1034 #define THING_NO_CREATE (1 << 0) /* should not show up after CREATE */
1035 #define THING_NO_DROP (1 << 1) /* should not show up after DROP */
1036 #define THING_NO_ALTER (1 << 2) /* should not show up after ALTER */
1037 #define THING_NO_SHOW (THING_NO_CREATE | THING_NO_DROP | THING_NO_ALTER)
1039 static const pgsql_thing_t words_after_create
[] = {
1040 {"ACCESS METHOD", NULL
, NULL
, NULL
, THING_NO_ALTER
},
1041 {"AGGREGATE", NULL
, NULL
, Query_for_list_of_aggregates
},
1042 {"CAST", NULL
, NULL
, NULL
}, /* Casts have complex structures for names, so
1044 {"COLLATION", NULL
, NULL
, &Query_for_list_of_collations
},
1047 * CREATE CONSTRAINT TRIGGER is not supported here because it is designed
1048 * to be used only by pg_dump.
1050 {"CONFIGURATION", Query_for_list_of_ts_configurations
, NULL
, NULL
, THING_NO_SHOW
},
1051 {"CONVERSION", "SELECT pg_catalog.quote_ident(conname) FROM pg_catalog.pg_conversion WHERE substring(pg_catalog.quote_ident(conname),1,%d)='%s'"},
1052 {"DATABASE", Query_for_list_of_databases
},
1053 {"DEFAULT PRIVILEGES", NULL
, NULL
, NULL
, THING_NO_CREATE
| THING_NO_DROP
},
1054 {"DICTIONARY", Query_for_list_of_ts_dictionaries
, NULL
, NULL
, THING_NO_SHOW
},
1055 {"DOMAIN", NULL
, NULL
, &Query_for_list_of_domains
},
1056 {"EVENT TRIGGER", NULL
, NULL
, NULL
},
1057 {"EXTENSION", Query_for_list_of_extensions
},
1058 {"FOREIGN DATA WRAPPER", NULL
, NULL
, NULL
},
1059 {"FOREIGN TABLE", NULL
, NULL
, NULL
},
1060 {"FUNCTION", NULL
, NULL
, Query_for_list_of_functions
},
1061 {"GROUP", Query_for_list_of_roles
},
1062 {"INDEX", NULL
, NULL
, &Query_for_list_of_indexes
},
1063 {"LANGUAGE", Query_for_list_of_languages
},
1064 {"LARGE OBJECT", NULL
, NULL
, NULL
, THING_NO_CREATE
| THING_NO_DROP
},
1065 {"MATERIALIZED VIEW", NULL
, NULL
, &Query_for_list_of_matviews
},
1066 {"OPERATOR", NULL
, NULL
, NULL
}, /* Querying for this is probably not such
1068 {"OR REPLACE", NULL
, NULL
, NULL
, THING_NO_DROP
| THING_NO_ALTER
},
1069 {"OWNED", NULL
, NULL
, NULL
, THING_NO_CREATE
| THING_NO_ALTER
}, /* for DROP OWNED BY ... */
1070 {"PARSER", Query_for_list_of_ts_parsers
, NULL
, NULL
, THING_NO_SHOW
},
1071 {"POLICY", NULL
, NULL
, NULL
},
1072 {"PROCEDURE", NULL
, NULL
, Query_for_list_of_procedures
},
1073 {"PUBLICATION", NULL
, Query_for_list_of_publications
},
1074 {"ROLE", Query_for_list_of_roles
},
1075 {"ROUTINE", NULL
, NULL
, &Query_for_list_of_routines
, THING_NO_CREATE
},
1076 {"RULE", "SELECT pg_catalog.quote_ident(rulename) FROM pg_catalog.pg_rules WHERE substring(pg_catalog.quote_ident(rulename),1,%d)='%s'"},
1077 {"SCHEMA", Query_for_list_of_schemas
},
1078 {"SEQUENCE", NULL
, NULL
, &Query_for_list_of_sequences
},
1079 {"SERVER", Query_for_list_of_servers
},
1080 {"STATISTICS", NULL
, NULL
, &Query_for_list_of_statistics
},
1081 {"SUBSCRIPTION", NULL
, Query_for_list_of_subscriptions
},
1082 {"SYSTEM", NULL
, NULL
, NULL
, THING_NO_CREATE
| THING_NO_DROP
},
1083 {"TABLE", NULL
, NULL
, &Query_for_list_of_tables
},
1084 {"TABLESPACE", Query_for_list_of_tablespaces
},
1085 {"TEMP", NULL
, NULL
, NULL
, THING_NO_DROP
| THING_NO_ALTER
}, /* for CREATE TEMP TABLE
1087 {"TEMPLATE", Query_for_list_of_ts_templates
, NULL
, NULL
, THING_NO_SHOW
},
1088 {"TEMPORARY", NULL
, NULL
, NULL
, THING_NO_DROP
| THING_NO_ALTER
}, /* for CREATE TEMPORARY
1090 {"TEXT SEARCH", NULL
, NULL
, NULL
},
1091 {"TRANSFORM", NULL
, NULL
, NULL
},
1092 {"TRIGGER", "SELECT pg_catalog.quote_ident(tgname) FROM pg_catalog.pg_trigger WHERE substring(pg_catalog.quote_ident(tgname),1,%d)='%s' AND NOT tgisinternal"},
1093 {"TYPE", NULL
, NULL
, &Query_for_list_of_datatypes
},
1094 {"UNIQUE", NULL
, NULL
, NULL
, THING_NO_DROP
| THING_NO_ALTER
}, /* for CREATE UNIQUE
1096 {"UNLOGGED", NULL
, NULL
, NULL
, THING_NO_DROP
| THING_NO_ALTER
}, /* for CREATE UNLOGGED
1098 {"USER", Query_for_list_of_roles
" UNION SELECT 'MAPPING FOR'"},
1099 {"USER MAPPING FOR", NULL
, NULL
, NULL
},
1100 {"VIEW", NULL
, NULL
, &Query_for_list_of_views
},
1101 {NULL
} /* end of list */
1104 /* Storage parameters for CREATE TABLE and ALTER TABLE */
1105 static const char *const table_storage_parameters
[] = {
1106 "autovacuum_analyze_scale_factor",
1107 "autovacuum_analyze_threshold",
1108 "autovacuum_enabled",
1109 "autovacuum_freeze_max_age",
1110 "autovacuum_freeze_min_age",
1111 "autovacuum_freeze_table_age",
1112 "autovacuum_multixact_freeze_max_age",
1113 "autovacuum_multixact_freeze_min_age",
1114 "autovacuum_multixact_freeze_table_age",
1115 "autovacuum_vacuum_cost_delay",
1116 "autovacuum_vacuum_cost_limit",
1117 "autovacuum_vacuum_insert_scale_factor",
1118 "autovacuum_vacuum_insert_threshold",
1119 "autovacuum_vacuum_scale_factor",
1120 "autovacuum_vacuum_threshold",
1122 "log_autovacuum_min_duration",
1124 "toast.autovacuum_enabled",
1125 "toast.autovacuum_freeze_max_age",
1126 "toast.autovacuum_freeze_min_age",
1127 "toast.autovacuum_freeze_table_age",
1128 "toast.autovacuum_multixact_freeze_max_age",
1129 "toast.autovacuum_multixact_freeze_min_age",
1130 "toast.autovacuum_multixact_freeze_table_age",
1131 "toast.autovacuum_vacuum_cost_delay",
1132 "toast.autovacuum_vacuum_cost_limit",
1133 "toast.autovacuum_vacuum_insert_scale_factor",
1134 "toast.autovacuum_vacuum_insert_threshold",
1135 "toast.autovacuum_vacuum_scale_factor",
1136 "toast.autovacuum_vacuum_threshold",
1137 "toast.log_autovacuum_min_duration",
1138 "toast.vacuum_index_cleanup",
1139 "toast.vacuum_truncate",
1140 "toast_tuple_target",
1141 "user_catalog_table",
1142 "vacuum_index_cleanup",
1148 /* Forward declaration of functions */
1149 static char **psql_completion(const char *text
, int start
, int end
);
1150 static char *create_command_generator(const char *text
, int state
);
1151 static char *drop_command_generator(const char *text
, int state
);
1152 static char *alter_command_generator(const char *text
, int state
);
1153 static char *complete_from_query(const char *text
, int state
);
1154 static char *complete_from_versioned_query(const char *text
, int state
);
1155 static char *complete_from_schema_query(const char *text
, int state
);
1156 static char *complete_from_versioned_schema_query(const char *text
, int state
);
1157 static char *_complete_from_query(const char *simple_query
,
1158 const SchemaQuery
*schema_query
,
1159 const char *text
, int state
);
1160 static char *complete_from_list(const char *text
, int state
);
1161 static char *complete_from_const(const char *text
, int state
);
1162 static void append_variable_names(char ***varnames
, int *nvars
,
1163 int *maxvars
, const char *varname
,
1164 const char *prefix
, const char *suffix
);
1165 static char **complete_from_variables(const char *text
,
1166 const char *prefix
, const char *suffix
, bool need_value
);
1167 static char *complete_from_files(const char *text
, int state
);
1169 static char *pg_strdup_keyword_case(const char *s
, const char *ref
);
1170 static char *escape_string(const char *text
);
1171 static PGresult
*exec_query(const char *query
);
1173 static char **get_previous_words(int point
, char **buffer
, int *nwords
);
1175 static char *get_guctype(const char *varname
);
1177 #ifdef USE_FILENAME_QUOTING_FUNCTIONS
1178 static char *quote_file_name(char *fname
, int match_type
, char *quote_pointer
);
1179 static char *dequote_file_name(char *fname
, int quote_char
);
1184 * Initialize the readline library for our purposes.
1187 initialize_readline(void)
1189 rl_readline_name
= (char *) pset
.progname
;
1190 rl_attempted_completion_function
= psql_completion
;
1192 #ifdef USE_FILENAME_QUOTING_FUNCTIONS
1193 rl_filename_quoting_function
= quote_file_name
;
1194 rl_filename_dequoting_function
= dequote_file_name
;
1197 rl_basic_word_break_characters
= WORD_BREAKS
;
1200 * We should include '"' in rl_completer_quote_characters too, but that
1201 * will require some upgrades to how we handle quoted identifiers, so
1202 * that's for another day.
1204 rl_completer_quote_characters
= "'";
1207 * Set rl_filename_quote_characters to "all possible characters",
1208 * otherwise Readline will skip filename quoting if it thinks a filename
1209 * doesn't need quoting. Readline actually interprets this as bytes, so
1210 * there are no encoding considerations here.
1212 #ifdef HAVE_RL_FILENAME_QUOTE_CHARACTERS
1214 unsigned char *fqc
= (unsigned char *) pg_malloc(256);
1216 for (int i
= 0; i
< 255; i
++)
1217 fqc
[i
] = (unsigned char) (i
+ 1);
1219 rl_filename_quote_characters
= (const char *) fqc
;
1223 completion_max_records
= 1000;
1226 * There is a variable rl_completion_query_items for this but apparently
1227 * it's not defined everywhere.
1232 * Check if 'word' matches any of the '|'-separated strings in 'pattern',
1233 * using case-insensitive or case-sensitive comparisons.
1235 * If pattern is NULL, it's a wild card that matches any word.
1236 * If pattern begins with '!', the result is negated, ie we check that 'word'
1237 * does *not* match any alternative appearing in the rest of 'pattern'.
1238 * Any alternative can contain '*' which is a wild card, i.e., it can match
1239 * any substring; however, we allow at most one '*' per alternative.
1241 * For readability, callers should use the macros MatchAny and MatchAnyExcept
1242 * to invoke those two special cases for 'pattern'. (But '|' and '*' must
1243 * just be written directly in patterns.)
1245 #define MatchAny NULL
1246 #define MatchAnyExcept(pattern) ("!" pattern)
1249 word_matches(const char *pattern
,
1251 bool case_sensitive
)
1255 #define cimatch(s1, s2, n) \
1256 (case_sensitive ? strncmp(s1, s2, n) == 0 : pg_strncasecmp(s1, s2, n) == 0)
1258 /* NULL pattern matches anything. */
1259 if (pattern
== NULL
)
1262 /* Handle negated patterns from the MatchAnyExcept macro. */
1263 if (*pattern
== '!')
1264 return !word_matches(pattern
+ 1, word
, case_sensitive
);
1266 /* Else consider each alternative in the pattern. */
1267 wordlen
= strlen(word
);
1270 const char *star
= NULL
;
1273 /* Find end of current alternative, and locate any wild card. */
1275 while (*c
!= '\0' && *c
!= '|')
1281 /* Was there a wild card? */
1284 /* Yes, wildcard match? */
1285 size_t beforelen
= star
- pattern
,
1286 afterlen
= c
- star
- 1;
1288 if (wordlen
>= (beforelen
+ afterlen
) &&
1289 cimatch(word
, pattern
, beforelen
) &&
1290 cimatch(word
+ wordlen
- afterlen
, star
+ 1, afterlen
))
1295 /* No, plain match? */
1296 if (wordlen
== (c
- pattern
) &&
1297 cimatch(word
, pattern
, wordlen
))
1300 /* Out of alternatives? */
1303 /* Nope, try next alternative. */
1311 * Implementation of TailMatches and TailMatchesCS macros: do the last N words
1312 * in previous_words match the variadic arguments?
1314 * The array indexing might look backwards, but remember that
1315 * previous_words[0] contains the *last* word on the line, not the first.
1318 TailMatchesImpl(bool case_sensitive
,
1319 int previous_words_count
, char **previous_words
,
1324 if (previous_words_count
< narg
)
1327 va_start(args
, narg
);
1329 for (int argno
= 0; argno
< narg
; argno
++)
1331 const char *arg
= va_arg(args
, const char *);
1333 if (!word_matches(arg
, previous_words
[narg
- argno
- 1],
1347 * Implementation of Matches and MatchesCS macros: do all of the words
1348 * in previous_words match the variadic arguments?
1351 MatchesImpl(bool case_sensitive
,
1352 int previous_words_count
, char **previous_words
,
1357 if (previous_words_count
!= narg
)
1360 va_start(args
, narg
);
1362 for (int argno
= 0; argno
< narg
; argno
++)
1364 const char *arg
= va_arg(args
, const char *);
1366 if (!word_matches(arg
, previous_words
[narg
- argno
- 1],
1380 * Implementation of HeadMatches and HeadMatchesCS macros: do the first N
1381 * words in previous_words match the variadic arguments?
1384 HeadMatchesImpl(bool case_sensitive
,
1385 int previous_words_count
, char **previous_words
,
1390 if (previous_words_count
< narg
)
1393 va_start(args
, narg
);
1395 for (int argno
= 0; argno
< narg
; argno
++)
1397 const char *arg
= va_arg(args
, const char *);
1399 if (!word_matches(arg
, previous_words
[previous_words_count
- argno
- 1],
1413 * Check if the final character of 's' is 'c'.
1416 ends_with(const char *s
, char c
)
1418 size_t length
= strlen(s
);
1420 return (length
> 0 && s
[length
- 1] == c
);
1424 * The completion function.
1426 * According to readline spec this gets passed the text entered so far and its
1427 * start and end positions in the readline buffer. The return value is some
1428 * partially obscure list format that can be generated by readline's
1429 * rl_completion_matches() function, so we don't have to worry about it.
1432 psql_completion(const char *text
, int start
, int end
)
1434 /* This is the variable we'll return. */
1435 char **matches
= NULL
;
1437 /* Workspace for parsed words. */
1440 /* This array will contain pointers to parsed words. */
1441 char **previous_words
;
1443 /* The number of words found on the input line. */
1444 int previous_words_count
;
1447 * For compactness, we use these macros to reference previous_words[].
1448 * Caution: do not access a previous_words[] entry without having checked
1449 * previous_words_count to be sure it's valid. In most cases below, that
1450 * check is implicit in a TailMatches() or similar macro, but in some
1451 * places we have to check it explicitly.
1453 #define prev_wd (previous_words[0])
1454 #define prev2_wd (previous_words[1])
1455 #define prev3_wd (previous_words[2])
1456 #define prev4_wd (previous_words[3])
1457 #define prev5_wd (previous_words[4])
1458 #define prev6_wd (previous_words[5])
1459 #define prev7_wd (previous_words[6])
1460 #define prev8_wd (previous_words[7])
1461 #define prev9_wd (previous_words[8])
1463 /* Match the last N words before point, case-insensitively. */
1464 #define TailMatches(...) \
1465 TailMatchesImpl(false, previous_words_count, previous_words, \
1466 VA_ARGS_NARGS(__VA_ARGS__), __VA_ARGS__)
1468 /* Match the last N words before point, case-sensitively. */
1469 #define TailMatchesCS(...) \
1470 TailMatchesImpl(true, previous_words_count, previous_words, \
1471 VA_ARGS_NARGS(__VA_ARGS__), __VA_ARGS__)
1473 /* Match N words representing all of the line, case-insensitively. */
1474 #define Matches(...) \
1475 MatchesImpl(false, previous_words_count, previous_words, \
1476 VA_ARGS_NARGS(__VA_ARGS__), __VA_ARGS__)
1478 /* Match N words representing all of the line, case-sensitively. */
1479 #define MatchesCS(...) \
1480 MatchesImpl(true, previous_words_count, previous_words, \
1481 VA_ARGS_NARGS(__VA_ARGS__), __VA_ARGS__)
1483 /* Match the first N words on the line, case-insensitively. */
1484 #define HeadMatches(...) \
1485 HeadMatchesImpl(false, previous_words_count, previous_words, \
1486 VA_ARGS_NARGS(__VA_ARGS__), __VA_ARGS__)
1488 /* Match the first N words on the line, case-sensitively. */
1489 #define HeadMatchesCS(...) \
1490 HeadMatchesImpl(true, previous_words_count, previous_words, \
1491 VA_ARGS_NARGS(__VA_ARGS__), __VA_ARGS__)
1493 /* Known command-starting keywords. */
1494 static const char *const sql_commands
[] = {
1495 "ABORT", "ALTER", "ANALYZE", "BEGIN", "CALL", "CHECKPOINT", "CLOSE", "CLUSTER",
1496 "COMMENT", "COMMIT", "COPY", "CREATE", "DEALLOCATE", "DECLARE",
1497 "DELETE FROM", "DISCARD", "DO", "DROP", "END", "EXECUTE", "EXPLAIN",
1498 "FETCH", "GRANT", "IMPORT FOREIGN SCHEMA", "INSERT INTO", "LISTEN", "LOAD", "LOCK",
1499 "MOVE", "NOTIFY", "PREPARE",
1500 "REASSIGN", "REFRESH MATERIALIZED VIEW", "REINDEX", "RELEASE",
1501 "RESET", "REVOKE", "ROLLBACK",
1502 "SAVEPOINT", "SECURITY LABEL", "SELECT", "SET", "SHOW", "START",
1503 "TABLE", "TRUNCATE", "UNLISTEN", "UPDATE", "VACUUM", "VALUES", "WITH",
1507 /* psql's backslash commands. */
1508 static const char *const backslash_commands
[] = {
1510 "\\connect", "\\conninfo", "\\C", "\\cd", "\\copy",
1511 "\\copyright", "\\crosstabview",
1512 "\\d", "\\da", "\\dA", "\\dAc", "\\dAf", "\\dAo", "\\dAp",
1513 "\\db", "\\dc", "\\dC", "\\dd", "\\ddp", "\\dD",
1514 "\\des", "\\det", "\\deu", "\\dew", "\\dE", "\\df",
1515 "\\dF", "\\dFd", "\\dFp", "\\dFt", "\\dg", "\\di", "\\dl", "\\dL",
1516 "\\dm", "\\dn", "\\do", "\\dO", "\\dp", "\\dP", "\\dPi", "\\dPt",
1517 "\\drds", "\\dRs", "\\dRp", "\\ds", "\\dS",
1518 "\\dt", "\\dT", "\\dv", "\\du", "\\dx", "\\dX", "\\dy",
1519 "\\e", "\\echo", "\\ef", "\\elif", "\\else", "\\encoding",
1520 "\\endif", "\\errverbose", "\\ev",
1522 "\\g", "\\gdesc", "\\gexec", "\\gset", "\\gx",
1523 "\\h", "\\help", "\\H",
1524 "\\i", "\\if", "\\ir",
1525 "\\l", "\\lo_import", "\\lo_export", "\\lo_list", "\\lo_unlink",
1527 "\\p", "\\password", "\\prompt", "\\pset",
1530 "\\s", "\\set", "\\setenv", "\\sf", "\\sv",
1531 "\\t", "\\T", "\\timing",
1534 "\\w", "\\warn", "\\watch",
1541 * Temporary workaround for a bug in recent (2019) libedit: it incorrectly
1542 * de-escapes the input "text", causing us to fail to recognize backslash
1543 * commands. So get the string to look at from rl_line_buffer instead.
1545 char *text_copy
= pnstrdup(rl_line_buffer
+ start
, end
- start
);
1548 /* Remember last char of the given input word. */
1549 completion_last_char
= (end
> start
) ? text
[end
- start
- 1] : '\0';
1551 /* We usually want the append character to be a space. */
1552 #ifdef HAVE_RL_COMPLETION_APPEND_CHARACTER
1553 rl_completion_append_character
= ' ';
1556 /* Clear a few things. */
1557 completion_charp
= NULL
;
1558 completion_charpp
= NULL
;
1559 completion_info_charp
= NULL
;
1560 completion_info_charp2
= NULL
;
1563 * Scan the input line to extract the words before our current position.
1564 * According to those we'll make some smart decisions on what the user is
1565 * probably intending to type.
1567 previous_words
= get_previous_words(start
,
1569 &previous_words_count
);
1571 /* If current word is a backslash command, offer completions for that */
1572 if (text
[0] == '\\')
1573 COMPLETE_WITH_LIST_CS(backslash_commands
);
1575 /* If current word is a variable interpolation, handle that case */
1576 else if (text
[0] == ':' && text
[1] != ':')
1578 if (text
[1] == '\'')
1579 matches
= complete_from_variables(text
, ":'", "'", true);
1580 else if (text
[1] == '"')
1581 matches
= complete_from_variables(text
, ":\"", "\"", true);
1583 matches
= complete_from_variables(text
, ":", "", true);
1586 /* If no previous word, suggest one of the basic sql commands */
1587 else if (previous_words_count
== 0)
1588 COMPLETE_WITH_LIST(sql_commands
);
1591 /* complete with something you can create */
1592 else if (TailMatches("CREATE"))
1593 matches
= rl_completion_matches(text
, create_command_generator
);
1595 /* complete with something you can create or replace */
1596 else if (TailMatches("CREATE", "OR", "REPLACE"))
1597 COMPLETE_WITH("FUNCTION", "PROCEDURE", "LANGUAGE", "RULE", "VIEW",
1598 "AGGREGATE", "TRANSFORM", "TRIGGER");
1600 /* DROP, but not DROP embedded in other commands */
1601 /* complete with something you can drop */
1602 else if (Matches("DROP"))
1603 matches
= rl_completion_matches(text
, drop_command_generator
);
1608 else if (Matches("ALTER", "TABLE"))
1609 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
,
1610 "UNION SELECT 'ALL IN TABLESPACE'");
1612 /* ALTER something */
1613 else if (Matches("ALTER"))
1614 matches
= rl_completion_matches(text
, alter_command_generator
);
1615 /* ALTER TABLE,INDEX,MATERIALIZED VIEW ALL IN TABLESPACE xxx */
1616 else if (TailMatches("ALL", "IN", "TABLESPACE", MatchAny
))
1617 COMPLETE_WITH("SET TABLESPACE", "OWNED BY");
1618 /* ALTER TABLE,INDEX,MATERIALIZED VIEW ALL IN TABLESPACE xxx OWNED BY */
1619 else if (TailMatches("ALL", "IN", "TABLESPACE", MatchAny
, "OWNED", "BY"))
1620 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
1621 /* ALTER TABLE,INDEX,MATERIALIZED VIEW ALL IN TABLESPACE xxx OWNED BY xxx */
1622 else if (TailMatches("ALL", "IN", "TABLESPACE", MatchAny
, "OWNED", "BY", MatchAny
))
1623 COMPLETE_WITH("SET TABLESPACE");
1624 /* ALTER AGGREGATE,FUNCTION,PROCEDURE,ROUTINE <name> */
1625 else if (Matches("ALTER", "AGGREGATE|FUNCTION|PROCEDURE|ROUTINE", MatchAny
))
1627 /* ALTER AGGREGATE <name> (...) */
1628 else if (Matches("ALTER", "AGGREGATE", MatchAny
, MatchAny
))
1630 if (ends_with(prev_wd
, ')'))
1631 COMPLETE_WITH("OWNER TO", "RENAME TO", "SET SCHEMA");
1633 COMPLETE_WITH_FUNCTION_ARG(prev2_wd
);
1635 /* ALTER FUNCTION,PROCEDURE,ROUTINE <name> (...) */
1636 else if (Matches("ALTER", "FUNCTION|PROCEDURE|ROUTINE", MatchAny
, MatchAny
))
1638 if (ends_with(prev_wd
, ')'))
1639 COMPLETE_WITH("OWNER TO", "RENAME TO", "SET SCHEMA",
1640 "DEPENDS ON EXTENSION", "NO DEPENDS ON EXTENSION");
1642 COMPLETE_WITH_FUNCTION_ARG(prev2_wd
);
1645 /* ALTER PUBLICATION <name> */
1646 else if (Matches("ALTER", "PUBLICATION", MatchAny
))
1647 COMPLETE_WITH("ADD TABLE", "DROP TABLE", "OWNER TO", "RENAME TO", "SET");
1648 /* ALTER PUBLICATION <name> SET */
1649 else if (Matches("ALTER", "PUBLICATION", MatchAny
, "SET"))
1650 COMPLETE_WITH("(", "TABLE");
1651 /* ALTER PUBLICATION <name> SET ( */
1652 else if (HeadMatches("ALTER", "PUBLICATION", MatchAny
) && TailMatches("SET", "("))
1653 COMPLETE_WITH("publish", "publish_via_partition_root");
1654 /* ALTER SUBSCRIPTION <name> */
1655 else if (Matches("ALTER", "SUBSCRIPTION", MatchAny
))
1656 COMPLETE_WITH("CONNECTION", "ENABLE", "DISABLE", "OWNER TO",
1657 "RENAME TO", "REFRESH PUBLICATION", "SET",
1658 "ADD PUBLICATION", "DROP PUBLICATION");
1659 /* ALTER SUBSCRIPTION <name> REFRESH PUBLICATION */
1660 else if (HeadMatches("ALTER", "SUBSCRIPTION", MatchAny
) &&
1661 TailMatches("REFRESH", "PUBLICATION"))
1662 COMPLETE_WITH("WITH (");
1663 /* ALTER SUBSCRIPTION <name> REFRESH PUBLICATION WITH ( */
1664 else if (HeadMatches("ALTER", "SUBSCRIPTION", MatchAny
) &&
1665 TailMatches("REFRESH", "PUBLICATION", "WITH", "("))
1666 COMPLETE_WITH("copy_data");
1667 /* ALTER SUBSCRIPTION <name> SET */
1668 else if (Matches("ALTER", "SUBSCRIPTION", MatchAny
, "SET"))
1669 COMPLETE_WITH("(", "PUBLICATION");
1670 /* ALTER SUBSCRIPTION <name> SET ( */
1671 else if (HeadMatches("ALTER", "SUBSCRIPTION", MatchAny
) && TailMatches("SET", "("))
1672 COMPLETE_WITH("binary", "slot_name", "streaming", "synchronous_commit");
1673 /* ALTER SUBSCRIPTION <name> SET PUBLICATION */
1674 else if (HeadMatches("ALTER", "SUBSCRIPTION", MatchAny
) && TailMatches("SET", "PUBLICATION"))
1676 /* complete with nothing here as this refers to remote publications */
1678 /* ALTER SUBSCRIPTION <name> ADD|DROP|SET PUBLICATION <name> */
1679 else if (HeadMatches("ALTER", "SUBSCRIPTION", MatchAny
) &&
1680 TailMatches("ADD|DROP|SET", "PUBLICATION", MatchAny
))
1681 COMPLETE_WITH("WITH (");
1682 /* ALTER SUBSCRIPTION <name> ADD|DROP|SET PUBLICATION <name> WITH ( */
1683 else if (HeadMatches("ALTER", "SUBSCRIPTION", MatchAny
) &&
1684 TailMatches("ADD|DROP|SET", "PUBLICATION", MatchAny
, "WITH", "("))
1685 COMPLETE_WITH("copy_data", "refresh");
1687 /* ALTER SCHEMA <name> */
1688 else if (Matches("ALTER", "SCHEMA", MatchAny
))
1689 COMPLETE_WITH("OWNER TO", "RENAME TO");
1691 /* ALTER COLLATION <name> */
1692 else if (Matches("ALTER", "COLLATION", MatchAny
))
1693 COMPLETE_WITH("OWNER TO", "RENAME TO", "SET SCHEMA");
1695 /* ALTER CONVERSION <name> */
1696 else if (Matches("ALTER", "CONVERSION", MatchAny
))
1697 COMPLETE_WITH("OWNER TO", "RENAME TO", "SET SCHEMA");
1699 /* ALTER DATABASE <name> */
1700 else if (Matches("ALTER", "DATABASE", MatchAny
))
1701 COMPLETE_WITH("RESET", "SET", "OWNER TO", "RENAME TO",
1702 "IS_TEMPLATE", "ALLOW_CONNECTIONS",
1703 "CONNECTION LIMIT");
1705 /* ALTER DATABASE <name> SET TABLESPACE */
1706 else if (Matches("ALTER", "DATABASE", MatchAny
, "SET", "TABLESPACE"))
1707 COMPLETE_WITH_QUERY(Query_for_list_of_tablespaces
);
1709 /* ALTER EVENT TRIGGER */
1710 else if (Matches("ALTER", "EVENT", "TRIGGER"))
1711 COMPLETE_WITH_QUERY(Query_for_list_of_event_triggers
);
1713 /* ALTER EVENT TRIGGER <name> */
1714 else if (Matches("ALTER", "EVENT", "TRIGGER", MatchAny
))
1715 COMPLETE_WITH("DISABLE", "ENABLE", "OWNER TO", "RENAME TO");
1717 /* ALTER EVENT TRIGGER <name> ENABLE */
1718 else if (Matches("ALTER", "EVENT", "TRIGGER", MatchAny
, "ENABLE"))
1719 COMPLETE_WITH("REPLICA", "ALWAYS");
1721 /* ALTER EXTENSION <name> */
1722 else if (Matches("ALTER", "EXTENSION", MatchAny
))
1723 COMPLETE_WITH("ADD", "DROP", "UPDATE", "SET SCHEMA");
1725 /* ALTER EXTENSION <name> UPDATE */
1726 else if (Matches("ALTER", "EXTENSION", MatchAny
, "UPDATE"))
1728 completion_info_charp
= prev2_wd
;
1729 COMPLETE_WITH_QUERY(Query_for_list_of_available_extension_versions_with_TO
);
1732 /* ALTER EXTENSION <name> UPDATE TO */
1733 else if (Matches("ALTER", "EXTENSION", MatchAny
, "UPDATE", "TO"))
1735 completion_info_charp
= prev3_wd
;
1736 COMPLETE_WITH_QUERY(Query_for_list_of_available_extension_versions
);
1740 else if (Matches("ALTER", "FOREIGN"))
1741 COMPLETE_WITH("DATA WRAPPER", "TABLE");
1743 /* ALTER FOREIGN DATA WRAPPER <name> */
1744 else if (Matches("ALTER", "FOREIGN", "DATA", "WRAPPER", MatchAny
))
1745 COMPLETE_WITH("HANDLER", "VALIDATOR", "OPTIONS", "OWNER TO", "RENAME TO");
1747 /* ALTER FOREIGN TABLE <name> */
1748 else if (Matches("ALTER", "FOREIGN", "TABLE", MatchAny
))
1749 COMPLETE_WITH("ADD", "ALTER", "DISABLE TRIGGER", "DROP", "ENABLE",
1750 "INHERIT", "NO INHERIT", "OPTIONS", "OWNER TO",
1751 "RENAME", "SET", "VALIDATE CONSTRAINT");
1754 else if (Matches("ALTER", "INDEX"))
1755 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
,
1756 "UNION SELECT 'ALL IN TABLESPACE'");
1757 /* ALTER INDEX <name> */
1758 else if (Matches("ALTER", "INDEX", MatchAny
))
1759 COMPLETE_WITH("ALTER COLUMN", "OWNER TO", "RENAME TO", "SET",
1760 "RESET", "ATTACH PARTITION",
1761 "DEPENDS ON EXTENSION", "NO DEPENDS ON EXTENSION");
1762 else if (Matches("ALTER", "INDEX", MatchAny
, "ATTACH"))
1763 COMPLETE_WITH("PARTITION");
1764 else if (Matches("ALTER", "INDEX", MatchAny
, "ATTACH", "PARTITION"))
1765 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
, NULL
);
1766 /* ALTER INDEX <name> ALTER */
1767 else if (Matches("ALTER", "INDEX", MatchAny
, "ALTER"))
1768 COMPLETE_WITH("COLUMN");
1769 /* ALTER INDEX <name> ALTER COLUMN */
1770 else if (Matches("ALTER", "INDEX", MatchAny
, "ALTER", "COLUMN"))
1772 completion_info_charp
= prev3_wd
;
1773 COMPLETE_WITH_QUERY(Query_for_list_of_attribute_numbers
);
1775 /* ALTER INDEX <name> ALTER COLUMN <colnum> */
1776 else if (Matches("ALTER", "INDEX", MatchAny
, "ALTER", "COLUMN", MatchAny
))
1777 COMPLETE_WITH("SET STATISTICS");
1778 /* ALTER INDEX <name> ALTER COLUMN <colnum> SET */
1779 else if (Matches("ALTER", "INDEX", MatchAny
, "ALTER", "COLUMN", MatchAny
, "SET"))
1780 COMPLETE_WITH("STATISTICS");
1781 /* ALTER INDEX <name> ALTER COLUMN <colnum> SET STATISTICS */
1782 else if (Matches("ALTER", "INDEX", MatchAny
, "ALTER", "COLUMN", MatchAny
, "SET", "STATISTICS"))
1784 /* Enforce no completion here, as an integer has to be specified */
1786 /* ALTER INDEX <name> SET */
1787 else if (Matches("ALTER", "INDEX", MatchAny
, "SET"))
1788 COMPLETE_WITH("(", "TABLESPACE");
1789 /* ALTER INDEX <name> RESET */
1790 else if (Matches("ALTER", "INDEX", MatchAny
, "RESET"))
1792 /* ALTER INDEX <foo> SET|RESET ( */
1793 else if (Matches("ALTER", "INDEX", MatchAny
, "RESET", "("))
1794 COMPLETE_WITH("fillfactor",
1795 "deduplicate_items", /* BTREE */
1796 "fastupdate", "gin_pending_list_limit", /* GIN */
1797 "buffering", /* GiST */
1798 "pages_per_range", "autosummarize" /* BRIN */
1800 else if (Matches("ALTER", "INDEX", MatchAny
, "SET", "("))
1801 COMPLETE_WITH("fillfactor =",
1802 "deduplicate_items =", /* BTREE */
1803 "fastupdate =", "gin_pending_list_limit =", /* GIN */
1804 "buffering =", /* GiST */
1805 "pages_per_range =", "autosummarize =" /* BRIN */
1807 else if (Matches("ALTER", "INDEX", MatchAny
, "NO", "DEPENDS"))
1808 COMPLETE_WITH("ON EXTENSION");
1809 else if (Matches("ALTER", "INDEX", MatchAny
, "DEPENDS"))
1810 COMPLETE_WITH("ON EXTENSION");
1812 /* ALTER LANGUAGE <name> */
1813 else if (Matches("ALTER", "LANGUAGE", MatchAny
))
1814 COMPLETE_WITH("OWNER TO", "RENAME TO");
1816 /* ALTER LARGE OBJECT <oid> */
1817 else if (Matches("ALTER", "LARGE", "OBJECT", MatchAny
))
1818 COMPLETE_WITH("OWNER TO");
1820 /* ALTER MATERIALIZED VIEW */
1821 else if (Matches("ALTER", "MATERIALIZED", "VIEW"))
1822 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_matviews
,
1823 "UNION SELECT 'ALL IN TABLESPACE'");
1825 /* ALTER USER,ROLE <name> */
1826 else if (Matches("ALTER", "USER|ROLE", MatchAny
) &&
1827 !TailMatches("USER", "MAPPING"))
1828 COMPLETE_WITH("BYPASSRLS", "CONNECTION LIMIT", "CREATEDB", "CREATEROLE",
1829 "ENCRYPTED PASSWORD", "INHERIT", "LOGIN", "NOBYPASSRLS",
1830 "NOCREATEDB", "NOCREATEROLE", "NOINHERIT",
1831 "NOLOGIN", "NOREPLICATION", "NOSUPERUSER", "PASSWORD",
1832 "RENAME TO", "REPLICATION", "RESET", "SET", "SUPERUSER",
1833 "VALID UNTIL", "WITH");
1835 /* ALTER USER,ROLE <name> WITH */
1836 else if (Matches("ALTER", "USER|ROLE", MatchAny
, "WITH"))
1837 /* Similar to the above, but don't complete "WITH" again. */
1838 COMPLETE_WITH("BYPASSRLS", "CONNECTION LIMIT", "CREATEDB", "CREATEROLE",
1839 "ENCRYPTED PASSWORD", "INHERIT", "LOGIN", "NOBYPASSRLS",
1840 "NOCREATEDB", "NOCREATEROLE", "NOINHERIT",
1841 "NOLOGIN", "NOREPLICATION", "NOSUPERUSER", "PASSWORD",
1842 "RENAME TO", "REPLICATION", "RESET", "SET", "SUPERUSER",
1845 /* ALTER DEFAULT PRIVILEGES */
1846 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES"))
1847 COMPLETE_WITH("FOR ROLE", "IN SCHEMA");
1848 /* ALTER DEFAULT PRIVILEGES FOR */
1849 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES", "FOR"))
1850 COMPLETE_WITH("ROLE");
1851 /* ALTER DEFAULT PRIVILEGES IN */
1852 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES", "IN"))
1853 COMPLETE_WITH("SCHEMA");
1854 /* ALTER DEFAULT PRIVILEGES FOR ROLE|USER ... */
1855 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES", "FOR", "ROLE|USER",
1857 COMPLETE_WITH("GRANT", "REVOKE", "IN SCHEMA");
1858 /* ALTER DEFAULT PRIVILEGES IN SCHEMA ... */
1859 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES", "IN", "SCHEMA",
1861 COMPLETE_WITH("GRANT", "REVOKE", "FOR ROLE");
1862 /* ALTER DEFAULT PRIVILEGES IN SCHEMA ... FOR */
1863 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES", "IN", "SCHEMA",
1865 COMPLETE_WITH("ROLE");
1866 /* ALTER DEFAULT PRIVILEGES FOR ROLE|USER ... IN SCHEMA ... */
1867 /* ALTER DEFAULT PRIVILEGES IN SCHEMA ... FOR ROLE|USER ... */
1868 else if (Matches("ALTER", "DEFAULT", "PRIVILEGES", "FOR", "ROLE|USER",
1869 MatchAny
, "IN", "SCHEMA", MatchAny
) ||
1870 Matches("ALTER", "DEFAULT", "PRIVILEGES", "IN", "SCHEMA",
1871 MatchAny
, "FOR", "ROLE|USER", MatchAny
))
1872 COMPLETE_WITH("GRANT", "REVOKE");
1873 /* ALTER DOMAIN <name> */
1874 else if (Matches("ALTER", "DOMAIN", MatchAny
))
1875 COMPLETE_WITH("ADD", "DROP", "OWNER TO", "RENAME", "SET",
1876 "VALIDATE CONSTRAINT");
1877 /* ALTER DOMAIN <sth> DROP */
1878 else if (Matches("ALTER", "DOMAIN", MatchAny
, "DROP"))
1879 COMPLETE_WITH("CONSTRAINT", "DEFAULT", "NOT NULL");
1880 /* ALTER DOMAIN <sth> DROP|RENAME|VALIDATE CONSTRAINT */
1881 else if (Matches("ALTER", "DOMAIN", MatchAny
, "DROP|RENAME|VALIDATE", "CONSTRAINT"))
1883 completion_info_charp
= prev3_wd
;
1884 COMPLETE_WITH_QUERY(Query_for_constraint_of_type
);
1886 /* ALTER DOMAIN <sth> RENAME */
1887 else if (Matches("ALTER", "DOMAIN", MatchAny
, "RENAME"))
1888 COMPLETE_WITH("CONSTRAINT", "TO");
1889 /* ALTER DOMAIN <sth> RENAME CONSTRAINT <sth> */
1890 else if (Matches("ALTER", "DOMAIN", MatchAny
, "RENAME", "CONSTRAINT", MatchAny
))
1891 COMPLETE_WITH("TO");
1893 /* ALTER DOMAIN <sth> SET */
1894 else if (Matches("ALTER", "DOMAIN", MatchAny
, "SET"))
1895 COMPLETE_WITH("DEFAULT", "NOT NULL", "SCHEMA");
1896 /* ALTER SEQUENCE <name> */
1897 else if (Matches("ALTER", "SEQUENCE", MatchAny
))
1898 COMPLETE_WITH("INCREMENT", "MINVALUE", "MAXVALUE", "RESTART", "NO",
1899 "CACHE", "CYCLE", "SET SCHEMA", "OWNED BY", "OWNER TO",
1901 /* ALTER SEQUENCE <name> NO */
1902 else if (Matches("ALTER", "SEQUENCE", MatchAny
, "NO"))
1903 COMPLETE_WITH("MINVALUE", "MAXVALUE", "CYCLE");
1904 /* ALTER SERVER <name> */
1905 else if (Matches("ALTER", "SERVER", MatchAny
))
1906 COMPLETE_WITH("VERSION", "OPTIONS", "OWNER TO", "RENAME TO");
1907 /* ALTER SERVER <name> VERSION <version> */
1908 else if (Matches("ALTER", "SERVER", MatchAny
, "VERSION", MatchAny
))
1909 COMPLETE_WITH("OPTIONS");
1910 /* ALTER SYSTEM SET, RESET, RESET ALL */
1911 else if (Matches("ALTER", "SYSTEM"))
1912 COMPLETE_WITH("SET", "RESET");
1913 else if (Matches("ALTER", "SYSTEM", "SET|RESET"))
1914 COMPLETE_WITH_QUERY(Query_for_list_of_alter_system_set_vars
);
1915 else if (Matches("ALTER", "SYSTEM", "SET", MatchAny
))
1916 COMPLETE_WITH("TO");
1917 /* ALTER VIEW <name> */
1918 else if (Matches("ALTER", "VIEW", MatchAny
))
1919 COMPLETE_WITH("ALTER COLUMN", "OWNER TO", "RENAME",
1921 /* ALTER VIEW xxx RENAME */
1922 else if (Matches("ALTER", "VIEW", MatchAny
, "RENAME"))
1923 COMPLETE_WITH_ATTR(prev2_wd
, " UNION SELECT 'COLUMN' UNION SELECT 'TO'");
1924 else if (Matches("ALTER", "VIEW", MatchAny
, "ALTER|RENAME", "COLUMN"))
1925 COMPLETE_WITH_ATTR(prev3_wd
, "");
1926 /* ALTER VIEW xxx RENAME yyy */
1927 else if (Matches("ALTER", "VIEW", MatchAny
, "RENAME", MatchAnyExcept("TO")))
1928 COMPLETE_WITH("TO");
1929 /* ALTER VIEW xxx RENAME COLUMN yyy */
1930 else if (Matches("ALTER", "VIEW", MatchAny
, "RENAME", "COLUMN", MatchAnyExcept("TO")))
1931 COMPLETE_WITH("TO");
1933 /* ALTER MATERIALIZED VIEW <name> */
1934 else if (Matches("ALTER", "MATERIALIZED", "VIEW", MatchAny
))
1935 COMPLETE_WITH("ALTER COLUMN", "CLUSTER ON", "DEPENDS ON EXTENSION",
1936 "NO DEPENDS ON EXTENSION", "OWNER TO", "RENAME",
1938 /* ALTER MATERIALIZED VIEW xxx RENAME */
1939 else if (Matches("ALTER", "MATERIALIZED", "VIEW", MatchAny
, "RENAME"))
1940 COMPLETE_WITH_ATTR(prev2_wd
, " UNION SELECT 'COLUMN' UNION SELECT 'TO'");
1941 else if (Matches("ALTER", "MATERIALIZED", "VIEW", MatchAny
, "ALTER|RENAME", "COLUMN"))
1942 COMPLETE_WITH_ATTR(prev3_wd
, "");
1943 /* ALTER MATERIALIZED VIEW xxx RENAME yyy */
1944 else if (Matches("ALTER", "MATERIALIZED", "VIEW", MatchAny
, "RENAME", MatchAnyExcept("TO")))
1945 COMPLETE_WITH("TO");
1946 /* ALTER MATERIALIZED VIEW xxx RENAME COLUMN yyy */
1947 else if (Matches("ALTER", "MATERIALIZED", "VIEW", MatchAny
, "RENAME", "COLUMN", MatchAnyExcept("TO")))
1948 COMPLETE_WITH("TO");
1949 /* ALTER MATERIALIZED VIEW xxx SET */
1950 else if (Matches("ALTER", "MATERIALIZED", "VIEW", MatchAny
, "SET"))
1951 COMPLETE_WITH("(", "SCHEMA", "TABLESPACE", "WITHOUT CLUSTER");
1952 /* ALTER POLICY <name> */
1953 else if (Matches("ALTER", "POLICY"))
1954 COMPLETE_WITH_QUERY(Query_for_list_of_policies
);
1955 /* ALTER POLICY <name> ON */
1956 else if (Matches("ALTER", "POLICY", MatchAny
))
1957 COMPLETE_WITH("ON");
1958 /* ALTER POLICY <name> ON <table> */
1959 else if (Matches("ALTER", "POLICY", MatchAny
, "ON"))
1961 completion_info_charp
= prev2_wd
;
1962 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_policy
);
1964 /* ALTER POLICY <name> ON <table> - show options */
1965 else if (Matches("ALTER", "POLICY", MatchAny
, "ON", MatchAny
))
1966 COMPLETE_WITH("RENAME TO", "TO", "USING (", "WITH CHECK (");
1967 /* ALTER POLICY <name> ON <table> TO <role> */
1968 else if (Matches("ALTER", "POLICY", MatchAny
, "ON", MatchAny
, "TO"))
1969 COMPLETE_WITH_QUERY(Query_for_list_of_grant_roles
);
1970 /* ALTER POLICY <name> ON <table> USING ( */
1971 else if (Matches("ALTER", "POLICY", MatchAny
, "ON", MatchAny
, "USING"))
1973 /* ALTER POLICY <name> ON <table> WITH CHECK ( */
1974 else if (Matches("ALTER", "POLICY", MatchAny
, "ON", MatchAny
, "WITH", "CHECK"))
1977 /* ALTER RULE <name>, add ON */
1978 else if (Matches("ALTER", "RULE", MatchAny
))
1979 COMPLETE_WITH("ON");
1981 /* If we have ALTER RULE <name> ON, then add the correct tablename */
1982 else if (Matches("ALTER", "RULE", MatchAny
, "ON"))
1984 completion_info_charp
= prev2_wd
;
1985 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_rule
);
1988 /* ALTER RULE <name> ON <name> */
1989 else if (Matches("ALTER", "RULE", MatchAny
, "ON", MatchAny
))
1990 COMPLETE_WITH("RENAME TO");
1992 /* ALTER STATISTICS <name> */
1993 else if (Matches("ALTER", "STATISTICS", MatchAny
))
1994 COMPLETE_WITH("OWNER TO", "RENAME TO", "SET SCHEMA", "SET STATISTICS");
1996 /* ALTER TRIGGER <name>, add ON */
1997 else if (Matches("ALTER", "TRIGGER", MatchAny
))
1998 COMPLETE_WITH("ON");
2000 else if (Matches("ALTER", "TRIGGER", MatchAny
, MatchAny
))
2002 completion_info_charp
= prev2_wd
;
2003 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_trigger
);
2007 * If we have ALTER TRIGGER <sth> ON, then add the correct tablename
2009 else if (Matches("ALTER", "TRIGGER", MatchAny
, "ON"))
2010 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
2012 /* ALTER TRIGGER <name> ON <name> */
2013 else if (Matches("ALTER", "TRIGGER", MatchAny
, "ON", MatchAny
))
2014 COMPLETE_WITH("RENAME TO", "DEPENDS ON EXTENSION",
2015 "NO DEPENDS ON EXTENSION");
2018 * If we detect ALTER TABLE <name>, suggest sub commands
2020 else if (Matches("ALTER", "TABLE", MatchAny
))
2021 COMPLETE_WITH("ADD", "ALTER", "CLUSTER ON", "DISABLE", "DROP",
2022 "ENABLE", "INHERIT", "NO", "RENAME", "RESET",
2023 "OWNER TO", "SET", "VALIDATE CONSTRAINT",
2024 "REPLICA IDENTITY", "ATTACH PARTITION",
2025 "DETACH PARTITION", "FORCE ROW LEVEL SECURITY");
2026 /* ALTER TABLE xxx ADD */
2027 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD"))
2029 /* make sure to keep this list and the !Matches() below in sync */
2030 COMPLETE_WITH("COLUMN", "CONSTRAINT", "CHECK", "UNIQUE", "PRIMARY KEY",
2031 "EXCLUDE", "FOREIGN KEY");
2033 /* ATER TABLE xxx ADD [COLUMN] yyy */
2034 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "COLUMN", MatchAny
) ||
2035 (Matches("ALTER", "TABLE", MatchAny
, "ADD", MatchAny
) &&
2036 !Matches("ALTER", "TABLE", MatchAny
, "ADD", "COLUMN|CONSTRAINT|CHECK|UNIQUE|PRIMARY|EXCLUDE|FOREIGN")))
2037 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
2038 /* ALTER TABLE xxx ADD CONSTRAINT yyy */
2039 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "CONSTRAINT", MatchAny
))
2040 COMPLETE_WITH("CHECK", "UNIQUE", "PRIMARY KEY", "EXCLUDE", "FOREIGN KEY");
2041 /* ALTER TABLE xxx ADD [CONSTRAINT yyy] (PRIMARY KEY|UNIQUE) */
2042 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "PRIMARY", "KEY") ||
2043 Matches("ALTER", "TABLE", MatchAny
, "ADD", "UNIQUE") ||
2044 Matches("ALTER", "TABLE", MatchAny
, "ADD", "CONSTRAINT", MatchAny
, "PRIMARY", "KEY") ||
2045 Matches("ALTER", "TABLE", MatchAny
, "ADD", "CONSTRAINT", MatchAny
, "UNIQUE"))
2046 COMPLETE_WITH("(", "USING INDEX");
2047 /* ALTER TABLE xxx ADD PRIMARY KEY USING INDEX */
2048 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "PRIMARY", "KEY", "USING", "INDEX"))
2050 completion_info_charp
= prev6_wd
;
2051 COMPLETE_WITH_QUERY(Query_for_unique_index_of_table
);
2053 /* ALTER TABLE xxx ADD UNIQUE USING INDEX */
2054 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "UNIQUE", "USING", "INDEX"))
2056 completion_info_charp
= prev5_wd
;
2057 COMPLETE_WITH_QUERY(Query_for_unique_index_of_table
);
2059 /* ALTER TABLE xxx ADD CONSTRAINT yyy PRIMARY KEY USING INDEX */
2060 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "CONSTRAINT", MatchAny
,
2061 "PRIMARY", "KEY", "USING", "INDEX"))
2063 completion_info_charp
= prev8_wd
;
2064 COMPLETE_WITH_QUERY(Query_for_unique_index_of_table
);
2066 /* ALTER TABLE xxx ADD CONSTRAINT yyy UNIQUE USING INDEX */
2067 else if (Matches("ALTER", "TABLE", MatchAny
, "ADD", "CONSTRAINT", MatchAny
,
2068 "UNIQUE", "USING", "INDEX"))
2070 completion_info_charp
= prev7_wd
;
2071 COMPLETE_WITH_QUERY(Query_for_unique_index_of_table
);
2073 /* ALTER TABLE xxx ENABLE */
2074 else if (Matches("ALTER", "TABLE", MatchAny
, "ENABLE"))
2075 COMPLETE_WITH("ALWAYS", "REPLICA", "ROW LEVEL SECURITY", "RULE",
2077 else if (Matches("ALTER", "TABLE", MatchAny
, "ENABLE", "REPLICA|ALWAYS"))
2078 COMPLETE_WITH("RULE", "TRIGGER");
2079 else if (Matches("ALTER", "TABLE", MatchAny
, "ENABLE", "RULE"))
2081 completion_info_charp
= prev3_wd
;
2082 COMPLETE_WITH_QUERY(Query_for_rule_of_table
);
2084 else if (Matches("ALTER", "TABLE", MatchAny
, "ENABLE", MatchAny
, "RULE"))
2086 completion_info_charp
= prev4_wd
;
2087 COMPLETE_WITH_QUERY(Query_for_rule_of_table
);
2089 else if (Matches("ALTER", "TABLE", MatchAny
, "ENABLE", "TRIGGER"))
2091 completion_info_charp
= prev3_wd
;
2092 COMPLETE_WITH_QUERY(Query_for_trigger_of_table
);
2094 else if (Matches("ALTER", "TABLE", MatchAny
, "ENABLE", MatchAny
, "TRIGGER"))
2096 completion_info_charp
= prev4_wd
;
2097 COMPLETE_WITH_QUERY(Query_for_trigger_of_table
);
2099 /* ALTER TABLE xxx INHERIT */
2100 else if (Matches("ALTER", "TABLE", MatchAny
, "INHERIT"))
2101 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, "");
2102 /* ALTER TABLE xxx NO */
2103 else if (Matches("ALTER", "TABLE", MatchAny
, "NO"))
2104 COMPLETE_WITH("FORCE ROW LEVEL SECURITY", "INHERIT");
2105 /* ALTER TABLE xxx NO INHERIT */
2106 else if (Matches("ALTER", "TABLE", MatchAny
, "NO", "INHERIT"))
2107 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, "");
2108 /* ALTER TABLE xxx DISABLE */
2109 else if (Matches("ALTER", "TABLE", MatchAny
, "DISABLE"))
2110 COMPLETE_WITH("ROW LEVEL SECURITY", "RULE", "TRIGGER");
2111 else if (Matches("ALTER", "TABLE", MatchAny
, "DISABLE", "RULE"))
2113 completion_info_charp
= prev3_wd
;
2114 COMPLETE_WITH_QUERY(Query_for_rule_of_table
);
2116 else if (Matches("ALTER", "TABLE", MatchAny
, "DISABLE", "TRIGGER"))
2118 completion_info_charp
= prev3_wd
;
2119 COMPLETE_WITH_QUERY(Query_for_trigger_of_table
);
2122 /* ALTER TABLE xxx ALTER */
2123 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER"))
2124 COMPLETE_WITH_ATTR(prev2_wd
, " UNION SELECT 'COLUMN' UNION SELECT 'CONSTRAINT'");
2126 /* ALTER TABLE xxx RENAME */
2127 else if (Matches("ALTER", "TABLE", MatchAny
, "RENAME"))
2128 COMPLETE_WITH_ATTR(prev2_wd
, " UNION SELECT 'COLUMN' UNION SELECT 'CONSTRAINT' UNION SELECT 'TO'");
2129 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER|RENAME", "COLUMN"))
2130 COMPLETE_WITH_ATTR(prev3_wd
, "");
2132 /* ALTER TABLE xxx RENAME yyy */
2133 else if (Matches("ALTER", "TABLE", MatchAny
, "RENAME", MatchAnyExcept("CONSTRAINT|TO")))
2134 COMPLETE_WITH("TO");
2136 /* ALTER TABLE xxx RENAME COLUMN/CONSTRAINT yyy */
2137 else if (Matches("ALTER", "TABLE", MatchAny
, "RENAME", "COLUMN|CONSTRAINT", MatchAnyExcept("TO")))
2138 COMPLETE_WITH("TO");
2140 /* If we have ALTER TABLE <sth> DROP, provide COLUMN or CONSTRAINT */
2141 else if (Matches("ALTER", "TABLE", MatchAny
, "DROP"))
2142 COMPLETE_WITH("COLUMN", "CONSTRAINT");
2143 /* If we have ALTER TABLE <sth> DROP COLUMN, provide list of columns */
2144 else if (Matches("ALTER", "TABLE", MatchAny
, "DROP", "COLUMN"))
2145 COMPLETE_WITH_ATTR(prev3_wd
, "");
2148 * If we have ALTER TABLE <sth> ALTER|DROP|RENAME|VALIDATE CONSTRAINT,
2149 * provide list of constraints
2151 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER|DROP|RENAME|VALIDATE", "CONSTRAINT"))
2153 completion_info_charp
= prev3_wd
;
2154 COMPLETE_WITH_QUERY(Query_for_constraint_of_table
);
2156 /* ALTER TABLE ALTER [COLUMN] <foo> */
2157 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER", "COLUMN", MatchAny
) ||
2158 Matches("ALTER", "TABLE", MatchAny
, "ALTER", MatchAny
))
2159 COMPLETE_WITH("TYPE", "SET", "RESET", "RESTART", "ADD", "DROP");
2160 /* ALTER TABLE ALTER [COLUMN] <foo> SET */
2161 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER", "COLUMN", MatchAny
, "SET") ||
2162 Matches("ALTER", "TABLE", MatchAny
, "ALTER", MatchAny
, "SET"))
2163 COMPLETE_WITH("(", "COMPRESSION", "DEFAULT", "NOT NULL", "STATISTICS", "STORAGE");
2164 /* ALTER TABLE ALTER [COLUMN] <foo> SET ( */
2165 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER", "COLUMN", MatchAny
, "SET", "(") ||
2166 Matches("ALTER", "TABLE", MatchAny
, "ALTER", MatchAny
, "SET", "("))
2167 COMPLETE_WITH("n_distinct", "n_distinct_inherited");
2168 /* ALTER TABLE ALTER [COLUMN] <foo> SET STORAGE */
2169 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER", "COLUMN", MatchAny
, "SET", "STORAGE") ||
2170 Matches("ALTER", "TABLE", MatchAny
, "ALTER", MatchAny
, "SET", "STORAGE"))
2171 COMPLETE_WITH("PLAIN", "EXTERNAL", "EXTENDED", "MAIN");
2172 /* ALTER TABLE ALTER [COLUMN] <foo> SET STATISTICS */
2173 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER", "COLUMN", MatchAny
, "SET", "STATISTICS") ||
2174 Matches("ALTER", "TABLE", MatchAny
, "ALTER", MatchAny
, "SET", "STATISTICS"))
2176 /* Enforce no completion here, as an integer has to be specified */
2178 /* ALTER TABLE ALTER [COLUMN] <foo> DROP */
2179 else if (Matches("ALTER", "TABLE", MatchAny
, "ALTER", "COLUMN", MatchAny
, "DROP") ||
2180 Matches("ALTER", "TABLE", MatchAny
, "ALTER", MatchAny
, "DROP"))
2181 COMPLETE_WITH("DEFAULT", "EXPRESSION", "IDENTITY", "NOT NULL");
2182 else if (Matches("ALTER", "TABLE", MatchAny
, "CLUSTER"))
2183 COMPLETE_WITH("ON");
2184 else if (Matches("ALTER", "TABLE", MatchAny
, "CLUSTER", "ON"))
2186 completion_info_charp
= prev3_wd
;
2187 COMPLETE_WITH_QUERY(Query_for_index_of_table
);
2189 /* If we have ALTER TABLE <sth> SET, provide list of attributes and '(' */
2190 else if (Matches("ALTER", "TABLE", MatchAny
, "SET"))
2191 COMPLETE_WITH("(", "ACCESS METHOD", "LOGGED", "SCHEMA",
2192 "TABLESPACE", "UNLOGGED", "WITH", "WITHOUT");
2195 * If we have ALTER TABLE <sth> SET ACCESS METHOD provide a list of table
2198 else if (Matches("ALTER", "TABLE", MatchAny
, "SET", "ACCESS", "METHOD"))
2199 COMPLETE_WITH_QUERY(Query_for_list_of_table_access_methods
);
2202 * If we have ALTER TABLE <sth> SET TABLESPACE provide a list of
2205 else if (Matches("ALTER", "TABLE", MatchAny
, "SET", "TABLESPACE"))
2206 COMPLETE_WITH_QUERY(Query_for_list_of_tablespaces
);
2207 /* If we have ALTER TABLE <sth> SET WITHOUT provide CLUSTER or OIDS */
2208 else if (Matches("ALTER", "TABLE", MatchAny
, "SET", "WITHOUT"))
2209 COMPLETE_WITH("CLUSTER", "OIDS");
2210 /* ALTER TABLE <foo> RESET */
2211 else if (Matches("ALTER", "TABLE", MatchAny
, "RESET"))
2213 /* ALTER TABLE <foo> SET|RESET ( */
2214 else if (Matches("ALTER", "TABLE", MatchAny
, "SET|RESET", "("))
2215 COMPLETE_WITH_LIST(table_storage_parameters
);
2216 else if (Matches("ALTER", "TABLE", MatchAny
, "REPLICA", "IDENTITY", "USING", "INDEX"))
2218 completion_info_charp
= prev5_wd
;
2219 COMPLETE_WITH_QUERY(Query_for_index_of_table
);
2221 else if (Matches("ALTER", "TABLE", MatchAny
, "REPLICA", "IDENTITY", "USING"))
2222 COMPLETE_WITH("INDEX");
2223 else if (Matches("ALTER", "TABLE", MatchAny
, "REPLICA", "IDENTITY"))
2224 COMPLETE_WITH("FULL", "NOTHING", "DEFAULT", "USING");
2225 else if (Matches("ALTER", "TABLE", MatchAny
, "REPLICA"))
2226 COMPLETE_WITH("IDENTITY");
2229 * If we have ALTER TABLE <foo> ATTACH PARTITION, provide a list of
2232 else if (Matches("ALTER", "TABLE", MatchAny
, "ATTACH", "PARTITION"))
2233 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, "");
2234 /* Limited completion support for partition bound specification */
2235 else if (TailMatches("ATTACH", "PARTITION", MatchAny
))
2236 COMPLETE_WITH("FOR VALUES", "DEFAULT");
2237 else if (TailMatches("FOR", "VALUES"))
2238 COMPLETE_WITH("FROM (", "IN (", "WITH (");
2241 * If we have ALTER TABLE <foo> DETACH PARTITION, provide a list of
2242 * partitions of <foo>.
2244 else if (Matches("ALTER", "TABLE", MatchAny
, "DETACH", "PARTITION"))
2246 completion_info_charp
= prev3_wd
;
2247 COMPLETE_WITH_QUERY(Query_for_partition_of_table
);
2249 else if (Matches("ALTER", "TABLE", MatchAny
, "DETACH", "PARTITION", MatchAny
))
2250 COMPLETE_WITH("CONCURRENTLY", "FINALIZE");
2252 /* ALTER TABLESPACE <foo> with RENAME TO, OWNER TO, SET, RESET */
2253 else if (Matches("ALTER", "TABLESPACE", MatchAny
))
2254 COMPLETE_WITH("RENAME TO", "OWNER TO", "SET", "RESET");
2255 /* ALTER TABLESPACE <foo> SET|RESET */
2256 else if (Matches("ALTER", "TABLESPACE", MatchAny
, "SET|RESET"))
2258 /* ALTER TABLESPACE <foo> SET|RESET ( */
2259 else if (Matches("ALTER", "TABLESPACE", MatchAny
, "SET|RESET", "("))
2260 COMPLETE_WITH("seq_page_cost", "random_page_cost",
2261 "effective_io_concurrency", "maintenance_io_concurrency");
2263 /* ALTER TEXT SEARCH */
2264 else if (Matches("ALTER", "TEXT", "SEARCH"))
2265 COMPLETE_WITH("CONFIGURATION", "DICTIONARY", "PARSER", "TEMPLATE");
2266 else if (Matches("ALTER", "TEXT", "SEARCH", "TEMPLATE|PARSER", MatchAny
))
2267 COMPLETE_WITH("RENAME TO", "SET SCHEMA");
2268 else if (Matches("ALTER", "TEXT", "SEARCH", "DICTIONARY", MatchAny
))
2269 COMPLETE_WITH("(", "OWNER TO", "RENAME TO", "SET SCHEMA");
2270 else if (Matches("ALTER", "TEXT", "SEARCH", "CONFIGURATION", MatchAny
))
2271 COMPLETE_WITH("ADD MAPPING FOR", "ALTER MAPPING",
2273 "OWNER TO", "RENAME TO", "SET SCHEMA");
2275 /* complete ALTER TYPE <foo> with actions */
2276 else if (Matches("ALTER", "TYPE", MatchAny
))
2277 COMPLETE_WITH("ADD ATTRIBUTE", "ADD VALUE", "ALTER ATTRIBUTE",
2279 "OWNER TO", "RENAME", "SET SCHEMA", "SET (");
2280 /* complete ALTER TYPE <foo> ADD with actions */
2281 else if (Matches("ALTER", "TYPE", MatchAny
, "ADD"))
2282 COMPLETE_WITH("ATTRIBUTE", "VALUE");
2283 /* ALTER TYPE <foo> RENAME */
2284 else if (Matches("ALTER", "TYPE", MatchAny
, "RENAME"))
2285 COMPLETE_WITH("ATTRIBUTE", "TO", "VALUE");
2286 /* ALTER TYPE xxx RENAME (ATTRIBUTE|VALUE) yyy */
2287 else if (Matches("ALTER", "TYPE", MatchAny
, "RENAME", "ATTRIBUTE|VALUE", MatchAny
))
2288 COMPLETE_WITH("TO");
2291 * If we have ALTER TYPE <sth> ALTER/DROP/RENAME ATTRIBUTE, provide list
2294 else if (Matches("ALTER", "TYPE", MatchAny
, "ALTER|DROP|RENAME", "ATTRIBUTE"))
2295 COMPLETE_WITH_ATTR(prev3_wd
, "");
2296 /* ALTER TYPE ALTER ATTRIBUTE <foo> */
2297 else if (Matches("ALTER", "TYPE", MatchAny
, "ALTER", "ATTRIBUTE", MatchAny
))
2298 COMPLETE_WITH("TYPE");
2299 /* complete ALTER GROUP <foo> */
2300 else if (Matches("ALTER", "GROUP", MatchAny
))
2301 COMPLETE_WITH("ADD USER", "DROP USER", "RENAME TO");
2302 /* complete ALTER GROUP <foo> ADD|DROP with USER */
2303 else if (Matches("ALTER", "GROUP", MatchAny
, "ADD|DROP"))
2304 COMPLETE_WITH("USER");
2305 /* complete ALTER GROUP <foo> ADD|DROP USER with a user name */
2306 else if (Matches("ALTER", "GROUP", MatchAny
, "ADD|DROP", "USER"))
2307 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
2310 * If we have ALTER TYPE <sth> RENAME VALUE, provide list of enum values
2312 else if (Matches("ALTER", "TYPE", MatchAny
, "RENAME", "VALUE"))
2313 COMPLETE_WITH_ENUM_VALUE(prev3_wd
);
2316 * ANALYZE [ ( option [, ...] ) ] [ table_and_columns [, ...] ]
2317 * ANALYZE [ VERBOSE ] [ table_and_columns [, ...] ]
2319 else if (Matches("ANALYZE"))
2320 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_analyzables
,
2321 " UNION SELECT 'VERBOSE'");
2322 else if (HeadMatches("ANALYZE", "(*") &&
2323 !HeadMatches("ANALYZE", "(*)"))
2326 * This fires if we're in an unfinished parenthesized option list.
2327 * get_previous_words treats a completed parenthesized option list as
2328 * one word, so the above test is correct.
2330 if (ends_with(prev_wd
, '(') || ends_with(prev_wd
, ','))
2331 COMPLETE_WITH("VERBOSE", "SKIP_LOCKED");
2332 else if (TailMatches("VERBOSE|SKIP_LOCKED"))
2333 COMPLETE_WITH("ON", "OFF");
2335 else if (HeadMatches("ANALYZE") && TailMatches("("))
2336 /* "ANALYZE (" should be caught above, so assume we want columns */
2337 COMPLETE_WITH_ATTR(prev2_wd
, "");
2338 else if (HeadMatches("ANALYZE"))
2339 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_analyzables
, NULL
);
2342 else if (Matches("BEGIN"))
2343 COMPLETE_WITH("WORK", "TRANSACTION", "ISOLATION LEVEL", "READ", "DEFERRABLE", "NOT DEFERRABLE");
2345 else if (Matches("END|ABORT"))
2346 COMPLETE_WITH("AND", "WORK", "TRANSACTION");
2348 else if (Matches("COMMIT"))
2349 COMPLETE_WITH("AND", "WORK", "TRANSACTION", "PREPARED");
2350 /* RELEASE SAVEPOINT */
2351 else if (Matches("RELEASE"))
2352 COMPLETE_WITH("SAVEPOINT");
2354 else if (Matches("ROLLBACK"))
2355 COMPLETE_WITH("AND", "WORK", "TRANSACTION", "TO SAVEPOINT", "PREPARED");
2356 else if (Matches("ABORT|END|COMMIT|ROLLBACK", "AND"))
2357 COMPLETE_WITH("CHAIN");
2359 else if (Matches("CALL"))
2360 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_procedures
, NULL
);
2361 else if (Matches("CALL", MatchAny
))
2364 else if (Matches("CLOSE"))
2365 COMPLETE_WITH_QUERY(Query_for_list_of_cursors
2366 " UNION SELECT 'ALL'");
2368 else if (Matches("CLUSTER"))
2369 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_clusterables
, "UNION SELECT 'VERBOSE'");
2370 else if (Matches("CLUSTER", "VERBOSE") ||
2371 Matches("CLUSTER", "(*)"))
2372 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_clusterables
, NULL
);
2373 /* If we have CLUSTER <sth>, then add "USING" */
2374 else if (Matches("CLUSTER", MatchAnyExcept("VERBOSE|ON|(|(*)")))
2375 COMPLETE_WITH("USING");
2376 /* If we have CLUSTER VERBOSE <sth>, then add "USING" */
2377 else if (Matches("CLUSTER", "VERBOSE|(*)", MatchAny
))
2378 COMPLETE_WITH("USING");
2379 /* If we have CLUSTER <sth> USING, then add the index as well */
2380 else if (Matches("CLUSTER", MatchAny
, "USING") ||
2381 Matches("CLUSTER", "VERBOSE|(*)", MatchAny
, "USING"))
2383 completion_info_charp
= prev2_wd
;
2384 COMPLETE_WITH_QUERY(Query_for_index_of_table
);
2386 else if (HeadMatches("CLUSTER", "(*") &&
2387 !HeadMatches("CLUSTER", "(*)"))
2390 * This fires if we're in an unfinished parenthesized option list.
2391 * get_previous_words treats a completed parenthesized option list as
2392 * one word, so the above test is correct.
2394 if (ends_with(prev_wd
, '(') || ends_with(prev_wd
, ','))
2395 COMPLETE_WITH("VERBOSE");
2399 else if (Matches("COMMENT"))
2400 COMPLETE_WITH("ON");
2401 else if (Matches("COMMENT", "ON"))
2402 COMPLETE_WITH("ACCESS METHOD", "CAST", "COLLATION", "CONVERSION",
2403 "DATABASE", "EVENT TRIGGER", "EXTENSION",
2404 "FOREIGN DATA WRAPPER", "FOREIGN TABLE", "SERVER",
2405 "INDEX", "LANGUAGE", "POLICY", "PUBLICATION", "RULE",
2406 "SCHEMA", "SEQUENCE", "STATISTICS", "SUBSCRIPTION",
2407 "TABLE", "TYPE", "VIEW", "MATERIALIZED VIEW",
2408 "COLUMN", "AGGREGATE", "FUNCTION",
2409 "PROCEDURE", "ROUTINE",
2410 "OPERATOR", "TRIGGER", "CONSTRAINT", "DOMAIN",
2411 "LARGE OBJECT", "TABLESPACE", "TEXT SEARCH", "ROLE");
2412 else if (Matches("COMMENT", "ON", "ACCESS", "METHOD"))
2413 COMPLETE_WITH_QUERY(Query_for_list_of_access_methods
);
2414 else if (Matches("COMMENT", "ON", "FOREIGN"))
2415 COMPLETE_WITH("DATA WRAPPER", "TABLE");
2416 else if (Matches("COMMENT", "ON", "TEXT", "SEARCH"))
2417 COMPLETE_WITH("CONFIGURATION", "DICTIONARY", "PARSER", "TEMPLATE");
2418 else if (Matches("COMMENT", "ON", "CONSTRAINT"))
2419 COMPLETE_WITH_QUERY(Query_for_all_table_constraints
);
2420 else if (Matches("COMMENT", "ON", "CONSTRAINT", MatchAny
))
2421 COMPLETE_WITH("ON");
2422 else if (Matches("COMMENT", "ON", "CONSTRAINT", MatchAny
, "ON"))
2424 completion_info_charp
= prev2_wd
;
2425 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_constraint
);
2427 else if (Matches("COMMENT", "ON", "MATERIALIZED", "VIEW"))
2428 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_matviews
, NULL
);
2429 else if (Matches("COMMENT", "ON", "EVENT", "TRIGGER"))
2430 COMPLETE_WITH_QUERY(Query_for_list_of_event_triggers
);
2431 else if (Matches("COMMENT", "ON", MatchAny
, MatchAnyExcept("IS")) ||
2432 Matches("COMMENT", "ON", MatchAny
, MatchAny
, MatchAnyExcept("IS")) ||
2433 Matches("COMMENT", "ON", MatchAny
, MatchAny
, MatchAny
, MatchAnyExcept("IS")))
2434 COMPLETE_WITH("IS");
2439 * If we have COPY, offer list of tables or "(" (Also cover the analogous
2440 * backslash command).
2442 else if (Matches("COPY|\\copy"))
2443 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
,
2444 " UNION ALL SELECT '('");
2445 /* Complete COPY ( with legal query commands */
2446 else if (Matches("COPY|\\copy", "("))
2447 COMPLETE_WITH("SELECT", "TABLE", "VALUES", "INSERT INTO", "UPDATE", "DELETE FROM", "WITH");
2448 /* Complete COPY <sth> */
2449 else if (Matches("COPY|\\copy", MatchAny
))
2450 COMPLETE_WITH("FROM", "TO");
2451 /* Complete COPY <sth> FROM|TO with filename */
2452 else if (Matches("COPY", MatchAny
, "FROM|TO"))
2454 completion_charp
= "";
2455 completion_force_quote
= true; /* COPY requires quoted filename */
2456 matches
= rl_completion_matches(text
, complete_from_files
);
2458 else if (Matches("\\copy", MatchAny
, "FROM|TO"))
2460 completion_charp
= "";
2461 completion_force_quote
= false;
2462 matches
= rl_completion_matches(text
, complete_from_files
);
2465 /* Complete COPY <sth> TO <sth> */
2466 else if (Matches("COPY|\\copy", MatchAny
, "TO", MatchAny
))
2467 COMPLETE_WITH("WITH (");
2469 /* Complete COPY <sth> FROM <sth> */
2470 else if (Matches("COPY|\\copy", MatchAny
, "FROM", MatchAny
))
2471 COMPLETE_WITH("WITH (", "WHERE");
2473 /* Complete COPY <sth> FROM|TO filename WITH ( */
2474 else if (Matches("COPY|\\copy", MatchAny
, "FROM|TO", MatchAny
, "WITH", "("))
2475 COMPLETE_WITH("FORMAT", "FREEZE", "DELIMITER", "NULL",
2476 "HEADER", "QUOTE", "ESCAPE", "FORCE_QUOTE",
2477 "FORCE_NOT_NULL", "FORCE_NULL", "ENCODING");
2479 /* Complete COPY <sth> FROM|TO filename WITH (FORMAT */
2480 else if (Matches("COPY|\\copy", MatchAny
, "FROM|TO", MatchAny
, "WITH", "(", "FORMAT"))
2481 COMPLETE_WITH("binary", "csv", "text");
2483 /* Complete COPY <sth> FROM <sth> WITH (<options>) */
2484 else if (Matches("COPY|\\copy", MatchAny
, "FROM", MatchAny
, "WITH", MatchAny
))
2485 COMPLETE_WITH("WHERE");
2487 /* CREATE ACCESS METHOD */
2488 /* Complete "CREATE ACCESS METHOD <name>" */
2489 else if (Matches("CREATE", "ACCESS", "METHOD", MatchAny
))
2490 COMPLETE_WITH("TYPE");
2491 /* Complete "CREATE ACCESS METHOD <name> TYPE" */
2492 else if (Matches("CREATE", "ACCESS", "METHOD", MatchAny
, "TYPE"))
2493 COMPLETE_WITH("INDEX", "TABLE");
2494 /* Complete "CREATE ACCESS METHOD <name> TYPE <type>" */
2495 else if (Matches("CREATE", "ACCESS", "METHOD", MatchAny
, "TYPE", MatchAny
))
2496 COMPLETE_WITH("HANDLER");
2498 /* CREATE COLLATION */
2499 else if (Matches("CREATE", "COLLATION", MatchAny
))
2500 COMPLETE_WITH("(", "FROM");
2501 else if (Matches("CREATE", "COLLATION", MatchAny
, "FROM"))
2502 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_collations
, NULL
);
2503 else if (HeadMatches("CREATE", "COLLATION", MatchAny
, "(*"))
2505 if (TailMatches("(|*,"))
2506 COMPLETE_WITH("LOCALE =", "LC_COLLATE =", "LC_CTYPE =",
2507 "PROVIDER =", "DETERMINISTIC =");
2508 else if (TailMatches("PROVIDER", "="))
2509 COMPLETE_WITH("libc", "icu");
2510 else if (TailMatches("DETERMINISTIC", "="))
2511 COMPLETE_WITH("true", "false");
2514 /* CREATE DATABASE */
2515 else if (Matches("CREATE", "DATABASE", MatchAny
))
2516 COMPLETE_WITH("OWNER", "TEMPLATE", "ENCODING", "TABLESPACE",
2518 "ALLOW_CONNECTIONS", "CONNECTION LIMIT",
2519 "LC_COLLATE", "LC_CTYPE", "LOCALE");
2521 else if (Matches("CREATE", "DATABASE", MatchAny
, "TEMPLATE"))
2522 COMPLETE_WITH_QUERY(Query_for_list_of_template_databases
);
2524 /* CREATE EXTENSION */
2525 /* Complete with available extensions rather than installed ones. */
2526 else if (Matches("CREATE", "EXTENSION"))
2527 COMPLETE_WITH_QUERY(Query_for_list_of_available_extensions
);
2528 /* CREATE EXTENSION <name> */
2529 else if (Matches("CREATE", "EXTENSION", MatchAny
))
2530 COMPLETE_WITH("WITH SCHEMA", "CASCADE", "VERSION");
2531 /* CREATE EXTENSION <name> VERSION */
2532 else if (Matches("CREATE", "EXTENSION", MatchAny
, "VERSION"))
2534 completion_info_charp
= prev2_wd
;
2535 COMPLETE_WITH_QUERY(Query_for_list_of_available_extension_versions
);
2538 /* CREATE FOREIGN */
2539 else if (Matches("CREATE", "FOREIGN"))
2540 COMPLETE_WITH("DATA WRAPPER", "TABLE");
2542 /* CREATE FOREIGN DATA WRAPPER */
2543 else if (Matches("CREATE", "FOREIGN", "DATA", "WRAPPER", MatchAny
))
2544 COMPLETE_WITH("HANDLER", "VALIDATOR", "OPTIONS");
2546 /* CREATE INDEX --- is allowed inside CREATE SCHEMA, so use TailMatches */
2547 /* First off we complete CREATE UNIQUE with "INDEX" */
2548 else if (TailMatches("CREATE", "UNIQUE"))
2549 COMPLETE_WITH("INDEX");
2552 * If we have CREATE|UNIQUE INDEX, then add "ON", "CONCURRENTLY", and
2555 else if (TailMatches("CREATE|UNIQUE", "INDEX"))
2556 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
,
2557 " UNION SELECT 'ON'"
2558 " UNION SELECT 'CONCURRENTLY'");
2561 * Complete ... INDEX|CONCURRENTLY [<name>] ON with a list of relations
2562 * that indexes can be created on
2564 else if (TailMatches("INDEX|CONCURRENTLY", MatchAny
, "ON") ||
2565 TailMatches("INDEX|CONCURRENTLY", "ON"))
2566 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexables
, NULL
);
2569 * Complete CREATE|UNIQUE INDEX CONCURRENTLY with "ON" and existing
2572 else if (TailMatches("CREATE|UNIQUE", "INDEX", "CONCURRENTLY"))
2573 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
,
2574 " UNION SELECT 'ON'");
2575 /* Complete CREATE|UNIQUE INDEX [CONCURRENTLY] <sth> with "ON" */
2576 else if (TailMatches("CREATE|UNIQUE", "INDEX", MatchAny
) ||
2577 TailMatches("CREATE|UNIQUE", "INDEX", "CONCURRENTLY", MatchAny
))
2578 COMPLETE_WITH("ON");
2581 * Complete INDEX <name> ON <table> with a list of table columns (which
2582 * should really be in parens)
2584 else if (TailMatches("INDEX", MatchAny
, "ON", MatchAny
) ||
2585 TailMatches("INDEX|CONCURRENTLY", "ON", MatchAny
))
2586 COMPLETE_WITH("(", "USING");
2587 else if (TailMatches("INDEX", MatchAny
, "ON", MatchAny
, "(") ||
2588 TailMatches("INDEX|CONCURRENTLY", "ON", MatchAny
, "("))
2589 COMPLETE_WITH_ATTR(prev2_wd
, "");
2590 /* same if you put in USING */
2591 else if (TailMatches("ON", MatchAny
, "USING", MatchAny
, "("))
2592 COMPLETE_WITH_ATTR(prev4_wd
, "");
2593 /* Complete USING with an index method */
2594 else if (TailMatches("INDEX", MatchAny
, MatchAny
, "ON", MatchAny
, "USING") ||
2595 TailMatches("INDEX", MatchAny
, "ON", MatchAny
, "USING") ||
2596 TailMatches("INDEX", "ON", MatchAny
, "USING"))
2597 COMPLETE_WITH_QUERY(Query_for_list_of_index_access_methods
);
2598 else if (TailMatches("ON", MatchAny
, "USING", MatchAny
) &&
2599 !TailMatches("POLICY", MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
) &&
2600 !TailMatches("FOR", MatchAny
, MatchAny
, MatchAny
))
2603 /* CREATE OR REPLACE */
2604 else if (Matches("CREATE", "OR"))
2605 COMPLETE_WITH("REPLACE");
2608 /* Complete "CREATE POLICY <name> ON" */
2609 else if (Matches("CREATE", "POLICY", MatchAny
))
2610 COMPLETE_WITH("ON");
2611 /* Complete "CREATE POLICY <name> ON <table>" */
2612 else if (Matches("CREATE", "POLICY", MatchAny
, "ON"))
2613 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
2614 /* Complete "CREATE POLICY <name> ON <table> AS|FOR|TO|USING|WITH CHECK" */
2615 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
))
2616 COMPLETE_WITH("AS", "FOR", "TO", "USING (", "WITH CHECK (");
2617 /* CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE */
2618 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS"))
2619 COMPLETE_WITH("PERMISSIVE", "RESTRICTIVE");
2622 * CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE
2623 * FOR|TO|USING|WITH CHECK
2625 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
))
2626 COMPLETE_WITH("FOR", "TO", "USING", "WITH CHECK");
2627 /* CREATE POLICY <name> ON <table> FOR ALL|SELECT|INSERT|UPDATE|DELETE */
2628 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "FOR"))
2629 COMPLETE_WITH("ALL", "SELECT", "INSERT", "UPDATE", "DELETE");
2630 /* Complete "CREATE POLICY <name> ON <table> FOR INSERT TO|WITH CHECK" */
2631 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "FOR", "INSERT"))
2632 COMPLETE_WITH("TO", "WITH CHECK (");
2633 /* Complete "CREATE POLICY <name> ON <table> FOR SELECT|DELETE TO|USING" */
2634 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "FOR", "SELECT|DELETE"))
2635 COMPLETE_WITH("TO", "USING (");
2636 /* CREATE POLICY <name> ON <table> FOR ALL|UPDATE TO|USING|WITH CHECK */
2637 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "FOR", "ALL|UPDATE"))
2638 COMPLETE_WITH("TO", "USING (", "WITH CHECK (");
2639 /* Complete "CREATE POLICY <name> ON <table> TO <role>" */
2640 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "TO"))
2641 COMPLETE_WITH_QUERY(Query_for_list_of_grant_roles
);
2642 /* Complete "CREATE POLICY <name> ON <table> USING (" */
2643 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "USING"))
2647 * CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE FOR
2648 * ALL|SELECT|INSERT|UPDATE|DELETE
2650 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
, "FOR"))
2651 COMPLETE_WITH("ALL", "SELECT", "INSERT", "UPDATE", "DELETE");
2654 * Complete "CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE FOR
2655 * INSERT TO|WITH CHECK"
2657 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
, "FOR", "INSERT"))
2658 COMPLETE_WITH("TO", "WITH CHECK (");
2661 * Complete "CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE FOR
2662 * SELECT|DELETE TO|USING"
2664 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
, "FOR", "SELECT|DELETE"))
2665 COMPLETE_WITH("TO", "USING (");
2668 * CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE FOR
2669 * ALL|UPDATE TO|USING|WITH CHECK
2671 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
, "FOR", "ALL|UPDATE"))
2672 COMPLETE_WITH("TO", "USING (", "WITH CHECK (");
2675 * Complete "CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE TO
2678 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
, "TO"))
2679 COMPLETE_WITH_QUERY(Query_for_list_of_grant_roles
);
2682 * Complete "CREATE POLICY <name> ON <table> AS PERMISSIVE|RESTRICTIVE
2685 else if (Matches("CREATE", "POLICY", MatchAny
, "ON", MatchAny
, "AS", MatchAny
, "USING"))
2689 /* CREATE PUBLICATION */
2690 else if (Matches("CREATE", "PUBLICATION", MatchAny
))
2691 COMPLETE_WITH("FOR TABLE", "FOR ALL TABLES", "WITH (");
2692 else if (Matches("CREATE", "PUBLICATION", MatchAny
, "FOR"))
2693 COMPLETE_WITH("TABLE", "ALL TABLES");
2694 else if (Matches("CREATE", "PUBLICATION", MatchAny
, "FOR", "ALL"))
2695 COMPLETE_WITH("TABLES");
2696 else if (Matches("CREATE", "PUBLICATION", MatchAny
, "FOR", "ALL", "TABLES")
2697 || Matches("CREATE", "PUBLICATION", MatchAny
, "FOR", "TABLE", MatchAny
))
2698 COMPLETE_WITH("WITH (");
2699 /* Complete "CREATE PUBLICATION <name> FOR TABLE" with "<table>, ..." */
2700 else if (Matches("CREATE", "PUBLICATION", MatchAny
, "FOR", "TABLE"))
2701 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
2702 /* Complete "CREATE PUBLICATION <name> [...] WITH" */
2703 else if (HeadMatches("CREATE", "PUBLICATION") && TailMatches("WITH", "("))
2704 COMPLETE_WITH("publish", "publish_via_partition_root");
2707 /* Complete "CREATE [ OR REPLACE ] RULE <sth>" with "AS ON" */
2708 else if (Matches("CREATE", "RULE", MatchAny
) ||
2709 Matches("CREATE", "OR", "REPLACE", "RULE", MatchAny
))
2710 COMPLETE_WITH("AS ON");
2711 /* Complete "CREATE [ OR REPLACE ] RULE <sth> AS" with "ON" */
2712 else if (Matches("CREATE", "RULE", MatchAny
, "AS") ||
2713 Matches("CREATE", "OR", "REPLACE", "RULE", MatchAny
, "AS"))
2714 COMPLETE_WITH("ON");
2717 * Complete "CREATE [ OR REPLACE ] RULE <sth> AS ON" with
2718 * SELECT|UPDATE|INSERT|DELETE
2720 else if (Matches("CREATE", "RULE", MatchAny
, "AS", "ON") ||
2721 Matches("CREATE", "OR", "REPLACE", "RULE", MatchAny
, "AS", "ON"))
2722 COMPLETE_WITH("SELECT", "UPDATE", "INSERT", "DELETE");
2723 /* Complete "AS ON SELECT|UPDATE|INSERT|DELETE" with a "TO" */
2724 else if (TailMatches("AS", "ON", "SELECT|UPDATE|INSERT|DELETE"))
2725 COMPLETE_WITH("TO");
2726 /* Complete "AS ON <sth> TO" with a table name */
2727 else if (TailMatches("AS", "ON", "SELECT|UPDATE|INSERT|DELETE", "TO"))
2728 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
2730 /* CREATE SEQUENCE --- is allowed inside CREATE SCHEMA, so use TailMatches */
2731 else if (TailMatches("CREATE", "SEQUENCE", MatchAny
) ||
2732 TailMatches("CREATE", "TEMP|TEMPORARY", "SEQUENCE", MatchAny
))
2733 COMPLETE_WITH("INCREMENT BY", "MINVALUE", "MAXVALUE", "NO", "CACHE",
2734 "CYCLE", "OWNED BY", "START WITH");
2735 else if (TailMatches("CREATE", "SEQUENCE", MatchAny
, "NO") ||
2736 TailMatches("CREATE", "TEMP|TEMPORARY", "SEQUENCE", MatchAny
, "NO"))
2737 COMPLETE_WITH("MINVALUE", "MAXVALUE", "CYCLE");
2739 /* CREATE SERVER <name> */
2740 else if (Matches("CREATE", "SERVER", MatchAny
))
2741 COMPLETE_WITH("TYPE", "VERSION", "FOREIGN DATA WRAPPER");
2743 /* CREATE STATISTICS <name> */
2744 else if (Matches("CREATE", "STATISTICS", MatchAny
))
2745 COMPLETE_WITH("(", "ON");
2746 else if (Matches("CREATE", "STATISTICS", MatchAny
, "("))
2747 COMPLETE_WITH("ndistinct", "dependencies", "mcv");
2748 else if (Matches("CREATE", "STATISTICS", MatchAny
, "(*)"))
2749 COMPLETE_WITH("ON");
2750 else if (HeadMatches("CREATE", "STATISTICS", MatchAny
) &&
2751 TailMatches("FROM"))
2752 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
2754 /* CREATE TABLE --- is allowed inside CREATE SCHEMA, so use TailMatches */
2755 /* Complete "CREATE TEMP/TEMPORARY" with the possible temp objects */
2756 else if (TailMatches("CREATE", "TEMP|TEMPORARY"))
2757 COMPLETE_WITH("SEQUENCE", "TABLE", "VIEW");
2758 /* Complete "CREATE UNLOGGED" with TABLE or MATVIEW */
2759 else if (TailMatches("CREATE", "UNLOGGED"))
2760 COMPLETE_WITH("TABLE", "MATERIALIZED VIEW");
2761 /* Complete PARTITION BY with RANGE ( or LIST ( or ... */
2762 else if (TailMatches("PARTITION", "BY"))
2763 COMPLETE_WITH("RANGE (", "LIST (", "HASH (");
2764 /* If we have xxx PARTITION OF, provide a list of partitioned tables */
2765 else if (TailMatches("PARTITION", "OF"))
2766 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_partitioned_tables
, "");
2767 /* Limited completion support for partition bound specification */
2768 else if (TailMatches("PARTITION", "OF", MatchAny
))
2769 COMPLETE_WITH("FOR VALUES", "DEFAULT");
2770 /* Complete CREATE TABLE <name> with '(', OF or PARTITION OF */
2771 else if (TailMatches("CREATE", "TABLE", MatchAny
) ||
2772 TailMatches("CREATE", "TEMP|TEMPORARY|UNLOGGED", "TABLE", MatchAny
))
2773 COMPLETE_WITH("(", "OF", "PARTITION OF");
2774 /* Complete CREATE TABLE <name> OF with list of composite types */
2775 else if (TailMatches("CREATE", "TABLE", MatchAny
, "OF") ||
2776 TailMatches("CREATE", "TEMP|TEMPORARY|UNLOGGED", "TABLE", MatchAny
, "OF"))
2777 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_composite_datatypes
, NULL
);
2778 /* Complete CREATE TABLE name (...) with supported options */
2779 else if (TailMatches("CREATE", "TABLE", MatchAny
, "(*)") ||
2780 TailMatches("CREATE", "UNLOGGED", "TABLE", MatchAny
, "(*)"))
2781 COMPLETE_WITH("INHERITS (", "PARTITION BY", "USING", "TABLESPACE", "WITH (");
2782 else if (TailMatches("CREATE", "TEMP|TEMPORARY", "TABLE", MatchAny
, "(*)"))
2783 COMPLETE_WITH("INHERITS (", "ON COMMIT", "PARTITION BY",
2784 "TABLESPACE", "WITH (");
2785 /* Complete CREATE TABLE (...) USING with table access methods */
2786 else if (TailMatches("CREATE", "TABLE", MatchAny
, "(*)", "USING") ||
2787 TailMatches("CREATE", "TEMP|TEMPORARY|UNLOGGED", "TABLE", MatchAny
, "(*)", "USING"))
2788 COMPLETE_WITH_QUERY(Query_for_list_of_table_access_methods
);
2789 /* Complete CREATE TABLE (...) WITH with storage parameters */
2790 else if (TailMatches("CREATE", "TABLE", MatchAny
, "(*)", "WITH", "(") ||
2791 TailMatches("CREATE", "TEMP|TEMPORARY|UNLOGGED", "TABLE", MatchAny
, "(*)", "WITH", "("))
2792 COMPLETE_WITH_LIST(table_storage_parameters
);
2793 /* Complete CREATE TABLE ON COMMIT with actions */
2794 else if (TailMatches("CREATE", "TEMP|TEMPORARY", "TABLE", MatchAny
, "(*)", "ON", "COMMIT"))
2795 COMPLETE_WITH("DELETE ROWS", "DROP", "PRESERVE ROWS");
2797 /* CREATE TABLESPACE */
2798 else if (Matches("CREATE", "TABLESPACE", MatchAny
))
2799 COMPLETE_WITH("OWNER", "LOCATION");
2800 /* Complete CREATE TABLESPACE name OWNER name with "LOCATION" */
2801 else if (Matches("CREATE", "TABLESPACE", MatchAny
, "OWNER", MatchAny
))
2802 COMPLETE_WITH("LOCATION");
2804 /* CREATE TEXT SEARCH */
2805 else if (Matches("CREATE", "TEXT", "SEARCH"))
2806 COMPLETE_WITH("CONFIGURATION", "DICTIONARY", "PARSER", "TEMPLATE");
2807 else if (Matches("CREATE", "TEXT", "SEARCH", "CONFIGURATION|DICTIONARY|PARSER|TEMPLATE", MatchAny
))
2810 /* CREATE SUBSCRIPTION */
2811 else if (Matches("CREATE", "SUBSCRIPTION", MatchAny
))
2812 COMPLETE_WITH("CONNECTION");
2813 else if (Matches("CREATE", "SUBSCRIPTION", MatchAny
, "CONNECTION", MatchAny
))
2814 COMPLETE_WITH("PUBLICATION");
2815 else if (Matches("CREATE", "SUBSCRIPTION", MatchAny
, "CONNECTION",
2816 MatchAny
, "PUBLICATION"))
2818 /* complete with nothing here as this refers to remote publications */
2820 else if (HeadMatches("CREATE", "SUBSCRIPTION") && TailMatches("PUBLICATION", MatchAny
))
2821 COMPLETE_WITH("WITH (");
2822 /* Complete "CREATE SUBSCRIPTION <name> ... WITH ( <opt>" */
2823 else if (HeadMatches("CREATE", "SUBSCRIPTION") && TailMatches("WITH", "("))
2824 COMPLETE_WITH("binary", "connect", "copy_data", "create_slot",
2825 "enabled", "slot_name", "streaming",
2826 "synchronous_commit", "two_phase");
2828 /* CREATE TRIGGER --- is allowed inside CREATE SCHEMA, so use TailMatches */
2831 * Complete CREATE [ OR REPLACE ] TRIGGER <name> with BEFORE|AFTER|INSTEAD
2834 else if (TailMatches("CREATE", "TRIGGER", MatchAny
) ||
2835 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
))
2836 COMPLETE_WITH("BEFORE", "AFTER", "INSTEAD OF");
2839 * Complete CREATE [ OR REPLACE ] TRIGGER <name> BEFORE,AFTER with an
2842 else if (TailMatches("CREATE", "TRIGGER", MatchAny
, "BEFORE|AFTER") ||
2843 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
, "BEFORE|AFTER"))
2844 COMPLETE_WITH("INSERT", "DELETE", "UPDATE", "TRUNCATE");
2845 /* Complete CREATE [ OR REPLACE ] TRIGGER <name> INSTEAD OF with an event */
2846 else if (TailMatches("CREATE", "TRIGGER", MatchAny
, "INSTEAD", "OF") ||
2847 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
, "INSTEAD", "OF"))
2848 COMPLETE_WITH("INSERT", "DELETE", "UPDATE");
2851 * Complete CREATE [ OR REPLACE ] TRIGGER <name> BEFORE,AFTER sth with
2854 else if (TailMatches("CREATE", "TRIGGER", MatchAny
, "BEFORE|AFTER", MatchAny
) ||
2855 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
, "BEFORE|AFTER", MatchAny
) ||
2856 TailMatches("CREATE", "TRIGGER", MatchAny
, "INSTEAD", "OF", MatchAny
) ||
2857 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
, "INSTEAD", "OF", MatchAny
))
2858 COMPLETE_WITH("ON", "OR");
2861 * Complete CREATE [ OR REPLACE ] TRIGGER <name> BEFORE,AFTER event ON
2862 * with a list of tables. EXECUTE FUNCTION is the recommended grammar
2863 * instead of EXECUTE PROCEDURE in version 11 and upwards.
2865 else if (TailMatches("CREATE", "TRIGGER", MatchAny
, "BEFORE|AFTER", MatchAny
, "ON") ||
2866 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
, "BEFORE|AFTER", MatchAny
, "ON"))
2867 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
2870 * Complete CREATE [ OR REPLACE ] TRIGGER ... INSTEAD OF event ON with a
2873 else if (TailMatches("CREATE", "TRIGGER", MatchAny
, "INSTEAD", "OF", MatchAny
, "ON") ||
2874 TailMatches("CREATE", "OR", "REPLACE", "TRIGGER", MatchAny
, "INSTEAD", "OF", MatchAny
, "ON"))
2875 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_views
, NULL
);
2876 else if ((HeadMatches("CREATE", "TRIGGER") ||
2877 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2878 TailMatches("ON", MatchAny
))
2880 if (pset
.sversion
>= 110000)
2881 COMPLETE_WITH("NOT DEFERRABLE", "DEFERRABLE", "INITIALLY",
2882 "REFERENCING", "FOR", "WHEN (", "EXECUTE FUNCTION");
2884 COMPLETE_WITH("NOT DEFERRABLE", "DEFERRABLE", "INITIALLY",
2885 "REFERENCING", "FOR", "WHEN (", "EXECUTE PROCEDURE");
2887 else if ((HeadMatches("CREATE", "TRIGGER") ||
2888 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2889 (TailMatches("DEFERRABLE") || TailMatches("INITIALLY", "IMMEDIATE|DEFERRED")))
2891 if (pset
.sversion
>= 110000)
2892 COMPLETE_WITH("REFERENCING", "FOR", "WHEN (", "EXECUTE FUNCTION");
2894 COMPLETE_WITH("REFERENCING", "FOR", "WHEN (", "EXECUTE PROCEDURE");
2896 else if ((HeadMatches("CREATE", "TRIGGER") ||
2897 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2898 TailMatches("REFERENCING"))
2899 COMPLETE_WITH("OLD TABLE", "NEW TABLE");
2900 else if ((HeadMatches("CREATE", "TRIGGER") ||
2901 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2902 TailMatches("OLD|NEW", "TABLE"))
2903 COMPLETE_WITH("AS");
2904 else if ((HeadMatches("CREATE", "TRIGGER") ||
2905 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2906 (TailMatches("REFERENCING", "OLD", "TABLE", "AS", MatchAny
) ||
2907 TailMatches("REFERENCING", "OLD", "TABLE", MatchAny
)))
2909 if (pset
.sversion
>= 110000)
2910 COMPLETE_WITH("NEW TABLE", "FOR", "WHEN (", "EXECUTE FUNCTION");
2912 COMPLETE_WITH("NEW TABLE", "FOR", "WHEN (", "EXECUTE PROCEDURE");
2914 else if ((HeadMatches("CREATE", "TRIGGER") ||
2915 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2916 (TailMatches("REFERENCING", "NEW", "TABLE", "AS", MatchAny
) ||
2917 TailMatches("REFERENCING", "NEW", "TABLE", MatchAny
)))
2919 if (pset
.sversion
>= 110000)
2920 COMPLETE_WITH("OLD TABLE", "FOR", "WHEN (", "EXECUTE FUNCTION");
2922 COMPLETE_WITH("OLD TABLE", "FOR", "WHEN (", "EXECUTE PROCEDURE");
2924 else if ((HeadMatches("CREATE", "TRIGGER") ||
2925 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2926 (TailMatches("REFERENCING", "OLD|NEW", "TABLE", "AS", MatchAny
, "OLD|NEW", "TABLE", "AS", MatchAny
) ||
2927 TailMatches("REFERENCING", "OLD|NEW", "TABLE", MatchAny
, "OLD|NEW", "TABLE", "AS", MatchAny
) ||
2928 TailMatches("REFERENCING", "OLD|NEW", "TABLE", "AS", MatchAny
, "OLD|NEW", "TABLE", MatchAny
) ||
2929 TailMatches("REFERENCING", "OLD|NEW", "TABLE", MatchAny
, "OLD|NEW", "TABLE", MatchAny
)))
2931 if (pset
.sversion
>= 110000)
2932 COMPLETE_WITH("FOR", "WHEN (", "EXECUTE FUNCTION");
2934 COMPLETE_WITH("FOR", "WHEN (", "EXECUTE PROCEDURE");
2936 else if ((HeadMatches("CREATE", "TRIGGER") ||
2937 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2939 COMPLETE_WITH("EACH", "ROW", "STATEMENT");
2940 else if ((HeadMatches("CREATE", "TRIGGER") ||
2941 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2942 TailMatches("FOR", "EACH"))
2943 COMPLETE_WITH("ROW", "STATEMENT");
2944 else if ((HeadMatches("CREATE", "TRIGGER") ||
2945 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2946 (TailMatches("FOR", "EACH", "ROW|STATEMENT") ||
2947 TailMatches("FOR", "ROW|STATEMENT")))
2949 if (pset
.sversion
>= 110000)
2950 COMPLETE_WITH("WHEN (", "EXECUTE FUNCTION");
2952 COMPLETE_WITH("WHEN (", "EXECUTE PROCEDURE");
2954 else if ((HeadMatches("CREATE", "TRIGGER") ||
2955 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2956 TailMatches("WHEN", "(*)"))
2958 if (pset
.sversion
>= 110000)
2959 COMPLETE_WITH("EXECUTE FUNCTION");
2961 COMPLETE_WITH("EXECUTE PROCEDURE");
2965 * Complete CREATE [ OR REPLACE ] TRIGGER ... EXECUTE with
2966 * PROCEDURE|FUNCTION.
2968 else if ((HeadMatches("CREATE", "TRIGGER") ||
2969 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2970 TailMatches("EXECUTE"))
2972 if (pset
.sversion
>= 110000)
2973 COMPLETE_WITH("FUNCTION");
2975 COMPLETE_WITH("PROCEDURE");
2977 else if ((HeadMatches("CREATE", "TRIGGER") ||
2978 HeadMatches("CREATE", "OR", "REPLACE", "TRIGGER")) &&
2979 TailMatches("EXECUTE", "FUNCTION|PROCEDURE"))
2980 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_functions
, NULL
);
2982 /* CREATE ROLE,USER,GROUP <name> */
2983 else if (Matches("CREATE", "ROLE|GROUP|USER", MatchAny
) &&
2984 !TailMatches("USER", "MAPPING"))
2985 COMPLETE_WITH("ADMIN", "BYPASSRLS", "CONNECTION LIMIT", "CREATEDB",
2986 "CREATEROLE", "ENCRYPTED PASSWORD", "IN", "INHERIT",
2987 "LOGIN", "NOBYPASSRLS",
2988 "NOCREATEDB", "NOCREATEROLE", "NOINHERIT",
2989 "NOLOGIN", "NOREPLICATION", "NOSUPERUSER", "PASSWORD",
2990 "REPLICATION", "ROLE", "SUPERUSER", "SYSID",
2991 "VALID UNTIL", "WITH");
2993 /* CREATE ROLE,USER,GROUP <name> WITH */
2994 else if (Matches("CREATE", "ROLE|GROUP|USER", MatchAny
, "WITH"))
2995 /* Similar to the above, but don't complete "WITH" again. */
2996 COMPLETE_WITH("ADMIN", "BYPASSRLS", "CONNECTION LIMIT", "CREATEDB",
2997 "CREATEROLE", "ENCRYPTED PASSWORD", "IN", "INHERIT",
2998 "LOGIN", "NOBYPASSRLS",
2999 "NOCREATEDB", "NOCREATEROLE", "NOINHERIT",
3000 "NOLOGIN", "NOREPLICATION", "NOSUPERUSER", "PASSWORD",
3001 "REPLICATION", "ROLE", "SUPERUSER", "SYSID",
3004 /* complete CREATE ROLE,USER,GROUP <name> IN with ROLE,GROUP */
3005 else if (Matches("CREATE", "ROLE|USER|GROUP", MatchAny
, "IN"))
3006 COMPLETE_WITH("GROUP", "ROLE");
3009 else if (Matches("CREATE", "TYPE", MatchAny
))
3010 COMPLETE_WITH("(", "AS");
3011 else if (Matches("CREATE", "TYPE", MatchAny
, "AS"))
3012 COMPLETE_WITH("ENUM", "RANGE", "(");
3013 else if (HeadMatches("CREATE", "TYPE", MatchAny
, "AS", "("))
3015 if (TailMatches("(|*,", MatchAny
))
3016 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
3017 else if (TailMatches("(|*,", MatchAny
, MatchAnyExcept("*)")))
3018 COMPLETE_WITH("COLLATE", ",", ")");
3020 else if (Matches("CREATE", "TYPE", MatchAny
, "AS", "ENUM|RANGE"))
3022 else if (HeadMatches("CREATE", "TYPE", MatchAny
, "("))
3024 if (TailMatches("(|*,"))
3025 COMPLETE_WITH("INPUT", "OUTPUT", "RECEIVE", "SEND",
3026 "TYPMOD_IN", "TYPMOD_OUT", "ANALYZE", "SUBSCRIPT",
3027 "INTERNALLENGTH", "PASSEDBYVALUE", "ALIGNMENT",
3028 "STORAGE", "LIKE", "CATEGORY", "PREFERRED",
3029 "DEFAULT", "ELEMENT", "DELIMITER",
3031 else if (TailMatches("(*|*,", MatchAnyExcept("*=")))
3033 else if (TailMatches("=", MatchAnyExcept("*)")))
3034 COMPLETE_WITH(",", ")");
3036 else if (HeadMatches("CREATE", "TYPE", MatchAny
, "AS", "RANGE", "("))
3038 if (TailMatches("(|*,"))
3039 COMPLETE_WITH("SUBTYPE", "SUBTYPE_OPCLASS", "COLLATION",
3040 "CANONICAL", "SUBTYPE_DIFF",
3041 "MULTIRANGE_TYPE_NAME");
3042 else if (TailMatches("(*|*,", MatchAnyExcept("*=")))
3044 else if (TailMatches("=", MatchAnyExcept("*)")))
3045 COMPLETE_WITH(",", ")");
3048 /* CREATE VIEW --- is allowed inside CREATE SCHEMA, so use TailMatches */
3049 /* Complete CREATE [ OR REPLACE ] VIEW <name> with AS */
3050 else if (TailMatches("CREATE", "VIEW", MatchAny
) ||
3051 TailMatches("CREATE", "OR", "REPLACE", "VIEW", MatchAny
))
3052 COMPLETE_WITH("AS");
3053 /* Complete "CREATE [ OR REPLACE ] VIEW <sth> AS with "SELECT" */
3054 else if (TailMatches("CREATE", "VIEW", MatchAny
, "AS") ||
3055 TailMatches("CREATE", "OR", "REPLACE", "VIEW", MatchAny
, "AS"))
3056 COMPLETE_WITH("SELECT");
3058 /* CREATE MATERIALIZED VIEW */
3059 else if (Matches("CREATE", "MATERIALIZED"))
3060 COMPLETE_WITH("VIEW");
3061 /* Complete CREATE MATERIALIZED VIEW <name> with AS */
3062 else if (Matches("CREATE", "MATERIALIZED", "VIEW", MatchAny
))
3063 COMPLETE_WITH("AS");
3064 /* Complete "CREATE MATERIALIZED VIEW <sth> AS with "SELECT" */
3065 else if (Matches("CREATE", "MATERIALIZED", "VIEW", MatchAny
, "AS"))
3066 COMPLETE_WITH("SELECT");
3068 /* CREATE EVENT TRIGGER */
3069 else if (Matches("CREATE", "EVENT"))
3070 COMPLETE_WITH("TRIGGER");
3071 /* Complete CREATE EVENT TRIGGER <name> with ON */
3072 else if (Matches("CREATE", "EVENT", "TRIGGER", MatchAny
))
3073 COMPLETE_WITH("ON");
3074 /* Complete CREATE EVENT TRIGGER <name> ON with event_type */
3075 else if (Matches("CREATE", "EVENT", "TRIGGER", MatchAny
, "ON"))
3076 COMPLETE_WITH("ddl_command_start", "ddl_command_end", "sql_drop");
3079 * Complete CREATE EVENT TRIGGER <name> ON <event_type>. EXECUTE FUNCTION
3080 * is the recommended grammar instead of EXECUTE PROCEDURE in version 11
3083 else if (Matches("CREATE", "EVENT", "TRIGGER", MatchAny
, "ON", MatchAny
))
3085 if (pset
.sversion
>= 110000)
3086 COMPLETE_WITH("WHEN TAG IN (", "EXECUTE FUNCTION");
3088 COMPLETE_WITH("WHEN TAG IN (", "EXECUTE PROCEDURE");
3090 else if (HeadMatches("CREATE", "EVENT", "TRIGGER") &&
3091 TailMatches("WHEN|AND", MatchAny
, "IN", "(*)"))
3093 if (pset
.sversion
>= 110000)
3094 COMPLETE_WITH("EXECUTE FUNCTION");
3096 COMPLETE_WITH("EXECUTE PROCEDURE");
3098 else if (HeadMatches("CREATE", "EVENT", "TRIGGER") &&
3099 TailMatches("EXECUTE", "FUNCTION|PROCEDURE"))
3100 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_functions
, NULL
);
3103 else if (Matches("DEALLOCATE"))
3104 COMPLETE_WITH_QUERY(Query_for_list_of_prepared_statements
3105 " UNION SELECT 'ALL'");
3110 * Complete DECLARE <name> with one of BINARY, ASENSITIVE, INSENSITIVE,
3111 * SCROLL, NO SCROLL, and CURSOR.
3113 else if (Matches("DECLARE", MatchAny
))
3114 COMPLETE_WITH("BINARY", "ASENSITIVE", "INSENSITIVE", "SCROLL", "NO SCROLL",
3118 * Complete DECLARE ... <option> with other options. The PostgreSQL parser
3119 * allows DECLARE options to be specified in any order. But the
3120 * tab-completion follows the ordering of them that the SQL standard
3121 * provides, like the syntax of DECLARE command in the documentation
3124 else if (HeadMatches("DECLARE") && TailMatches("BINARY"))
3125 COMPLETE_WITH("ASENSITIVE", "INSENSITIVE", "SCROLL", "NO SCROLL", "CURSOR");
3126 else if (HeadMatches("DECLARE") && TailMatches("ASENSITIVE|INSENSITIVE"))
3127 COMPLETE_WITH("SCROLL", "NO SCROLL", "CURSOR");
3128 else if (HeadMatches("DECLARE") && TailMatches("SCROLL"))
3129 COMPLETE_WITH("CURSOR");
3130 /* Complete DECLARE ... [options] NO with SCROLL */
3131 else if (HeadMatches("DECLARE") && TailMatches("NO"))
3132 COMPLETE_WITH("SCROLL");
3135 * Complete DECLARE ... CURSOR with one of WITH HOLD, WITHOUT HOLD, and
3138 else if (HeadMatches("DECLARE") && TailMatches("CURSOR"))
3139 COMPLETE_WITH("WITH HOLD", "WITHOUT HOLD", "FOR");
3140 /* Complete DECLARE ... CURSOR WITH|WITHOUT with HOLD */
3141 else if (HeadMatches("DECLARE") && TailMatches("CURSOR", "WITH|WITHOUT"))
3142 COMPLETE_WITH("HOLD");
3143 /* Complete DECLARE ... CURSOR WITH|WITHOUT HOLD with FOR */
3144 else if (HeadMatches("DECLARE") && TailMatches("CURSOR", "WITH|WITHOUT", "HOLD"))
3145 COMPLETE_WITH("FOR");
3147 /* DELETE --- can be inside EXPLAIN, RULE, etc */
3148 /* Complete DELETE with "FROM" */
3149 else if (Matches("DELETE"))
3150 COMPLETE_WITH("FROM");
3151 /* Complete DELETE FROM with a list of tables */
3152 else if (TailMatches("DELETE", "FROM"))
3153 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_updatables
, NULL
);
3154 /* Complete DELETE FROM <table> */
3155 else if (TailMatches("DELETE", "FROM", MatchAny
))
3156 COMPLETE_WITH("USING", "WHERE");
3157 /* XXX: implement tab completion for DELETE ... USING */
3160 else if (Matches("DISCARD"))
3161 COMPLETE_WITH("ALL", "PLANS", "SEQUENCES", "TEMP");
3164 else if (Matches("DO"))
3165 COMPLETE_WITH("LANGUAGE");
3168 /* Complete DROP object with CASCADE / RESTRICT */
3169 else if (Matches("DROP",
3170 "COLLATION|CONVERSION|DOMAIN|EXTENSION|LANGUAGE|PUBLICATION|SCHEMA|SEQUENCE|SERVER|SUBSCRIPTION|STATISTICS|TABLE|TYPE|VIEW",
3172 Matches("DROP", "ACCESS", "METHOD", MatchAny
) ||
3173 (Matches("DROP", "AGGREGATE|FUNCTION|PROCEDURE|ROUTINE", MatchAny
, MatchAny
) &&
3174 ends_with(prev_wd
, ')')) ||
3175 Matches("DROP", "EVENT", "TRIGGER", MatchAny
) ||
3176 Matches("DROP", "FOREIGN", "DATA", "WRAPPER", MatchAny
) ||
3177 Matches("DROP", "FOREIGN", "TABLE", MatchAny
) ||
3178 Matches("DROP", "TEXT", "SEARCH", "CONFIGURATION|DICTIONARY|PARSER|TEMPLATE", MatchAny
))
3179 COMPLETE_WITH("CASCADE", "RESTRICT");
3181 /* help completing some of the variants */
3182 else if (Matches("DROP", "AGGREGATE|FUNCTION|PROCEDURE|ROUTINE", MatchAny
))
3184 else if (Matches("DROP", "AGGREGATE|FUNCTION|PROCEDURE|ROUTINE", MatchAny
, "("))
3185 COMPLETE_WITH_FUNCTION_ARG(prev2_wd
);
3186 else if (Matches("DROP", "FOREIGN"))
3187 COMPLETE_WITH("DATA WRAPPER", "TABLE");
3188 else if (Matches("DROP", "DATABASE", MatchAny
))
3189 COMPLETE_WITH("WITH (");
3190 else if (HeadMatches("DROP", "DATABASE") && (ends_with(prev_wd
, '(')))
3191 COMPLETE_WITH("FORCE");
3194 else if (Matches("DROP", "INDEX"))
3195 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
,
3196 " UNION SELECT 'CONCURRENTLY'");
3197 else if (Matches("DROP", "INDEX", "CONCURRENTLY"))
3198 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
, NULL
);
3199 else if (Matches("DROP", "INDEX", MatchAny
))
3200 COMPLETE_WITH("CASCADE", "RESTRICT");
3201 else if (Matches("DROP", "INDEX", "CONCURRENTLY", MatchAny
))
3202 COMPLETE_WITH("CASCADE", "RESTRICT");
3204 /* DROP MATERIALIZED VIEW */
3205 else if (Matches("DROP", "MATERIALIZED"))
3206 COMPLETE_WITH("VIEW");
3207 else if (Matches("DROP", "MATERIALIZED", "VIEW"))
3208 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_matviews
, NULL
);
3211 else if (Matches("DROP", "OWNED"))
3212 COMPLETE_WITH("BY");
3213 else if (Matches("DROP", "OWNED", "BY"))
3214 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
3216 /* DROP TEXT SEARCH */
3217 else if (Matches("DROP", "TEXT", "SEARCH"))
3218 COMPLETE_WITH("CONFIGURATION", "DICTIONARY", "PARSER", "TEMPLATE");
3221 else if (Matches("DROP", "TRIGGER", MatchAny
))
3222 COMPLETE_WITH("ON");
3223 else if (Matches("DROP", "TRIGGER", MatchAny
, "ON"))
3225 completion_info_charp
= prev2_wd
;
3226 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_trigger
);
3228 else if (Matches("DROP", "TRIGGER", MatchAny
, "ON", MatchAny
))
3229 COMPLETE_WITH("CASCADE", "RESTRICT");
3231 /* DROP ACCESS METHOD */
3232 else if (Matches("DROP", "ACCESS"))
3233 COMPLETE_WITH("METHOD");
3234 else if (Matches("DROP", "ACCESS", "METHOD"))
3235 COMPLETE_WITH_QUERY(Query_for_list_of_access_methods
);
3237 /* DROP EVENT TRIGGER */
3238 else if (Matches("DROP", "EVENT"))
3239 COMPLETE_WITH("TRIGGER");
3240 else if (Matches("DROP", "EVENT", "TRIGGER"))
3241 COMPLETE_WITH_QUERY(Query_for_list_of_event_triggers
);
3243 /* DROP POLICY <name> */
3244 else if (Matches("DROP", "POLICY"))
3245 COMPLETE_WITH_QUERY(Query_for_list_of_policies
);
3246 /* DROP POLICY <name> ON */
3247 else if (Matches("DROP", "POLICY", MatchAny
))
3248 COMPLETE_WITH("ON");
3249 /* DROP POLICY <name> ON <table> */
3250 else if (Matches("DROP", "POLICY", MatchAny
, "ON"))
3252 completion_info_charp
= prev2_wd
;
3253 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_policy
);
3257 else if (Matches("DROP", "RULE", MatchAny
))
3258 COMPLETE_WITH("ON");
3259 else if (Matches("DROP", "RULE", MatchAny
, "ON"))
3261 completion_info_charp
= prev2_wd
;
3262 COMPLETE_WITH_QUERY(Query_for_list_of_tables_for_rule
);
3264 else if (Matches("DROP", "RULE", MatchAny
, "ON", MatchAny
))
3265 COMPLETE_WITH("CASCADE", "RESTRICT");
3268 else if (Matches("EXECUTE"))
3269 COMPLETE_WITH_QUERY(Query_for_list_of_prepared_statements
);
3272 * EXPLAIN [ ( option [, ...] ) ] statement
3273 * EXPLAIN [ ANALYZE ] [ VERBOSE ] statement
3275 else if (Matches("EXPLAIN"))
3276 COMPLETE_WITH("SELECT", "INSERT INTO", "DELETE FROM", "UPDATE", "DECLARE",
3277 "EXECUTE", "ANALYZE", "VERBOSE");
3278 else if (HeadMatches("EXPLAIN", "(*") &&
3279 !HeadMatches("EXPLAIN", "(*)"))
3282 * This fires if we're in an unfinished parenthesized option list.
3283 * get_previous_words treats a completed parenthesized option list as
3284 * one word, so the above test is correct.
3286 if (ends_with(prev_wd
, '(') || ends_with(prev_wd
, ','))
3287 COMPLETE_WITH("ANALYZE", "VERBOSE", "COSTS", "SETTINGS",
3288 "BUFFERS", "WAL", "TIMING", "SUMMARY", "FORMAT");
3289 else if (TailMatches("ANALYZE|VERBOSE|COSTS|SETTINGS|BUFFERS|WAL|TIMING|SUMMARY"))
3290 COMPLETE_WITH("ON", "OFF");
3291 else if (TailMatches("FORMAT"))
3292 COMPLETE_WITH("TEXT", "XML", "JSON", "YAML");
3294 else if (Matches("EXPLAIN", "ANALYZE"))
3295 COMPLETE_WITH("SELECT", "INSERT INTO", "DELETE FROM", "UPDATE", "DECLARE",
3296 "EXECUTE", "VERBOSE");
3297 else if (Matches("EXPLAIN", "(*)") ||
3298 Matches("EXPLAIN", "VERBOSE") ||
3299 Matches("EXPLAIN", "ANALYZE", "VERBOSE"))
3300 COMPLETE_WITH("SELECT", "INSERT INTO", "DELETE FROM", "UPDATE", "DECLARE",
3306 * Complete FETCH with one of ABSOLUTE, BACKWARD, FORWARD, RELATIVE, ALL,
3307 * NEXT, PRIOR, FIRST, LAST, FROM, IN, and a list of cursors
3309 else if (Matches("FETCH|MOVE"))
3310 COMPLETE_WITH_QUERY(Query_for_list_of_cursors
3311 " UNION SELECT 'ABSOLUTE'"
3312 " UNION SELECT 'BACKWARD'"
3313 " UNION SELECT 'FORWARD'"
3314 " UNION SELECT 'RELATIVE'"
3315 " UNION SELECT 'ALL'"
3316 " UNION SELECT 'NEXT'"
3317 " UNION SELECT 'PRIOR'"
3318 " UNION SELECT 'FIRST'"
3319 " UNION SELECT 'LAST'"
3320 " UNION SELECT 'FROM'"
3321 " UNION SELECT 'IN'");
3324 * Complete FETCH BACKWARD or FORWARD with one of ALL, FROM, IN, and a
3327 else if (Matches("FETCH|MOVE", "BACKWARD|FORWARD"))
3328 COMPLETE_WITH_QUERY(Query_for_list_of_cursors
3329 " UNION SELECT 'ALL'"
3330 " UNION SELECT 'FROM'"
3331 " UNION SELECT 'IN'");
3334 * Complete FETCH <direction> with "FROM" or "IN". These are equivalent,
3335 * but we may as well tab-complete both: perhaps some users prefer one
3336 * variant or the other.
3338 else if (Matches("FETCH|MOVE", "ABSOLUTE|BACKWARD|FORWARD|RELATIVE",
3339 MatchAnyExcept("FROM|IN")) ||
3340 Matches("FETCH|MOVE", "ALL|NEXT|PRIOR|FIRST|LAST"))
3341 COMPLETE_WITH_QUERY(Query_for_list_of_cursors
3342 " UNION SELECT 'FROM'"
3343 " UNION SELECT 'IN'");
3344 /* Complete FETCH <direction> "FROM" or "IN" with a list of cursors */
3345 else if (HeadMatches("FETCH|MOVE") &&
3346 TailMatches("FROM|IN"))
3347 COMPLETE_WITH_QUERY(Query_for_list_of_cursors
);
3349 /* FOREIGN DATA WRAPPER */
3350 /* applies in ALTER/DROP FDW and in CREATE SERVER */
3351 else if (TailMatches("FOREIGN", "DATA", "WRAPPER") &&
3352 !TailMatches("CREATE", MatchAny
, MatchAny
, MatchAny
))
3353 COMPLETE_WITH_QUERY(Query_for_list_of_fdws
);
3354 /* applies in CREATE SERVER */
3355 else if (TailMatches("FOREIGN", "DATA", "WRAPPER", MatchAny
) &&
3356 HeadMatches("CREATE", "SERVER"))
3357 COMPLETE_WITH("OPTIONS");
3360 else if (TailMatches("FOREIGN", "TABLE") &&
3361 !TailMatches("CREATE", MatchAny
, MatchAny
))
3362 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_foreign_tables
, NULL
);
3364 /* FOREIGN SERVER */
3365 else if (TailMatches("FOREIGN", "SERVER"))
3366 COMPLETE_WITH_QUERY(Query_for_list_of_servers
);
3369 * GRANT and REVOKE are allowed inside CREATE SCHEMA and
3370 * ALTER DEFAULT PRIVILEGES, so use TailMatches
3372 /* Complete GRANT/REVOKE with a list of roles and privileges */
3373 else if (TailMatches("GRANT|REVOKE"))
3376 * With ALTER DEFAULT PRIVILEGES, restrict completion to grantable
3377 * privileges (can't grant roles)
3379 if (HeadMatches("ALTER", "DEFAULT", "PRIVILEGES"))
3380 COMPLETE_WITH("SELECT", "INSERT", "UPDATE",
3381 "DELETE", "TRUNCATE", "REFERENCES", "TRIGGER",
3382 "EXECUTE", "USAGE", "ALL");
3384 COMPLETE_WITH_QUERY(Query_for_list_of_roles
3385 " UNION SELECT 'SELECT'"
3386 " UNION SELECT 'INSERT'"
3387 " UNION SELECT 'UPDATE'"
3388 " UNION SELECT 'DELETE'"
3389 " UNION SELECT 'TRUNCATE'"
3390 " UNION SELECT 'REFERENCES'"
3391 " UNION SELECT 'TRIGGER'"
3392 " UNION SELECT 'CREATE'"
3393 " UNION SELECT 'CONNECT'"
3394 " UNION SELECT 'TEMPORARY'"
3395 " UNION SELECT 'EXECUTE'"
3396 " UNION SELECT 'USAGE'"
3397 " UNION SELECT 'ALL'");
3401 * Complete GRANT/REVOKE <privilege> with "ON", GRANT/REVOKE <role> with
3404 else if (TailMatches("GRANT|REVOKE", MatchAny
))
3406 if (TailMatches("SELECT|INSERT|UPDATE|DELETE|TRUNCATE|REFERENCES|TRIGGER|CREATE|CONNECT|TEMPORARY|TEMP|EXECUTE|USAGE|ALL"))
3407 COMPLETE_WITH("ON");
3408 else if (TailMatches("GRANT", MatchAny
))
3409 COMPLETE_WITH("TO");
3411 COMPLETE_WITH("FROM");
3415 * Complete GRANT/REVOKE <sth> ON with a list of appropriate relations.
3417 * Keywords like DATABASE, FUNCTION, LANGUAGE and SCHEMA added to query
3418 * result via UNION; seems to work intuitively.
3420 * Note: GRANT/REVOKE can get quite complex; tab-completion as implemented
3421 * here will only work if the privilege list contains exactly one
3424 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON"))
3427 * With ALTER DEFAULT PRIVILEGES, restrict completion to the kinds of
3428 * objects supported.
3430 if (HeadMatches("ALTER", "DEFAULT", "PRIVILEGES"))
3431 COMPLETE_WITH("TABLES", "SEQUENCES", "FUNCTIONS", "PROCEDURES", "ROUTINES", "TYPES", "SCHEMAS");
3433 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_grantables
,
3434 " UNION SELECT 'ALL FUNCTIONS IN SCHEMA'"
3435 " UNION SELECT 'ALL PROCEDURES IN SCHEMA'"
3436 " UNION SELECT 'ALL ROUTINES IN SCHEMA'"
3437 " UNION SELECT 'ALL SEQUENCES IN SCHEMA'"
3438 " UNION SELECT 'ALL TABLES IN SCHEMA'"
3439 " UNION SELECT 'DATABASE'"
3440 " UNION SELECT 'DOMAIN'"
3441 " UNION SELECT 'FOREIGN DATA WRAPPER'"
3442 " UNION SELECT 'FOREIGN SERVER'"
3443 " UNION SELECT 'FUNCTION'"
3444 " UNION SELECT 'LANGUAGE'"
3445 " UNION SELECT 'LARGE OBJECT'"
3446 " UNION SELECT 'PROCEDURE'"
3447 " UNION SELECT 'ROUTINE'"
3448 " UNION SELECT 'SCHEMA'"
3449 " UNION SELECT 'SEQUENCE'"
3450 " UNION SELECT 'TABLE'"
3451 " UNION SELECT 'TABLESPACE'"
3452 " UNION SELECT 'TYPE'");
3454 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON", "ALL"))
3455 COMPLETE_WITH("FUNCTIONS IN SCHEMA",
3456 "PROCEDURES IN SCHEMA",
3457 "ROUTINES IN SCHEMA",
3458 "SEQUENCES IN SCHEMA",
3459 "TABLES IN SCHEMA");
3460 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON", "FOREIGN"))
3461 COMPLETE_WITH("DATA WRAPPER", "SERVER");
3464 * Complete "GRANT/REVOKE * ON DATABASE/DOMAIN/..." with a list of
3465 * appropriate objects.
3467 * Complete "GRANT/REVOKE * ON *" with "TO/FROM".
3469 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON", MatchAny
))
3471 if (TailMatches("DATABASE"))
3472 COMPLETE_WITH_QUERY(Query_for_list_of_databases
);
3473 else if (TailMatches("DOMAIN"))
3474 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_domains
, NULL
);
3475 else if (TailMatches("FUNCTION"))
3476 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_functions
, NULL
);
3477 else if (TailMatches("LANGUAGE"))
3478 COMPLETE_WITH_QUERY(Query_for_list_of_languages
);
3479 else if (TailMatches("PROCEDURE"))
3480 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_procedures
, NULL
);
3481 else if (TailMatches("ROUTINE"))
3482 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_routines
, NULL
);
3483 else if (TailMatches("SCHEMA"))
3484 COMPLETE_WITH_QUERY(Query_for_list_of_schemas
);
3485 else if (TailMatches("SEQUENCE"))
3486 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_sequences
, NULL
);
3487 else if (TailMatches("TABLE"))
3488 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_grantables
, NULL
);
3489 else if (TailMatches("TABLESPACE"))
3490 COMPLETE_WITH_QUERY(Query_for_list_of_tablespaces
);
3491 else if (TailMatches("TYPE"))
3492 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
3493 else if (TailMatches("GRANT", MatchAny
, MatchAny
, MatchAny
))
3494 COMPLETE_WITH("TO");
3496 COMPLETE_WITH("FROM");
3500 * Complete "GRANT/REVOKE ... TO/FROM" with username, PUBLIC,
3501 * CURRENT_ROLE, CURRENT_USER, or SESSION_USER.
3503 else if ((HeadMatches("GRANT") && TailMatches("TO")) ||
3504 (HeadMatches("REVOKE") && TailMatches("FROM")))
3505 COMPLETE_WITH_QUERY(Query_for_list_of_grant_roles
);
3506 /* Complete "ALTER DEFAULT PRIVILEGES ... GRANT/REVOKE ... TO/FROM */
3507 else if (HeadMatches("ALTER", "DEFAULT", "PRIVILEGES") && TailMatches("TO|FROM"))
3508 COMPLETE_WITH_QUERY(Query_for_list_of_grant_roles
);
3509 /* Complete "GRANT/REVOKE ... ON * *" with TO/FROM */
3510 else if (HeadMatches("GRANT") && TailMatches("ON", MatchAny
, MatchAny
))
3511 COMPLETE_WITH("TO");
3512 else if (HeadMatches("REVOKE") && TailMatches("ON", MatchAny
, MatchAny
))
3513 COMPLETE_WITH("FROM");
3515 /* Complete "GRANT/REVOKE * ON ALL * IN SCHEMA *" with TO/FROM */
3516 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON", "ALL", MatchAny
, "IN", "SCHEMA", MatchAny
))
3518 if (TailMatches("GRANT", MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
))
3519 COMPLETE_WITH("TO");
3521 COMPLETE_WITH("FROM");
3524 /* Complete "GRANT/REVOKE * ON FOREIGN DATA WRAPPER *" with TO/FROM */
3525 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON", "FOREIGN", "DATA", "WRAPPER", MatchAny
))
3527 if (TailMatches("GRANT", MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
))
3528 COMPLETE_WITH("TO");
3530 COMPLETE_WITH("FROM");
3533 /* Complete "GRANT/REVOKE * ON FOREIGN SERVER *" with TO/FROM */
3534 else if (TailMatches("GRANT|REVOKE", MatchAny
, "ON", "FOREIGN", "SERVER", MatchAny
))
3536 if (TailMatches("GRANT", MatchAny
, MatchAny
, MatchAny
, MatchAny
, MatchAny
))
3537 COMPLETE_WITH("TO");
3539 COMPLETE_WITH("FROM");
3543 else if (TailMatches("FROM", MatchAny
, "GROUP"))
3544 COMPLETE_WITH("BY");
3546 /* IMPORT FOREIGN SCHEMA */
3547 else if (Matches("IMPORT"))
3548 COMPLETE_WITH("FOREIGN SCHEMA");
3549 else if (Matches("IMPORT", "FOREIGN"))
3550 COMPLETE_WITH("SCHEMA");
3551 else if (Matches("IMPORT", "FOREIGN", "SCHEMA", MatchAny
))
3552 COMPLETE_WITH("EXCEPT (", "FROM SERVER", "LIMIT TO (");
3553 else if (TailMatches("LIMIT", "TO", "(*)") ||
3554 TailMatches("EXCEPT", "(*)"))
3555 COMPLETE_WITH("FROM SERVER");
3556 else if (TailMatches("FROM", "SERVER", MatchAny
))
3557 COMPLETE_WITH("INTO");
3558 else if (TailMatches("FROM", "SERVER", MatchAny
, "INTO"))
3559 COMPLETE_WITH_QUERY(Query_for_list_of_schemas
);
3560 else if (TailMatches("FROM", "SERVER", MatchAny
, "INTO", MatchAny
))
3561 COMPLETE_WITH("OPTIONS (");
3563 /* INSERT --- can be inside EXPLAIN, RULE, etc */
3564 /* Complete INSERT with "INTO" */
3565 else if (TailMatches("INSERT"))
3566 COMPLETE_WITH("INTO");
3567 /* Complete INSERT INTO with table names */
3568 else if (TailMatches("INSERT", "INTO"))
3569 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_updatables
, NULL
);
3570 /* Complete "INSERT INTO <table> (" with attribute names */
3571 else if (TailMatches("INSERT", "INTO", MatchAny
, "("))
3572 COMPLETE_WITH_ATTR(prev2_wd
, "");
3575 * Complete INSERT INTO <table> with "(" or "VALUES" or "SELECT" or
3576 * "TABLE" or "DEFAULT VALUES" or "OVERRIDING"
3578 else if (TailMatches("INSERT", "INTO", MatchAny
))
3579 COMPLETE_WITH("(", "DEFAULT VALUES", "SELECT", "TABLE", "VALUES", "OVERRIDING");
3582 * Complete INSERT INTO <table> (attribs) with "VALUES" or "SELECT" or
3583 * "TABLE" or "OVERRIDING"
3585 else if (TailMatches("INSERT", "INTO", MatchAny
, MatchAny
) &&
3586 ends_with(prev_wd
, ')'))
3587 COMPLETE_WITH("SELECT", "TABLE", "VALUES", "OVERRIDING");
3589 /* Complete OVERRIDING */
3590 else if (TailMatches("OVERRIDING"))
3591 COMPLETE_WITH("SYSTEM VALUE", "USER VALUE");
3593 /* Complete after OVERRIDING clause */
3594 else if (TailMatches("OVERRIDING", MatchAny
, "VALUE"))
3595 COMPLETE_WITH("SELECT", "TABLE", "VALUES");
3597 /* Insert an open parenthesis after "VALUES" */
3598 else if (TailMatches("VALUES") && !TailMatches("DEFAULT", "VALUES"))
3602 /* Complete LOCK [TABLE] with a list of tables */
3603 else if (Matches("LOCK"))
3604 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
,
3605 " UNION SELECT 'TABLE'");
3606 else if (Matches("LOCK", "TABLE"))
3607 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, "");
3609 /* For the following, handle the case of a single table only for now */
3611 /* Complete LOCK [TABLE] <table> with "IN" */
3612 else if (Matches("LOCK", MatchAnyExcept("TABLE")) ||
3613 Matches("LOCK", "TABLE", MatchAny
))
3614 COMPLETE_WITH("IN");
3616 /* Complete LOCK [TABLE] <table> IN with a lock mode */
3617 else if (Matches("LOCK", MatchAny
, "IN") ||
3618 Matches("LOCK", "TABLE", MatchAny
, "IN"))
3619 COMPLETE_WITH("ACCESS SHARE MODE",
3620 "ROW SHARE MODE", "ROW EXCLUSIVE MODE",
3621 "SHARE UPDATE EXCLUSIVE MODE", "SHARE MODE",
3622 "SHARE ROW EXCLUSIVE MODE",
3623 "EXCLUSIVE MODE", "ACCESS EXCLUSIVE MODE");
3625 /* Complete LOCK [TABLE] <table> IN ACCESS|ROW with rest of lock mode */
3626 else if (Matches("LOCK", MatchAny
, "IN", "ACCESS|ROW") ||
3627 Matches("LOCK", "TABLE", MatchAny
, "IN", "ACCESS|ROW"))
3628 COMPLETE_WITH("EXCLUSIVE MODE", "SHARE MODE");
3630 /* Complete LOCK [TABLE] <table> IN SHARE with rest of lock mode */
3631 else if (Matches("LOCK", MatchAny
, "IN", "SHARE") ||
3632 Matches("LOCK", "TABLE", MatchAny
, "IN", "SHARE"))
3633 COMPLETE_WITH("MODE", "ROW EXCLUSIVE MODE",
3634 "UPDATE EXCLUSIVE MODE");
3636 /* NOTIFY --- can be inside EXPLAIN, RULE, etc */
3637 else if (TailMatches("NOTIFY"))
3638 COMPLETE_WITH_QUERY("SELECT pg_catalog.quote_ident(channel) FROM pg_catalog.pg_listening_channels() AS channel WHERE substring(pg_catalog.quote_ident(channel),1,%d)='%s'");
3641 else if (TailMatches("OPTIONS"))
3644 /* OWNER TO - complete with available roles */
3645 else if (TailMatches("OWNER", "TO"))
3646 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
3649 else if (TailMatches("FROM", MatchAny
, "ORDER"))
3650 COMPLETE_WITH("BY");
3651 else if (TailMatches("FROM", MatchAny
, "ORDER", "BY"))
3652 COMPLETE_WITH_ATTR(prev3_wd
, "");
3655 else if (Matches("PREPARE", MatchAny
, "AS"))
3656 COMPLETE_WITH("SELECT", "UPDATE", "INSERT INTO", "DELETE FROM");
3659 * PREPARE TRANSACTION is missing on purpose. It's intended for transaction
3660 * managers, not for manual use in interactive sessions.
3663 /* REASSIGN OWNED BY xxx TO yyy */
3664 else if (Matches("REASSIGN"))
3665 COMPLETE_WITH("OWNED BY");
3666 else if (Matches("REASSIGN", "OWNED"))
3667 COMPLETE_WITH("BY");
3668 else if (Matches("REASSIGN", "OWNED", "BY"))
3669 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
3670 else if (Matches("REASSIGN", "OWNED", "BY", MatchAny
))
3671 COMPLETE_WITH("TO");
3672 else if (Matches("REASSIGN", "OWNED", "BY", MatchAny
, "TO"))
3673 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
3675 /* REFRESH MATERIALIZED VIEW */
3676 else if (Matches("REFRESH"))
3677 COMPLETE_WITH("MATERIALIZED VIEW");
3678 else if (Matches("REFRESH", "MATERIALIZED"))
3679 COMPLETE_WITH("VIEW");
3680 else if (Matches("REFRESH", "MATERIALIZED", "VIEW"))
3681 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_matviews
,
3682 " UNION SELECT 'CONCURRENTLY'");
3683 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", "CONCURRENTLY"))
3684 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_matviews
, NULL
);
3685 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", MatchAny
))
3686 COMPLETE_WITH("WITH");
3687 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", "CONCURRENTLY", MatchAny
))
3688 COMPLETE_WITH("WITH");
3689 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", MatchAny
, "WITH"))
3690 COMPLETE_WITH("NO DATA", "DATA");
3691 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", "CONCURRENTLY", MatchAny
, "WITH"))
3692 COMPLETE_WITH("NO DATA", "DATA");
3693 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", MatchAny
, "WITH", "NO"))
3694 COMPLETE_WITH("DATA");
3695 else if (Matches("REFRESH", "MATERIALIZED", "VIEW", "CONCURRENTLY", MatchAny
, "WITH", "NO"))
3696 COMPLETE_WITH("DATA");
3699 else if (Matches("REINDEX") ||
3700 Matches("REINDEX", "(*)"))
3701 COMPLETE_WITH("TABLE", "INDEX", "SYSTEM", "SCHEMA", "DATABASE");
3702 else if (Matches("REINDEX", "TABLE") ||
3703 Matches("REINDEX", "(*)", "TABLE"))
3704 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexables
,
3705 " UNION SELECT 'CONCURRENTLY'");
3706 else if (Matches("REINDEX", "INDEX") ||
3707 Matches("REINDEX", "(*)", "INDEX"))
3708 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
,
3709 " UNION SELECT 'CONCURRENTLY'");
3710 else if (Matches("REINDEX", "SCHEMA") ||
3711 Matches("REINDEX", "(*)", "SCHEMA"))
3712 COMPLETE_WITH_QUERY(Query_for_list_of_schemas
3713 " UNION SELECT 'CONCURRENTLY'");
3714 else if (Matches("REINDEX", "SYSTEM|DATABASE") ||
3715 Matches("REINDEX", "(*)", "SYSTEM|DATABASE"))
3716 COMPLETE_WITH_QUERY(Query_for_list_of_databases
3717 " UNION SELECT 'CONCURRENTLY'");
3718 else if (Matches("REINDEX", "TABLE", "CONCURRENTLY") ||
3719 Matches("REINDEX", "(*)", "TABLE", "CONCURRENTLY"))
3720 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexables
, NULL
);
3721 else if (Matches("REINDEX", "INDEX", "CONCURRENTLY") ||
3722 Matches("REINDEX", "(*)", "INDEX", "CONCURRENTLY"))
3723 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
, NULL
);
3724 else if (Matches("REINDEX", "SCHEMA", "CONCURRENTLY") ||
3725 Matches("REINDEX", "(*)", "SCHEMA", "CONCURRENTLY"))
3726 COMPLETE_WITH_QUERY(Query_for_list_of_schemas
);
3727 else if (Matches("REINDEX", "SYSTEM|DATABASE", "CONCURRENTLY") ||
3728 Matches("REINDEX", "(*)", "SYSTEM|DATABASE", "CONCURRENTLY"))
3729 COMPLETE_WITH_QUERY(Query_for_list_of_databases
);
3730 else if (HeadMatches("REINDEX", "(*") &&
3731 !HeadMatches("REINDEX", "(*)"))
3734 * This fires if we're in an unfinished parenthesized option list.
3735 * get_previous_words treats a completed parenthesized option list as
3736 * one word, so the above test is correct.
3738 if (ends_with(prev_wd
, '(') || ends_with(prev_wd
, ','))
3739 COMPLETE_WITH("CONCURRENTLY", "TABLESPACE", "VERBOSE");
3740 else if (TailMatches("TABLESPACE"))
3741 COMPLETE_WITH_QUERY(Query_for_list_of_tablespaces
);
3744 /* SECURITY LABEL */
3745 else if (Matches("SECURITY"))
3746 COMPLETE_WITH("LABEL");
3747 else if (Matches("SECURITY", "LABEL"))
3748 COMPLETE_WITH("ON", "FOR");
3749 else if (Matches("SECURITY", "LABEL", "FOR", MatchAny
))
3750 COMPLETE_WITH("ON");
3751 else if (Matches("SECURITY", "LABEL", "ON") ||
3752 Matches("SECURITY", "LABEL", "FOR", MatchAny
, "ON"))
3753 COMPLETE_WITH("TABLE", "COLUMN", "AGGREGATE", "DATABASE", "DOMAIN",
3754 "EVENT TRIGGER", "FOREIGN TABLE", "FUNCTION",
3755 "LARGE OBJECT", "MATERIALIZED VIEW", "LANGUAGE",
3756 "PUBLICATION", "PROCEDURE", "ROLE", "ROUTINE", "SCHEMA",
3757 "SEQUENCE", "SUBSCRIPTION", "TABLESPACE", "TYPE", "VIEW");
3758 else if (Matches("SECURITY", "LABEL", "ON", MatchAny
, MatchAny
))
3759 COMPLETE_WITH("IS");
3764 /* SET, RESET, SHOW */
3765 /* Complete with a variable name */
3766 else if (TailMatches("SET|RESET") && !TailMatches("UPDATE", MatchAny
, "SET"))
3767 COMPLETE_WITH_QUERY(Query_for_list_of_set_vars
);
3768 else if (Matches("SHOW"))
3769 COMPLETE_WITH_QUERY(Query_for_list_of_show_vars
);
3770 /* Complete "SET TRANSACTION" */
3771 else if (Matches("SET", "TRANSACTION"))
3772 COMPLETE_WITH("SNAPSHOT", "ISOLATION LEVEL", "READ", "DEFERRABLE", "NOT DEFERRABLE");
3773 else if (Matches("BEGIN|START", "TRANSACTION") ||
3774 Matches("BEGIN", "WORK") ||
3776 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION"))
3777 COMPLETE_WITH("ISOLATION LEVEL", "READ", "DEFERRABLE", "NOT DEFERRABLE");
3778 else if (Matches("SET|BEGIN|START", "TRANSACTION|WORK", "NOT") ||
3779 Matches("BEGIN", "NOT") ||
3780 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION", "NOT"))
3781 COMPLETE_WITH("DEFERRABLE");
3782 else if (Matches("SET|BEGIN|START", "TRANSACTION|WORK", "ISOLATION") ||
3783 Matches("BEGIN", "ISOLATION") ||
3784 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION", "ISOLATION"))
3785 COMPLETE_WITH("LEVEL");
3786 else if (Matches("SET|BEGIN|START", "TRANSACTION|WORK", "ISOLATION", "LEVEL") ||
3787 Matches("BEGIN", "ISOLATION", "LEVEL") ||
3788 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION", "ISOLATION", "LEVEL"))
3789 COMPLETE_WITH("READ", "REPEATABLE READ", "SERIALIZABLE");
3790 else if (Matches("SET|BEGIN|START", "TRANSACTION|WORK", "ISOLATION", "LEVEL", "READ") ||
3791 Matches("BEGIN", "ISOLATION", "LEVEL", "READ") ||
3792 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION", "ISOLATION", "LEVEL", "READ"))
3793 COMPLETE_WITH("UNCOMMITTED", "COMMITTED");
3794 else if (Matches("SET|BEGIN|START", "TRANSACTION|WORK", "ISOLATION", "LEVEL", "REPEATABLE") ||
3795 Matches("BEGIN", "ISOLATION", "LEVEL", "REPEATABLE") ||
3796 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION", "ISOLATION", "LEVEL", "REPEATABLE"))
3797 COMPLETE_WITH("READ");
3798 else if (Matches("SET|BEGIN|START", "TRANSACTION|WORK", "READ") ||
3799 Matches("BEGIN", "READ") ||
3800 Matches("SET", "SESSION", "CHARACTERISTICS", "AS", "TRANSACTION", "READ"))
3801 COMPLETE_WITH("ONLY", "WRITE");
3802 /* SET CONSTRAINTS */
3803 else if (Matches("SET", "CONSTRAINTS"))
3804 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_constraints_with_schema
, "UNION SELECT 'ALL'");
3805 /* Complete SET CONSTRAINTS <foo> with DEFERRED|IMMEDIATE */
3806 else if (Matches("SET", "CONSTRAINTS", MatchAny
))
3807 COMPLETE_WITH("DEFERRED", "IMMEDIATE");
3808 /* Complete SET ROLE */
3809 else if (Matches("SET", "ROLE"))
3810 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
3811 /* Complete SET SESSION with AUTHORIZATION or CHARACTERISTICS... */
3812 else if (Matches("SET", "SESSION"))
3813 COMPLETE_WITH("AUTHORIZATION", "CHARACTERISTICS AS TRANSACTION");
3814 /* Complete SET SESSION AUTHORIZATION with username */
3815 else if (Matches("SET", "SESSION", "AUTHORIZATION"))
3816 COMPLETE_WITH_QUERY(Query_for_list_of_roles
" UNION SELECT 'DEFAULT'");
3817 /* Complete RESET SESSION with AUTHORIZATION */
3818 else if (Matches("RESET", "SESSION"))
3819 COMPLETE_WITH("AUTHORIZATION");
3820 /* Complete SET <var> with "TO" */
3821 else if (Matches("SET", MatchAny
))
3822 COMPLETE_WITH("TO");
3825 * Complete ALTER DATABASE|FUNCTION||PROCEDURE|ROLE|ROUTINE|USER ... SET
3828 else if (HeadMatches("ALTER", "DATABASE|FUNCTION|PROCEDURE|ROLE|ROUTINE|USER") &&
3829 TailMatches("SET", MatchAny
))
3830 COMPLETE_WITH("FROM CURRENT", "TO");
3833 * Suggest possible variable values in SET variable TO|=, along with the
3834 * preceding ALTER syntaxes.
3836 else if (TailMatches("SET", MatchAny
, "TO|=") &&
3837 !TailMatches("UPDATE", MatchAny
, "SET", MatchAny
, "TO|="))
3839 /* special cased code for individual GUCs */
3840 if (TailMatches("DateStyle", "TO|="))
3841 COMPLETE_WITH("ISO", "SQL", "Postgres", "German",
3842 "YMD", "DMY", "MDY",
3843 "US", "European", "NonEuropean",
3845 else if (TailMatches("search_path", "TO|="))
3846 COMPLETE_WITH_QUERY(Query_for_list_of_schemas
3847 " AND nspname not like 'pg\\_toast%%' "
3848 " AND nspname not like 'pg\\_temp%%' "
3849 " UNION SELECT 'DEFAULT' ");
3852 /* generic, type based, GUC support */
3853 char *guctype
= get_guctype(prev2_wd
);
3856 * Note: if we don't recognize the GUC name, it's important to not
3857 * offer any completions, as most likely we've misinterpreted the
3858 * context and this isn't a GUC-setting command at all.
3862 if (strcmp(guctype
, "enum") == 0)
3864 char querybuf
[1024];
3866 snprintf(querybuf
, sizeof(querybuf
),
3867 Query_for_enum
, prev2_wd
);
3868 COMPLETE_WITH_QUERY(querybuf
);
3870 else if (strcmp(guctype
, "bool") == 0)
3871 COMPLETE_WITH("on", "off", "true", "false", "yes", "no",
3872 "1", "0", "DEFAULT");
3874 COMPLETE_WITH("DEFAULT");
3881 /* START TRANSACTION */
3882 else if (Matches("START"))
3883 COMPLETE_WITH("TRANSACTION");
3885 /* TABLE, but not TABLE embedded in other commands */
3886 else if (Matches("TABLE"))
3887 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_selectables
, NULL
);
3890 else if (TailMatches("TABLESAMPLE"))
3891 COMPLETE_WITH_QUERY(Query_for_list_of_tablesample_methods
);
3892 else if (TailMatches("TABLESAMPLE", MatchAny
))
3896 else if (Matches("TRUNCATE"))
3897 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_truncatables
,
3898 " UNION SELECT 'TABLE'"
3899 " UNION SELECT 'ONLY'");
3900 else if (Matches("TRUNCATE", "TABLE"))
3901 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_truncatables
,
3902 " UNION SELECT 'ONLY'");
3903 else if (HeadMatches("TRUNCATE") && TailMatches("ONLY"))
3904 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_truncatables
, NULL
);
3905 else if (Matches("TRUNCATE", MatchAny
) ||
3906 Matches("TRUNCATE", "TABLE|ONLY", MatchAny
) ||
3907 Matches("TRUNCATE", "TABLE", "ONLY", MatchAny
))
3908 COMPLETE_WITH("RESTART IDENTITY", "CONTINUE IDENTITY", "CASCADE", "RESTRICT");
3909 else if (HeadMatches("TRUNCATE") && TailMatches("IDENTITY"))
3910 COMPLETE_WITH("CASCADE", "RESTRICT");
3913 else if (Matches("UNLISTEN"))
3914 COMPLETE_WITH_QUERY("SELECT pg_catalog.quote_ident(channel) FROM pg_catalog.pg_listening_channels() AS channel WHERE substring(pg_catalog.quote_ident(channel),1,%d)='%s' UNION SELECT '*'");
3916 /* UPDATE --- can be inside EXPLAIN, RULE, etc */
3917 /* If prev. word is UPDATE suggest a list of tables */
3918 else if (TailMatches("UPDATE"))
3919 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_updatables
, NULL
);
3920 /* Complete UPDATE <table> with "SET" */
3921 else if (TailMatches("UPDATE", MatchAny
))
3922 COMPLETE_WITH("SET");
3923 /* Complete UPDATE <table> SET with list of attributes */
3924 else if (TailMatches("UPDATE", MatchAny
, "SET"))
3925 COMPLETE_WITH_ATTR(prev2_wd
, "");
3926 /* UPDATE <table> SET <attr> = */
3927 else if (TailMatches("UPDATE", MatchAny
, "SET", MatchAnyExcept("*=")))
3931 else if (Matches("ALTER|CREATE|DROP", "USER", "MAPPING"))
3932 COMPLETE_WITH("FOR");
3933 else if (Matches("CREATE", "USER", "MAPPING", "FOR"))
3934 COMPLETE_WITH_QUERY(Query_for_list_of_roles
3935 " UNION SELECT 'CURRENT_ROLE'"
3936 " UNION SELECT 'CURRENT_USER'"
3937 " UNION SELECT 'PUBLIC'"
3938 " UNION SELECT 'USER'");
3939 else if (Matches("ALTER|DROP", "USER", "MAPPING", "FOR"))
3940 COMPLETE_WITH_QUERY(Query_for_list_of_user_mappings
);
3941 else if (Matches("CREATE|ALTER|DROP", "USER", "MAPPING", "FOR", MatchAny
))
3942 COMPLETE_WITH("SERVER");
3943 else if (Matches("CREATE|ALTER", "USER", "MAPPING", "FOR", MatchAny
, "SERVER", MatchAny
))
3944 COMPLETE_WITH("OPTIONS");
3947 * VACUUM [ ( option [, ...] ) ] [ table_and_columns [, ...] ]
3948 * VACUUM [ FULL ] [ FREEZE ] [ VERBOSE ] [ ANALYZE ] [ table_and_columns [, ...] ]
3950 else if (Matches("VACUUM"))
3951 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_vacuumables
,
3952 " UNION SELECT 'FULL'"
3953 " UNION SELECT 'FREEZE'"
3954 " UNION SELECT 'ANALYZE'"
3955 " UNION SELECT 'VERBOSE'");
3956 else if (Matches("VACUUM", "FULL"))
3957 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_vacuumables
,
3958 " UNION SELECT 'FREEZE'"
3959 " UNION SELECT 'ANALYZE'"
3960 " UNION SELECT 'VERBOSE'");
3961 else if (Matches("VACUUM", "FREEZE") ||
3962 Matches("VACUUM", "FULL", "FREEZE"))
3963 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_vacuumables
,
3964 " UNION SELECT 'VERBOSE'"
3965 " UNION SELECT 'ANALYZE'");
3966 else if (Matches("VACUUM", "VERBOSE") ||
3967 Matches("VACUUM", "FULL|FREEZE", "VERBOSE") ||
3968 Matches("VACUUM", "FULL", "FREEZE", "VERBOSE"))
3969 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_vacuumables
,
3970 " UNION SELECT 'ANALYZE'");
3971 else if (HeadMatches("VACUUM", "(*") &&
3972 !HeadMatches("VACUUM", "(*)"))
3975 * This fires if we're in an unfinished parenthesized option list.
3976 * get_previous_words treats a completed parenthesized option list as
3977 * one word, so the above test is correct.
3979 if (ends_with(prev_wd
, '(') || ends_with(prev_wd
, ','))
3980 COMPLETE_WITH("FULL", "FREEZE", "ANALYZE", "VERBOSE",
3981 "DISABLE_PAGE_SKIPPING", "SKIP_LOCKED",
3982 "INDEX_CLEANUP", "PROCESS_TOAST",
3983 "TRUNCATE", "PARALLEL");
3984 else if (TailMatches("FULL|FREEZE|ANALYZE|VERBOSE|DISABLE_PAGE_SKIPPING|SKIP_LOCKED|PROCESS_TOAST|TRUNCATE"))
3985 COMPLETE_WITH("ON", "OFF");
3986 else if (TailMatches("INDEX_CLEANUP"))
3987 COMPLETE_WITH("AUTO", "ON", "OFF");
3989 else if (HeadMatches("VACUUM") && TailMatches("("))
3990 /* "VACUUM (" should be caught above, so assume we want columns */
3991 COMPLETE_WITH_ATTR(prev2_wd
, "");
3992 else if (HeadMatches("VACUUM"))
3993 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_vacuumables
, NULL
);
3995 /* WITH [RECURSIVE] */
3998 * Only match when WITH is the first word, as WITH may appear in many
4001 else if (Matches("WITH"))
4002 COMPLETE_WITH("RECURSIVE");
4005 /* Simple case of the word before the where being the table name */
4006 else if (TailMatches(MatchAny
, "WHERE"))
4007 COMPLETE_WITH_ATTR(prev2_wd
, "");
4010 /* TODO: also include SRF ? */
4011 else if (TailMatches("FROM") && !Matches("COPY|\\copy", MatchAny
, "FROM"))
4012 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_selectables
, NULL
);
4015 else if (TailMatches("JOIN"))
4016 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_selectables
, NULL
);
4018 /* Backslash commands */
4019 /* TODO: \dc \dd \dl */
4020 else if (TailMatchesCS("\\?"))
4021 COMPLETE_WITH_CS("commands", "options", "variables");
4022 else if (TailMatchesCS("\\connect|\\c"))
4024 if (!recognized_connection_string(text
))
4025 COMPLETE_WITH_QUERY(Query_for_list_of_databases
);
4027 else if (TailMatchesCS("\\connect|\\c", MatchAny
))
4029 if (!recognized_connection_string(prev_wd
))
4030 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
4032 else if (TailMatchesCS("\\da*"))
4033 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_aggregates
, NULL
);
4034 else if (TailMatchesCS("\\dAc*", MatchAny
) ||
4035 TailMatchesCS("\\dAf*", MatchAny
))
4036 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
4037 else if (TailMatchesCS("\\dAo*", MatchAny
) ||
4038 TailMatchesCS("\\dAp*", MatchAny
))
4039 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_operator_families
, NULL
);
4040 else if (TailMatchesCS("\\dA*"))
4041 COMPLETE_WITH_QUERY(Query_for_list_of_access_methods
);
4042 else if (TailMatchesCS("\\db*"))
4043 COMPLETE_WITH_QUERY(Query_for_list_of_tablespaces
);
4044 else if (TailMatchesCS("\\dD*"))
4045 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_domains
, NULL
);
4046 else if (TailMatchesCS("\\des*"))
4047 COMPLETE_WITH_QUERY(Query_for_list_of_servers
);
4048 else if (TailMatchesCS("\\deu*"))
4049 COMPLETE_WITH_QUERY(Query_for_list_of_user_mappings
);
4050 else if (TailMatchesCS("\\dew*"))
4051 COMPLETE_WITH_QUERY(Query_for_list_of_fdws
);
4052 else if (TailMatchesCS("\\df*"))
4053 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_functions
, NULL
);
4054 else if (HeadMatchesCS("\\df*"))
4055 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
4057 else if (TailMatchesCS("\\dFd*"))
4058 COMPLETE_WITH_QUERY(Query_for_list_of_ts_dictionaries
);
4059 else if (TailMatchesCS("\\dFp*"))
4060 COMPLETE_WITH_QUERY(Query_for_list_of_ts_parsers
);
4061 else if (TailMatchesCS("\\dFt*"))
4062 COMPLETE_WITH_QUERY(Query_for_list_of_ts_templates
);
4063 /* must be at end of \dF alternatives: */
4064 else if (TailMatchesCS("\\dF*"))
4065 COMPLETE_WITH_QUERY(Query_for_list_of_ts_configurations
);
4067 else if (TailMatchesCS("\\di*"))
4068 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_indexes
, NULL
);
4069 else if (TailMatchesCS("\\dL*"))
4070 COMPLETE_WITH_QUERY(Query_for_list_of_languages
);
4071 else if (TailMatchesCS("\\dn*"))
4072 COMPLETE_WITH_QUERY(Query_for_list_of_schemas
);
4073 /* no support for completing operators, but we can complete types: */
4074 else if (HeadMatchesCS("\\do*", MatchAny
))
4075 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
4076 else if (TailMatchesCS("\\dp") || TailMatchesCS("\\z"))
4077 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_grantables
, NULL
);
4078 else if (TailMatchesCS("\\dPi*"))
4079 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_partitioned_indexes
, NULL
);
4080 else if (TailMatchesCS("\\dPt*"))
4081 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_partitioned_tables
, NULL
);
4082 else if (TailMatchesCS("\\dP*"))
4083 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_partitioned_relations
, NULL
);
4084 else if (TailMatchesCS("\\ds*"))
4085 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_sequences
, NULL
);
4086 else if (TailMatchesCS("\\dt*"))
4087 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_tables
, NULL
);
4088 else if (TailMatchesCS("\\dT*"))
4089 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_datatypes
, NULL
);
4090 else if (TailMatchesCS("\\du*") || TailMatchesCS("\\dg*"))
4091 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
4092 else if (TailMatchesCS("\\dv*"))
4093 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_views
, NULL
);
4094 else if (TailMatchesCS("\\dx*"))
4095 COMPLETE_WITH_QUERY(Query_for_list_of_extensions
);
4096 else if (TailMatchesCS("\\dX*"))
4097 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_statistics
, NULL
);
4098 else if (TailMatchesCS("\\dm*"))
4099 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_matviews
, NULL
);
4100 else if (TailMatchesCS("\\dE*"))
4101 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_foreign_tables
, NULL
);
4102 else if (TailMatchesCS("\\dy*"))
4103 COMPLETE_WITH_QUERY(Query_for_list_of_event_triggers
);
4105 /* must be at end of \d alternatives: */
4106 else if (TailMatchesCS("\\d*"))
4107 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_relations
, NULL
);
4109 else if (TailMatchesCS("\\ef"))
4110 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_routines
, NULL
);
4111 else if (TailMatchesCS("\\ev"))
4112 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_views
, NULL
);
4114 else if (TailMatchesCS("\\encoding"))
4115 COMPLETE_WITH_QUERY(Query_for_list_of_encodings
);
4116 else if (TailMatchesCS("\\h|\\help"))
4117 COMPLETE_WITH_LIST(sql_commands
);
4118 else if (TailMatchesCS("\\h|\\help", MatchAny
))
4120 if (TailMatches("DROP"))
4121 matches
= rl_completion_matches(text
, drop_command_generator
);
4122 else if (TailMatches("ALTER"))
4123 matches
= rl_completion_matches(text
, alter_command_generator
);
4126 * CREATE is recognized by tail match elsewhere, so doesn't need to be
4130 else if (TailMatchesCS("\\h|\\help", MatchAny
, MatchAny
))
4132 if (TailMatches("CREATE|DROP", "ACCESS"))
4133 COMPLETE_WITH("METHOD");
4134 else if (TailMatches("ALTER", "DEFAULT"))
4135 COMPLETE_WITH("PRIVILEGES");
4136 else if (TailMatches("CREATE|ALTER|DROP", "EVENT"))
4137 COMPLETE_WITH("TRIGGER");
4138 else if (TailMatches("CREATE|ALTER|DROP", "FOREIGN"))
4139 COMPLETE_WITH("DATA WRAPPER", "TABLE");
4140 else if (TailMatches("ALTER", "LARGE"))
4141 COMPLETE_WITH("OBJECT");
4142 else if (TailMatches("CREATE|ALTER|DROP", "MATERIALIZED"))
4143 COMPLETE_WITH("VIEW");
4144 else if (TailMatches("CREATE|ALTER|DROP", "TEXT"))
4145 COMPLETE_WITH("SEARCH");
4146 else if (TailMatches("CREATE|ALTER|DROP", "USER"))
4147 COMPLETE_WITH("MAPPING FOR");
4149 else if (TailMatchesCS("\\h|\\help", MatchAny
, MatchAny
, MatchAny
))
4151 if (TailMatches("CREATE|ALTER|DROP", "FOREIGN", "DATA"))
4152 COMPLETE_WITH("WRAPPER");
4153 else if (TailMatches("CREATE|ALTER|DROP", "TEXT", "SEARCH"))
4154 COMPLETE_WITH("CONFIGURATION", "DICTIONARY", "PARSER", "TEMPLATE");
4155 else if (TailMatches("CREATE|ALTER|DROP", "USER", "MAPPING"))
4156 COMPLETE_WITH("FOR");
4158 else if (TailMatchesCS("\\l*") && !TailMatchesCS("\\lo*"))
4159 COMPLETE_WITH_QUERY(Query_for_list_of_databases
);
4160 else if (TailMatchesCS("\\password"))
4161 COMPLETE_WITH_QUERY(Query_for_list_of_roles
);
4162 else if (TailMatchesCS("\\pset"))
4163 COMPLETE_WITH_CS("border", "columns", "csv_fieldsep", "expanded",
4164 "fieldsep", "fieldsep_zero", "footer", "format",
4165 "linestyle", "null", "numericlocale",
4166 "pager", "pager_min_lines",
4167 "recordsep", "recordsep_zero",
4168 "tableattr", "title", "tuples_only",
4169 "unicode_border_linestyle",
4170 "unicode_column_linestyle",
4171 "unicode_header_linestyle");
4172 else if (TailMatchesCS("\\pset", MatchAny
))
4174 if (TailMatchesCS("format"))
4175 COMPLETE_WITH_CS("aligned", "asciidoc", "csv", "html", "latex",
4176 "latex-longtable", "troff-ms", "unaligned",
4178 else if (TailMatchesCS("linestyle"))
4179 COMPLETE_WITH_CS("ascii", "old-ascii", "unicode");
4180 else if (TailMatchesCS("pager"))
4181 COMPLETE_WITH_CS("on", "off", "always");
4182 else if (TailMatchesCS("unicode_border_linestyle|"
4183 "unicode_column_linestyle|"
4184 "unicode_header_linestyle"))
4185 COMPLETE_WITH_CS("single", "double");
4187 else if (TailMatchesCS("\\unset"))
4188 matches
= complete_from_variables(text
, "", "", true);
4189 else if (TailMatchesCS("\\set"))
4190 matches
= complete_from_variables(text
, "", "", false);
4191 else if (TailMatchesCS("\\set", MatchAny
))
4193 if (TailMatchesCS("AUTOCOMMIT|ON_ERROR_STOP|QUIET|"
4194 "SINGLELINE|SINGLESTEP"))
4195 COMPLETE_WITH_CS("on", "off");
4196 else if (TailMatchesCS("COMP_KEYWORD_CASE"))
4197 COMPLETE_WITH_CS("lower", "upper",
4198 "preserve-lower", "preserve-upper");
4199 else if (TailMatchesCS("ECHO"))
4200 COMPLETE_WITH_CS("errors", "queries", "all", "none");
4201 else if (TailMatchesCS("ECHO_HIDDEN"))
4202 COMPLETE_WITH_CS("noexec", "off", "on");
4203 else if (TailMatchesCS("HISTCONTROL"))
4204 COMPLETE_WITH_CS("ignorespace", "ignoredups",
4205 "ignoreboth", "none");
4206 else if (TailMatchesCS("ON_ERROR_ROLLBACK"))
4207 COMPLETE_WITH_CS("on", "off", "interactive");
4208 else if (TailMatchesCS("SHOW_CONTEXT"))
4209 COMPLETE_WITH_CS("never", "errors", "always");
4210 else if (TailMatchesCS("VERBOSITY"))
4211 COMPLETE_WITH_CS("default", "verbose", "terse", "sqlstate");
4213 else if (TailMatchesCS("\\sf*"))
4214 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_routines
, NULL
);
4215 else if (TailMatchesCS("\\sv*"))
4216 COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_views
, NULL
);
4217 else if (TailMatchesCS("\\cd|\\e|\\edit|\\g|\\gx|\\i|\\include|"
4218 "\\ir|\\include_relative|\\o|\\out|"
4219 "\\s|\\w|\\write|\\lo_import"))
4221 completion_charp
= "\\";
4222 completion_force_quote
= false;
4223 matches
= rl_completion_matches(text
, complete_from_files
);
4227 * Finally, we look through the list of "things", such as TABLE, INDEX and
4228 * check if that was the previous word. If so, execute the query to get a
4235 for (i
= 0; words_after_create
[i
].name
; i
++)
4237 if (pg_strcasecmp(prev_wd
, words_after_create
[i
].name
) == 0)
4239 if (words_after_create
[i
].query
)
4240 COMPLETE_WITH_QUERY(words_after_create
[i
].query
);
4241 else if (words_after_create
[i
].vquery
)
4242 COMPLETE_WITH_VERSIONED_QUERY(words_after_create
[i
].vquery
);
4243 else if (words_after_create
[i
].squery
)
4244 COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(words_after_create
[i
].squery
,
4252 * If we still don't have anything to match we have to fabricate some sort
4253 * of default list. If we were to just return NULL, readline automatically
4254 * attempts filename completion, and that's usually no good.
4256 if (matches
== NULL
)
4258 COMPLETE_WITH_CONST(true, "");
4259 #ifdef HAVE_RL_COMPLETION_APPEND_CHARACTER
4260 rl_completion_append_character
= '\0';
4265 free(previous_words
);
4269 /* Return our Grand List O' Matches */
4275 * GENERATOR FUNCTIONS
4277 * These functions do all the actual work of completing the input. They get
4278 * passed the text so far and the count how many times they have been called
4279 * so far with the same text.
4280 * If you read the above carefully, you'll see that these don't get called
4281 * directly but through the readline interface.
4282 * The return value is expected to be the full completion of the text, going
4283 * through a list each time, or NULL if there are no more matches. The string
4284 * will be free()'d by readline, so you must run it through strdup() or
4285 * something of that sort.
4289 * Common routine for create_command_generator and drop_command_generator.
4290 * Entries that have 'excluded' flags are not returned.
4293 create_or_drop_command_generator(const char *text
, int state
, bits32 excluded
)
4295 static int list_index
,
4299 /* If this is the first time for this completion, init some values */
4303 string_length
= strlen(text
);
4306 /* find something that matches */
4307 while ((name
= words_after_create
[list_index
++].name
))
4309 if ((pg_strncasecmp(name
, text
, string_length
) == 0) &&
4310 !(words_after_create
[list_index
- 1].flags
& excluded
))
4311 return pg_strdup_keyword_case(name
, text
);
4313 /* if nothing matches, return NULL */
4318 * This one gives you one from a list of things you can put after CREATE
4322 create_command_generator(const char *text
, int state
)
4324 return create_or_drop_command_generator(text
, state
, THING_NO_CREATE
);
4328 * This function gives you a list of things you can put after a DROP command.
4331 drop_command_generator(const char *text
, int state
)
4333 return create_or_drop_command_generator(text
, state
, THING_NO_DROP
);
4337 * This function gives you a list of things you can put after an ALTER command.
4340 alter_command_generator(const char *text
, int state
)
4342 return create_or_drop_command_generator(text
, state
, THING_NO_ALTER
);
4346 * These functions generate lists using server queries.
4347 * They are all wrappers for _complete_from_query.
4351 complete_from_query(const char *text
, int state
)
4353 /* query is assumed to work for any server version */
4354 return _complete_from_query(completion_charp
, NULL
, text
, state
);
4358 complete_from_versioned_query(const char *text
, int state
)
4360 const VersionedQuery
*vquery
= completion_vquery
;
4362 /* Find appropriate array element */
4363 while (pset
.sversion
< vquery
->min_server_version
)
4365 /* Fail completion if server is too old */
4366 if (vquery
->query
== NULL
)
4369 return _complete_from_query(vquery
->query
, NULL
, text
, state
);
4373 complete_from_schema_query(const char *text
, int state
)
4375 /* query is assumed to work for any server version */
4376 return _complete_from_query(completion_charp
, completion_squery
,
4381 complete_from_versioned_schema_query(const char *text
, int state
)
4383 const SchemaQuery
*squery
= completion_squery
;
4384 const VersionedQuery
*vquery
= completion_vquery
;
4386 /* Find appropriate array element */
4387 while (pset
.sversion
< squery
->min_server_version
)
4389 /* Fail completion if server is too old */
4390 if (squery
->catname
== NULL
)
4393 /* Likewise for the add-on text, if any */
4396 while (pset
.sversion
< vquery
->min_server_version
)
4398 if (vquery
->query
== NULL
)
4402 return _complete_from_query(vquery
? vquery
->query
: NULL
,
4403 squery
, text
, state
);
4408 * This creates a list of matching things, according to a query described by
4409 * the initial arguments. The caller has already done any work needed to
4410 * select the appropriate query for the server's version.
4412 * The query can be one of two kinds:
4414 * 1. A simple query which must contain a %d and a %s, which will be replaced
4415 * by the string length of the text and the text itself. The query may also
4416 * have up to four more %s in it; the first two such will be replaced by the
4417 * value of completion_info_charp, the next two by the value of
4418 * completion_info_charp2.
4420 * 2. A schema query used for completion of both schema and relation names.
4421 * These are more complex and must contain in the following order:
4422 * %d %s %d %s %d %s %s %d %s
4423 * where %d is the string length of the text and %s the text itself.
4425 * If both simple_query and schema_query are non-NULL, then we construct
4426 * a schema query and append the (uninterpreted) string simple_query to it.
4428 * It is assumed that strings should be escaped to become SQL literals
4429 * (that is, what is in the query is actually ... '%s' ...)
4431 * See top of file for examples of both kinds of query.
4433 * "text" and "state" are supplied by readline.
4436 _complete_from_query(const char *simple_query
,
4437 const SchemaQuery
*schema_query
,
4438 const char *text
, int state
)
4440 static int list_index
,
4442 static PGresult
*result
= NULL
;
4445 * If this is the first time for this completion, we fetch a list of our
4446 * "things" from the backend.
4450 PQExpBufferData query_buffer
;
4453 char *e_info_charp2
;
4454 const char *pstr
= text
;
4455 int char_length
= 0;
4458 byte_length
= strlen(text
);
4461 * Count length as number of characters (not bytes), for passing to
4467 pstr
+= PQmblenBounded(pstr
, pset
.encoding
);
4470 /* Free any prior result */
4474 /* Set up suitably-escaped copies of textual inputs */
4475 e_text
= escape_string(text
);
4477 if (completion_info_charp
)
4478 e_info_charp
= escape_string(completion_info_charp
);
4480 e_info_charp
= NULL
;
4482 if (completion_info_charp2
)
4483 e_info_charp2
= escape_string(completion_info_charp2
);
4485 e_info_charp2
= NULL
;
4487 initPQExpBuffer(&query_buffer
);
4491 /* schema_query gives us the pieces to assemble */
4492 const char *qualresult
= schema_query
->qualresult
;
4494 if (qualresult
== NULL
)
4495 qualresult
= schema_query
->result
;
4497 /* Get unqualified names matching the input-so-far */
4498 appendPQExpBuffer(&query_buffer
, "SELECT %s FROM %s WHERE ",
4499 schema_query
->result
,
4500 schema_query
->catname
);
4501 if (schema_query
->selcondition
)
4502 appendPQExpBuffer(&query_buffer
, "%s AND ",
4503 schema_query
->selcondition
);
4504 appendPQExpBuffer(&query_buffer
, "substring(%s,1,%d)='%s'",
4505 schema_query
->result
,
4506 char_length
, e_text
);
4507 appendPQExpBuffer(&query_buffer
, " AND %s",
4508 schema_query
->viscondition
);
4511 * When fetching relation names, suppress system catalogs unless
4512 * the input-so-far begins with "pg_". This is a compromise
4513 * between not offering system catalogs for completion at all, and
4514 * having them swamp the result when the input is just "p".
4516 if (strcmp(schema_query
->catname
,
4517 "pg_catalog.pg_class c") == 0 &&
4518 strncmp(text
, "pg_", 3) != 0)
4520 appendPQExpBufferStr(&query_buffer
,
4521 " AND c.relnamespace <> (SELECT oid FROM"
4522 " pg_catalog.pg_namespace WHERE nspname = 'pg_catalog')");
4526 * Add in matching schema names, but only if there is more than
4527 * one potential match among schema names.
4529 appendPQExpBuffer(&query_buffer
, "\nUNION\n"
4530 "SELECT pg_catalog.quote_ident(n.nspname) || '.' "
4531 "FROM pg_catalog.pg_namespace n "
4532 "WHERE substring(pg_catalog.quote_ident(n.nspname) || '.',1,%d)='%s'",
4533 char_length
, e_text
);
4534 appendPQExpBuffer(&query_buffer
,
4535 " AND (SELECT pg_catalog.count(*)"
4536 " FROM pg_catalog.pg_namespace"
4537 " WHERE substring(pg_catalog.quote_ident(nspname) || '.',1,%d) ="
4538 " substring('%s',1,pg_catalog.length(pg_catalog.quote_ident(nspname))+1)) > 1",
4539 char_length
, e_text
);
4542 * Add in matching qualified names, but only if there is exactly
4543 * one schema matching the input-so-far.
4545 appendPQExpBuffer(&query_buffer
, "\nUNION\n"
4546 "SELECT pg_catalog.quote_ident(n.nspname) || '.' || %s "
4547 "FROM %s, pg_catalog.pg_namespace n "
4548 "WHERE %s = n.oid AND ",
4550 schema_query
->catname
,
4551 schema_query
->namespace);
4552 if (schema_query
->selcondition
)
4553 appendPQExpBuffer(&query_buffer
, "%s AND ",
4554 schema_query
->selcondition
);
4555 appendPQExpBuffer(&query_buffer
, "substring(pg_catalog.quote_ident(n.nspname) || '.' || %s,1,%d)='%s'",
4557 char_length
, e_text
);
4560 * This condition exploits the single-matching-schema rule to
4561 * speed up the query
4563 appendPQExpBuffer(&query_buffer
,
4564 " AND substring(pg_catalog.quote_ident(n.nspname) || '.',1,%d) ="
4565 " substring('%s',1,pg_catalog.length(pg_catalog.quote_ident(n.nspname))+1)",
4566 char_length
, e_text
);
4567 appendPQExpBuffer(&query_buffer
,
4568 " AND (SELECT pg_catalog.count(*)"
4569 " FROM pg_catalog.pg_namespace"
4570 " WHERE substring(pg_catalog.quote_ident(nspname) || '.',1,%d) ="
4571 " substring('%s',1,pg_catalog.length(pg_catalog.quote_ident(nspname))+1)) = 1",
4572 char_length
, e_text
);
4574 /* If an addon query was provided, use it */
4576 appendPQExpBuffer(&query_buffer
, "\n%s", simple_query
);
4580 Assert(simple_query
);
4581 /* simple_query is an sprintf-style format string */
4582 appendPQExpBuffer(&query_buffer
, simple_query
,
4583 char_length
, e_text
,
4584 e_info_charp
, e_info_charp
,
4585 e_info_charp2
, e_info_charp2
);
4588 /* Limit the number of records in the result */
4589 appendPQExpBuffer(&query_buffer
, "\nLIMIT %d",
4590 completion_max_records
);
4592 result
= exec_query(query_buffer
.data
);
4594 termPQExpBuffer(&query_buffer
);
4599 free(e_info_charp2
);
4602 /* Find something that matches */
4603 if (result
&& PQresultStatus(result
) == PGRES_TUPLES_OK
)
4607 while (list_index
< PQntuples(result
) &&
4608 (item
= PQgetvalue(result
, list_index
++, 0)))
4609 if (pg_strncasecmp(text
, item
, byte_length
) == 0)
4610 return pg_strdup(item
);
4613 /* If nothing matches, free the db structure and return null */
4621 * This function returns in order one of a fixed, NULL pointer terminated list
4622 * of strings (if matching). This can be used if there are only a fixed number
4623 * SQL words that can appear at certain spot.
4626 complete_from_list(const char *text
, int state
)
4628 static int string_length
,
4631 static bool casesensitive
;
4634 /* need to have a list */
4635 Assert(completion_charpp
!= NULL
);
4637 /* Initialization */
4641 string_length
= strlen(text
);
4642 casesensitive
= completion_case_sensitive
;
4646 while ((item
= completion_charpp
[list_index
++]))
4648 /* First pass is case sensitive */
4649 if (casesensitive
&& strncmp(text
, item
, string_length
) == 0)
4652 return pg_strdup(item
);
4655 /* Second pass is case insensitive, don't bother counting matches */
4656 if (!casesensitive
&& pg_strncasecmp(text
, item
, string_length
) == 0)
4658 if (completion_case_sensitive
)
4659 return pg_strdup(item
);
4663 * If case insensitive matching was requested initially,
4664 * adjust the case according to setting.
4666 return pg_strdup_keyword_case(item
, text
);
4671 * No matches found. If we're not case insensitive already, lets switch to
4672 * being case insensitive and try again
4674 if (casesensitive
&& matches
== 0)
4676 casesensitive
= false;
4679 return complete_from_list(text
, state
);
4682 /* If no more matches, return null. */
4688 * This function returns one fixed string the first time even if it doesn't
4689 * match what's there, and nothing the second time. The string
4690 * to be used must be in completion_charp.
4692 * If the given string is "", this has the effect of preventing readline
4693 * from doing any completion. (Without this, readline tries to do filename
4694 * completion which is seldom the right thing.)
4696 * If the given string is not empty, readline will replace whatever the
4697 * user typed with that string. This behavior might be useful if it's
4698 * completely certain that we know what must appear at a certain spot,
4699 * so that it's okay to overwrite misspellings. In practice, given the
4700 * relatively lame parsing technology used in this file, the level of
4701 * certainty is seldom that high, so that you probably don't want to
4702 * use this. Use complete_from_list with a one-element list instead;
4703 * that won't try to auto-correct "misspellings".
4706 complete_from_const(const char *text
, int state
)
4708 Assert(completion_charp
!= NULL
);
4711 if (completion_case_sensitive
)
4712 return pg_strdup(completion_charp
);
4716 * If case insensitive matching was requested initially, adjust
4717 * the case according to setting.
4719 return pg_strdup_keyword_case(completion_charp
, text
);
4727 * This function appends the variable name with prefix and suffix to
4728 * the variable names array.
4731 append_variable_names(char ***varnames
, int *nvars
,
4732 int *maxvars
, const char *varname
,
4733 const char *prefix
, const char *suffix
)
4735 if (*nvars
>= *maxvars
)
4738 *varnames
= (char **) pg_realloc(*varnames
,
4739 ((*maxvars
) + 1) * sizeof(char *));
4742 (*varnames
)[(*nvars
)++] = psprintf("%s%s%s", prefix
, varname
, suffix
);
4747 * This function supports completion with the name of a psql variable.
4748 * The variable names can be prefixed and suffixed with additional text
4749 * to support quoting usages. If need_value is true, only variables
4750 * that are currently set are included; otherwise, special variables
4751 * (those that have hooks) are included even if currently unset.
4754 complete_from_variables(const char *text
, const char *prefix
, const char *suffix
,
4762 struct _variable
*ptr
;
4764 varnames
= (char **) pg_malloc((maxvars
+ 1) * sizeof(char *));
4766 for (ptr
= pset
.vars
->next
; ptr
; ptr
= ptr
->next
)
4768 if (need_value
&& !(ptr
->value
))
4770 append_variable_names(&varnames
, &nvars
, &maxvars
, ptr
->name
,
4774 varnames
[nvars
] = NULL
;
4775 COMPLETE_WITH_LIST_CS((const char *const *) varnames
);
4777 for (i
= 0; i
< nvars
; i
++)
4786 * This function wraps rl_filename_completion_function() to strip quotes from
4787 * the input before searching for matches and to quote any matches for which
4788 * the consuming command will require it.
4790 * Caller must set completion_charp to a zero- or one-character string
4791 * containing the escape character. This is necessary since \copy has no
4792 * escape character, but every other backslash command recognizes "\" as an
4795 * Caller must also set completion_force_quote to indicate whether to force
4796 * quotes around the result. (The SQL COPY command requires that.)
4799 complete_from_files(const char *text
, int state
)
4801 #ifdef USE_FILENAME_QUOTING_FUNCTIONS
4804 * If we're using a version of Readline that supports filename quoting
4805 * hooks, rely on those, and invoke rl_filename_completion_function()
4806 * without messing with its arguments. Readline does stuff internally
4807 * that does not work well at all if we try to handle dequoting here.
4808 * Instead, Readline will call quote_file_name() and dequote_file_name()
4809 * (see below) at appropriate times.
4811 * ... or at least, mostly it will. There are some paths involving
4812 * unmatched file names in which Readline never calls quote_file_name(),
4813 * and if left to its own devices it will incorrectly append a quote
4814 * anyway. Set rl_completion_suppress_quote to prevent that. If we do
4815 * get to quote_file_name(), we'll clear this again. (Yes, this seems
4816 * like it's working around Readline bugs.)
4818 #ifdef HAVE_RL_COMPLETION_SUPPRESS_QUOTE
4819 rl_completion_suppress_quote
= 1;
4822 /* If user typed a quote, force quoting (never remove user's quote) */
4824 completion_force_quote
= true;
4826 return rl_filename_completion_function(text
, state
);
4830 * Otherwise, we have to do the best we can.
4832 static const char *unquoted_text
;
4833 char *unquoted_match
;
4836 /* If user typed a quote, force quoting (never remove user's quote) */
4838 completion_force_quote
= true;
4842 /* Initialization: stash the unquoted input. */
4843 unquoted_text
= strtokx(text
, "", NULL
, "'", *completion_charp
,
4844 false, true, pset
.encoding
);
4845 /* expect a NULL return for the empty string only */
4848 Assert(*text
== '\0');
4849 unquoted_text
= text
;
4853 unquoted_match
= rl_filename_completion_function(unquoted_text
, state
);
4856 struct stat statbuf
;
4857 bool is_dir
= (stat(unquoted_match
, &statbuf
) == 0 &&
4858 S_ISDIR(statbuf
.st_mode
) != 0);
4860 /* Re-quote the result, if needed. */
4861 ret
= quote_if_needed(unquoted_match
, " \t\r\n\"`",
4862 '\'', *completion_charp
,
4863 completion_force_quote
,
4866 free(unquoted_match
);
4868 ret
= unquoted_match
;
4871 * If it's a directory, replace trailing quote with a slash; this is
4872 * usually more convenient. (If we didn't quote, leave this to
4875 if (*ret
== '\'' && is_dir
)
4877 char *retend
= ret
+ strlen(ret
) - 1;
4879 Assert(*retend
== '\'');
4881 /* Try to prevent libedit from adding a space, too */
4882 #ifdef HAVE_RL_COMPLETION_APPEND_CHARACTER
4883 rl_completion_append_character
= '\0';
4889 #endif /* USE_FILENAME_QUOTING_FUNCTIONS */
4893 /* HELPER FUNCTIONS */
4897 * Make a pg_strdup copy of s and convert the case according to
4898 * COMP_KEYWORD_CASE setting, using ref as the text that was already entered.
4901 pg_strdup_keyword_case(const char *s
, const char *ref
)
4905 unsigned char first
= ref
[0];
4909 if (pset
.comp_case
== PSQL_COMP_CASE_LOWER
||
4910 ((pset
.comp_case
== PSQL_COMP_CASE_PRESERVE_LOWER
||
4911 pset
.comp_case
== PSQL_COMP_CASE_PRESERVE_UPPER
) && islower(first
)) ||
4912 (pset
.comp_case
== PSQL_COMP_CASE_PRESERVE_LOWER
&& !isalpha(first
)))
4914 for (p
= ret
; *p
; p
++)
4915 *p
= pg_tolower((unsigned char) *p
);
4919 for (p
= ret
; *p
; p
++)
4920 *p
= pg_toupper((unsigned char) *p
);
4928 * escape_string - Escape argument for use as string literal.
4930 * The returned value has to be freed.
4933 escape_string(const char *text
)
4938 text_length
= strlen(text
);
4940 result
= pg_malloc(text_length
* 2 + 1);
4941 PQescapeStringConn(pset
.db
, result
, text
, text_length
, NULL
);
4948 * Execute a query and report any errors. This should be the preferred way of
4949 * talking to the database in this file.
4952 exec_query(const char *query
)
4956 if (query
== NULL
|| !pset
.db
|| PQstatus(pset
.db
) != CONNECTION_OK
)
4959 result
= PQexec(pset
.db
, query
);
4961 if (PQresultStatus(result
) != PGRES_TUPLES_OK
)
4964 pg_log_error("tab completion query failed: %s\nQuery was:\n%s",
4965 PQerrorMessage(pset
.db
), query
);
4976 * Parse all the word(s) before point.
4978 * Returns a malloc'd array of character pointers that point into the malloc'd
4979 * data array returned to *buffer; caller must free() both of these when done.
4980 * *nwords receives the number of words found, ie, the valid length of the
4983 * Words are returned right to left, that is, previous_words[0] gets the last
4984 * word before point, previous_words[1] the next-to-last, etc.
4987 get_previous_words(int point
, char **buffer
, int *nwords
)
4989 char **previous_words
;
4992 int words_found
= 0;
4996 * If we have anything in tab_completion_query_buf, paste it together with
4997 * rl_line_buffer to construct the full query. Otherwise we can just use
4998 * rl_line_buffer as the input string.
5000 if (tab_completion_query_buf
&& tab_completion_query_buf
->len
> 0)
5002 i
= tab_completion_query_buf
->len
;
5003 buf
= pg_malloc(point
+ i
+ 2);
5004 memcpy(buf
, tab_completion_query_buf
->data
, i
);
5006 memcpy(buf
+ i
, rl_line_buffer
, point
);
5009 /* Readjust point to reference appropriate offset in buf */
5013 buf
= rl_line_buffer
;
5016 * Allocate an array of string pointers and a buffer to hold the strings
5017 * themselves. The worst case is that the line contains only
5018 * non-whitespace WORD_BREAKS characters, making each one a separate word.
5019 * This is usually much more space than we need, but it's cheaper than
5020 * doing a separate malloc() for each word.
5022 previous_words
= (char **) pg_malloc(point
* sizeof(char *));
5023 *buffer
= outptr
= (char *) pg_malloc(point
* 2);
5026 * First we look for a non-word char before the current point. (This is
5027 * probably useless, if readline is on the same page as we are about what
5028 * is a word, but if so it's cheap.)
5030 for (i
= point
- 1; i
>= 0; i
--)
5032 if (strchr(WORD_BREAKS
, buf
[i
]))
5038 * Now parse words, working backwards, until we hit start of line. The
5039 * backwards scan has some interesting but intentional properties
5040 * concerning parenthesis handling.
5046 bool inquotes
= false;
5047 int parentheses
= 0;
5049 /* now find the first non-space which then constitutes the end */
5051 for (i
= point
; i
>= 0; i
--)
5053 if (!isspace((unsigned char) buf
[i
]))
5059 /* if no end found, we're done */
5064 * Otherwise we now look for the start. The start is either the last
5065 * character before any word-break character going backwards from the
5066 * end, or it's simply character 0. We also handle open quotes and
5069 for (start
= end
; start
> 0; start
--)
5071 if (buf
[start
] == '"')
5072 inquotes
= !inquotes
;
5075 if (buf
[start
] == ')')
5077 else if (buf
[start
] == '(')
5079 if (--parentheses
<= 0)
5082 else if (parentheses
== 0 &&
5083 strchr(WORD_BREAKS
, buf
[start
- 1]))
5088 /* Return the word located at start to end inclusive */
5089 previous_words
[words_found
++] = outptr
;
5090 i
= end
- start
+ 1;
5091 memcpy(outptr
, &buf
[start
], i
);
5095 /* Continue searching */
5099 /* Release parsing input workspace, if we made one above */
5100 if (buf
!= rl_line_buffer
)
5103 *nwords
= words_found
;
5104 return previous_words
;
5108 * Look up the type for the GUC variable with the passed name.
5110 * Returns NULL if the variable is unknown. Otherwise the returned string,
5111 * containing the type, has to be freed.
5114 get_guctype(const char *varname
)
5116 PQExpBufferData query_buffer
;
5119 char *guctype
= NULL
;
5121 e_varname
= escape_string(varname
);
5123 initPQExpBuffer(&query_buffer
);
5124 appendPQExpBuffer(&query_buffer
,
5125 "SELECT vartype FROM pg_catalog.pg_settings "
5126 "WHERE pg_catalog.lower(name) = pg_catalog.lower('%s')",
5129 result
= exec_query(query_buffer
.data
);
5130 termPQExpBuffer(&query_buffer
);
5133 if (PQresultStatus(result
) == PGRES_TUPLES_OK
&& PQntuples(result
) > 0)
5134 guctype
= pg_strdup(PQgetvalue(result
, 0, 0));
5141 #ifdef USE_FILENAME_QUOTING_FUNCTIONS
5144 * Quote a filename according to SQL rules, returning a malloc'd string.
5145 * completion_charp must point to escape character or '\0', and
5146 * completion_force_quote must be set correctly, as per comments for
5147 * complete_from_files().
5150 quote_file_name(char *fname
, int match_type
, char *quote_pointer
)
5153 struct stat statbuf
;
5155 /* Quote if needed. */
5156 s
= quote_if_needed(fname
, " \t\r\n\"`",
5157 '\'', *completion_charp
,
5158 completion_force_quote
,
5161 s
= pg_strdup(fname
);
5164 * However, some of the time we have to strip the trailing quote from what
5165 * we send back. Never strip the trailing quote if the user already typed
5166 * one; otherwise, suppress the trailing quote if we have multiple/no
5167 * matches (because we don't want to add a quote if the input is seemingly
5168 * unfinished), or if the input was already quoted (because Readline will
5169 * do arguably-buggy things otherwise), or if the file does not exist, or
5170 * if it's a directory.
5173 completion_last_char
!= '\'' &&
5174 (match_type
!= SINGLE_MATCH
||
5175 (quote_pointer
&& *quote_pointer
== '\'') ||
5176 stat(fname
, &statbuf
) != 0 ||
5177 S_ISDIR(statbuf
.st_mode
)))
5179 char *send
= s
+ strlen(s
) - 1;
5181 Assert(*send
== '\'');
5186 * And now we can let Readline do its thing with possibly adding a quote
5187 * on its own accord. (This covers some additional cases beyond those
5188 * dealt with above.)
5190 #ifdef HAVE_RL_COMPLETION_SUPPRESS_QUOTE
5191 rl_completion_suppress_quote
= 0;
5195 * If user typed a leading quote character other than single quote (i.e.,
5196 * double quote), zap it, so that we replace it with the correct single
5199 if (quote_pointer
&& *quote_pointer
!= '\'')
5200 *quote_pointer
= '\0';
5206 * Dequote a filename, if it's quoted.
5207 * completion_charp must point to escape character or '\0', as per
5208 * comments for complete_from_files().
5211 dequote_file_name(char *fname
, int quote_char
)
5213 char *unquoted_fname
;
5216 * If quote_char is set, it's not included in "fname". We have to add it
5217 * or strtokx will not interpret the string correctly (notably, it won't
5218 * recognize escapes).
5220 if (quote_char
== '\'')
5222 char *workspace
= (char *) pg_malloc(strlen(fname
) + 2);
5224 workspace
[0] = quote_char
;
5225 strcpy(workspace
+ 1, fname
);
5226 unquoted_fname
= strtokx(workspace
, "", NULL
, "'", *completion_charp
,
5227 false, true, pset
.encoding
);
5231 unquoted_fname
= strtokx(fname
, "", NULL
, "'", *completion_charp
,
5232 false, true, pset
.encoding
);
5234 /* expect a NULL return for the empty string only */
5235 if (!unquoted_fname
)
5237 Assert(*fname
== '\0');
5238 unquoted_fname
= fname
;
5241 /* readline expects a malloc'd result that it is to free */
5242 return pg_strdup(unquoted_fname
);
5245 #endif /* USE_FILENAME_QUOTING_FUNCTIONS */
5247 #endif /* USE_READLINE */