Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * namespace.c
4 : * code to support accessing and searching namespaces
5 : *
6 : * This is separate from pg_namespace.c, which contains the routines that
7 : * directly manipulate the pg_namespace system catalog. This module
8 : * provides routines associated with defining a "namespace search path"
9 : * and implementing search-path-controlled searches.
10 : *
11 : *
12 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
13 : * Portions Copyright (c) 1994, Regents of the University of California
14 : *
15 : * IDENTIFICATION
16 : * src/backend/catalog/namespace.c
17 : *
18 : *-------------------------------------------------------------------------
19 : */
20 : #include "postgres.h"
21 :
22 : #include "access/htup_details.h"
23 : #include "access/parallel.h"
24 : #include "access/xact.h"
25 : #include "access/xlog.h"
26 : #include "catalog/dependency.h"
27 : #include "catalog/namespace.h"
28 : #include "catalog/objectaccess.h"
29 : #include "catalog/pg_authid.h"
30 : #include "catalog/pg_collation.h"
31 : #include "catalog/pg_conversion.h"
32 : #include "catalog/pg_database.h"
33 : #include "catalog/pg_namespace.h"
34 : #include "catalog/pg_opclass.h"
35 : #include "catalog/pg_operator.h"
36 : #include "catalog/pg_opfamily.h"
37 : #include "catalog/pg_proc.h"
38 : #include "catalog/pg_statistic_ext.h"
39 : #include "catalog/pg_ts_config.h"
40 : #include "catalog/pg_ts_dict.h"
41 : #include "catalog/pg_ts_parser.h"
42 : #include "catalog/pg_ts_template.h"
43 : #include "catalog/pg_type.h"
44 : #include "commands/dbcommands.h"
45 : #include "common/hashfn_unstable.h"
46 : #include "funcapi.h"
47 : #include "mb/pg_wchar.h"
48 : #include "miscadmin.h"
49 : #include "nodes/makefuncs.h"
50 : #include "storage/ipc.h"
51 : #include "storage/lmgr.h"
52 : #include "storage/procarray.h"
53 : #include "utils/acl.h"
54 : #include "utils/builtins.h"
55 : #include "utils/catcache.h"
56 : #include "utils/guc_hooks.h"
57 : #include "utils/inval.h"
58 : #include "utils/lsyscache.h"
59 : #include "utils/memutils.h"
60 : #include "utils/snapmgr.h"
61 : #include "utils/syscache.h"
62 : #include "utils/varlena.h"
63 :
64 :
65 : /*
66 : * The namespace search path is a possibly-empty list of namespace OIDs.
67 : * In addition to the explicit list, implicitly-searched namespaces
68 : * may be included:
69 : *
70 : * 1. If a TEMP table namespace has been initialized in this session, it
71 : * is implicitly searched first.
72 : *
73 : * 2. The system catalog namespace is always searched. If the system
74 : * namespace is present in the explicit path then it will be searched in
75 : * the specified order; otherwise it will be searched after TEMP tables and
76 : * *before* the explicit list. (It might seem that the system namespace
77 : * should be implicitly last, but this behavior appears to be required by
78 : * SQL99. Also, this provides a way to search the system namespace first
79 : * without thereby making it the default creation target namespace.)
80 : *
81 : * For security reasons, searches using the search path will ignore the temp
82 : * namespace when searching for any object type other than relations and
83 : * types. (We must allow types since temp tables have rowtypes.)
84 : *
85 : * The default creation target namespace is always the first element of the
86 : * explicit list. If the explicit list is empty, there is no default target.
87 : *
88 : * The textual specification of search_path can include "$user" to refer to
89 : * the namespace named the same as the current user, if any. (This is just
90 : * ignored if there is no such namespace.) Also, it can include "pg_temp"
91 : * to refer to the current backend's temp namespace. This is usually also
92 : * ignorable if the temp namespace hasn't been set up, but there's a special
93 : * case: if "pg_temp" appears first then it should be the default creation
94 : * target. We kluge this case a little bit so that the temp namespace isn't
95 : * set up until the first attempt to create something in it. (The reason for
96 : * klugery is that we can't create the temp namespace outside a transaction,
97 : * but initial GUC processing of search_path happens outside a transaction.)
98 : * activeTempCreationPending is true if "pg_temp" appears first in the string
99 : * but is not reflected in activeCreationNamespace because the namespace isn't
100 : * set up yet.
101 : *
102 : * In bootstrap mode, the search path is set equal to "pg_catalog", so that
103 : * the system namespace is the only one searched or inserted into.
104 : * initdb is also careful to set search_path to "pg_catalog" for its
105 : * post-bootstrap standalone backend runs. Otherwise the default search
106 : * path is determined by GUC. The factory default path contains the PUBLIC
107 : * namespace (if it exists), preceded by the user's personal namespace
108 : * (if one exists).
109 : *
110 : * activeSearchPath is always the actually active path; it points to
111 : * baseSearchPath which is the list derived from namespace_search_path.
112 : *
113 : * If baseSearchPathValid is false, then baseSearchPath (and other derived
114 : * variables) need to be recomputed from namespace_search_path, or retrieved
115 : * from the search path cache if there haven't been any syscache
116 : * invalidations. We mark it invalid upon an assignment to
117 : * namespace_search_path or receipt of a syscache invalidation event for
118 : * pg_namespace or pg_authid. The recomputation is done during the next
119 : * lookup attempt.
120 : *
121 : * Any namespaces mentioned in namespace_search_path that are not readable
122 : * by the current user ID are simply left out of baseSearchPath; so
123 : * we have to be willing to recompute the path when current userid changes.
124 : * namespaceUser is the userid the path has been computed for.
125 : *
126 : * Note: all data pointed to by these List variables is in TopMemoryContext.
127 : *
128 : * activePathGeneration is incremented whenever the effective values of
129 : * activeSearchPath/activeCreationNamespace/activeTempCreationPending change.
130 : * This can be used to quickly detect whether any change has happened since
131 : * a previous examination of the search path state.
132 : */
133 :
134 : /* These variables define the actually active state: */
135 :
136 : static List *activeSearchPath = NIL;
137 :
138 : /* default place to create stuff; if InvalidOid, no default */
139 : static Oid activeCreationNamespace = InvalidOid;
140 :
141 : /* if true, activeCreationNamespace is wrong, it should be temp namespace */
142 : static bool activeTempCreationPending = false;
143 :
144 : /* current generation counter; make sure this is never zero */
145 : static uint64 activePathGeneration = 1;
146 :
147 : /* These variables are the values last derived from namespace_search_path: */
148 :
149 : static List *baseSearchPath = NIL;
150 :
151 : static Oid baseCreationNamespace = InvalidOid;
152 :
153 : static bool baseTempCreationPending = false;
154 :
155 : static Oid namespaceUser = InvalidOid;
156 :
157 : /* The above four values are valid only if baseSearchPathValid */
158 : static bool baseSearchPathValid = true;
159 :
160 : /*
161 : * Storage for search path cache. Clear searchPathCacheValid as a simple
162 : * way to invalidate *all* the cache entries, not just the active one.
163 : */
164 : static bool searchPathCacheValid = false;
165 : static MemoryContext SearchPathCacheContext = NULL;
166 :
167 : typedef struct SearchPathCacheKey
168 : {
169 : const char *searchPath;
170 : Oid roleid;
171 : } SearchPathCacheKey;
172 :
173 : typedef struct SearchPathCacheEntry
174 : {
175 : SearchPathCacheKey key;
176 : List *oidlist; /* namespace OIDs that pass ACL checks */
177 : List *finalPath; /* cached final computed search path */
178 : Oid firstNS; /* first explicitly-listed namespace */
179 : bool temp_missing;
180 : bool forceRecompute; /* force recompute of finalPath */
181 :
182 : /* needed for simplehash */
183 : char status;
184 : } SearchPathCacheEntry;
185 :
186 : /*
187 : * myTempNamespace is InvalidOid until and unless a TEMP namespace is set up
188 : * in a particular backend session (this happens when a CREATE TEMP TABLE
189 : * command is first executed). Thereafter it's the OID of the temp namespace.
190 : *
191 : * myTempToastNamespace is the OID of the namespace for my temp tables' toast
192 : * tables. It is set when myTempNamespace is, and is InvalidOid before that.
193 : *
194 : * myTempNamespaceSubID shows whether we've created the TEMP namespace in the
195 : * current subtransaction. The flag propagates up the subtransaction tree,
196 : * so the main transaction will correctly recognize the flag if all
197 : * intermediate subtransactions commit. When it is InvalidSubTransactionId,
198 : * we either haven't made the TEMP namespace yet, or have successfully
199 : * committed its creation, depending on whether myTempNamespace is valid.
200 : */
201 : static Oid myTempNamespace = InvalidOid;
202 :
203 : static Oid myTempToastNamespace = InvalidOid;
204 :
205 : static SubTransactionId myTempNamespaceSubID = InvalidSubTransactionId;
206 :
207 : /*
208 : * This is the user's textual search path specification --- it's the value
209 : * of the GUC variable 'search_path'.
210 : */
211 : char *namespace_search_path = NULL;
212 :
213 :
214 : /* Local functions */
215 : static bool RelationIsVisibleExt(Oid relid, bool *is_missing);
216 : static bool TypeIsVisibleExt(Oid typid, bool *is_missing);
217 : static bool FunctionIsVisibleExt(Oid funcid, bool *is_missing);
218 : static bool OperatorIsVisibleExt(Oid oprid, bool *is_missing);
219 : static bool OpclassIsVisibleExt(Oid opcid, bool *is_missing);
220 : static bool OpfamilyIsVisibleExt(Oid opfid, bool *is_missing);
221 : static bool CollationIsVisibleExt(Oid collid, bool *is_missing);
222 : static bool ConversionIsVisibleExt(Oid conid, bool *is_missing);
223 : static bool StatisticsObjIsVisibleExt(Oid stxid, bool *is_missing);
224 : static bool TSParserIsVisibleExt(Oid prsId, bool *is_missing);
225 : static bool TSDictionaryIsVisibleExt(Oid dictId, bool *is_missing);
226 : static bool TSTemplateIsVisibleExt(Oid tmplId, bool *is_missing);
227 : static bool TSConfigIsVisibleExt(Oid cfgid, bool *is_missing);
228 : static void recomputeNamespacePath(void);
229 : static void AccessTempTableNamespace(bool force);
230 : static void InitTempTableNamespace(void);
231 : static void RemoveTempRelations(Oid tempNamespaceId);
232 : static void RemoveTempRelationsCallback(int code, Datum arg);
233 : static void InvalidationCallback(Datum arg, int cacheid, uint32 hashvalue);
234 : static bool MatchNamedCall(HeapTuple proctup, int nargs, List *argnames,
235 : bool include_out_arguments, int pronargs,
236 : int **argnumbers);
237 :
238 : /*
239 : * Recomputing the namespace path can be costly when done frequently, such as
240 : * when a function has search_path set in proconfig. Add a search path cache
241 : * that can be used by recomputeNamespacePath().
242 : *
243 : * The cache is also used to remember already-validated strings in
244 : * check_search_path() to avoid the need to call SplitIdentifierString()
245 : * repeatedly.
246 : *
247 : * The search path cache is based on a wrapper around a simplehash hash table
248 : * (nsphash, defined below). The spcache wrapper deals with OOM while trying
249 : * to initialize a key, optimizes repeated lookups of the same key, and also
250 : * offers a more convenient API.
251 : */
252 :
253 : static inline uint32
254 164530 : spcachekey_hash(SearchPathCacheKey key)
255 : {
256 : fasthash_state hs;
257 : int sp_len;
258 :
259 164530 : fasthash_init(&hs, 0);
260 :
261 164530 : hs.accum = key.roleid;
262 164530 : fasthash_combine(&hs);
263 :
264 : /*
265 : * Combine search path into the hash and save the length for tweaking the
266 : * final mix.
267 : */
268 164530 : sp_len = fasthash_accum_cstring(&hs, key.searchPath);
269 :
270 164530 : return fasthash_final32(&hs, sp_len);
271 : }
272 :
273 : static inline bool
274 76454 : spcachekey_equal(SearchPathCacheKey a, SearchPathCacheKey b)
275 : {
276 152508 : return a.roleid == b.roleid &&
277 76054 : strcmp(a.searchPath, b.searchPath) == 0;
278 : }
279 :
280 : #define SH_PREFIX nsphash
281 : #define SH_ELEMENT_TYPE SearchPathCacheEntry
282 : #define SH_KEY_TYPE SearchPathCacheKey
283 : #define SH_KEY key
284 : #define SH_HASH_KEY(tb, key) spcachekey_hash(key)
285 : #define SH_EQUAL(tb, a, b) spcachekey_equal(a, b)
286 : #define SH_SCOPE static inline
287 : #define SH_DECLARE
288 : #define SH_DEFINE
289 : #include "lib/simplehash.h"
290 :
291 : /*
292 : * We only expect a small number of unique search_path strings to be used. If
293 : * this cache grows to an unreasonable size, reset it to avoid steady-state
294 : * memory growth. Most likely, only a few of those entries will benefit from
295 : * the cache, and the cache will be quickly repopulated with such entries.
296 : */
297 : #define SPCACHE_RESET_THRESHOLD 256
298 :
299 : static nsphash_hash *SearchPathCache = NULL;
300 : static SearchPathCacheEntry *LastSearchPathCacheEntry = NULL;
301 :
302 : /*
303 : * Create or reset search_path cache as necessary.
304 : */
305 : static void
306 214698 : spcache_init(void)
307 : {
308 214698 : if (SearchPathCache && searchPathCacheValid &&
309 182136 : SearchPathCache->members < SPCACHE_RESET_THRESHOLD)
310 182136 : return;
311 :
312 32562 : searchPathCacheValid = false;
313 32562 : baseSearchPathValid = false;
314 :
315 : /*
316 : * Make sure we don't leave dangling pointers if a failure happens during
317 : * initialization.
318 : */
319 32562 : SearchPathCache = NULL;
320 32562 : LastSearchPathCacheEntry = NULL;
321 :
322 32562 : if (SearchPathCacheContext == NULL)
323 : {
324 : /* Make the context we'll keep search path cache hashtable in */
325 20674 : SearchPathCacheContext = AllocSetContextCreate(TopMemoryContext,
326 : "search_path processing cache",
327 : ALLOCSET_DEFAULT_SIZES);
328 : }
329 : else
330 : {
331 11888 : MemoryContextReset(SearchPathCacheContext);
332 : }
333 :
334 : /* arbitrary initial starting size of 16 elements */
335 32562 : SearchPathCache = nsphash_create(SearchPathCacheContext, 16, NULL);
336 32562 : searchPathCacheValid = true;
337 : }
338 :
339 : /*
340 : * Look up entry in search path cache without inserting. Returns NULL if not
341 : * present.
342 : */
343 : static SearchPathCacheEntry *
344 138752 : spcache_lookup(const char *searchPath, Oid roleid)
345 : {
346 138752 : if (LastSearchPathCacheEntry &&
347 136900 : LastSearchPathCacheEntry->key.roleid == roleid &&
348 136670 : strcmp(LastSearchPathCacheEntry->key.searchPath, searchPath) == 0)
349 : {
350 95706 : return LastSearchPathCacheEntry;
351 : }
352 : else
353 : {
354 : SearchPathCacheEntry *entry;
355 43046 : SearchPathCacheKey cachekey = {
356 : .searchPath = searchPath,
357 : .roleid = roleid
358 : };
359 :
360 43046 : entry = nsphash_lookup(SearchPathCache, cachekey);
361 43046 : if (entry)
362 35092 : LastSearchPathCacheEntry = entry;
363 43046 : return entry;
364 : }
365 : }
366 :
367 : /*
368 : * Look up or insert entry in search path cache.
369 : *
370 : * Initialize key safely, so that OOM does not leave an entry without a valid
371 : * key. Caller must ensure that non-key contents are properly initialized.
372 : */
373 : static SearchPathCacheEntry *
374 83900 : spcache_insert(const char *searchPath, Oid roleid)
375 : {
376 83900 : if (LastSearchPathCacheEntry &&
377 51338 : LastSearchPathCacheEntry->key.roleid == roleid &&
378 48004 : strcmp(LastSearchPathCacheEntry->key.searchPath, searchPath) == 0)
379 : {
380 2898 : return LastSearchPathCacheEntry;
381 : }
382 : else
383 : {
384 : SearchPathCacheEntry *entry;
385 81002 : SearchPathCacheKey cachekey = {
386 : .searchPath = searchPath,
387 : .roleid = roleid
388 : };
389 :
390 : /*
391 : * searchPath is not saved in SearchPathCacheContext. First perform a
392 : * lookup, and copy searchPath only if we need to create a new entry.
393 : */
394 81002 : entry = nsphash_lookup(SearchPathCache, cachekey);
395 :
396 81002 : if (!entry)
397 : {
398 : bool found;
399 :
400 40362 : cachekey.searchPath = MemoryContextStrdup(SearchPathCacheContext, searchPath);
401 40362 : entry = nsphash_insert(SearchPathCache, cachekey, &found);
402 : Assert(!found);
403 :
404 40362 : entry->oidlist = NIL;
405 40362 : entry->finalPath = NIL;
406 40362 : entry->firstNS = InvalidOid;
407 40362 : entry->temp_missing = false;
408 40362 : entry->forceRecompute = false;
409 : /* do not touch entry->status, used by simplehash */
410 : }
411 :
412 81002 : LastSearchPathCacheEntry = entry;
413 81002 : return entry;
414 : }
415 : }
416 :
417 : /*
418 : * RangeVarGetRelidExtended
419 : * Given a RangeVar describing an existing relation,
420 : * select the proper namespace and look up the relation OID.
421 : *
422 : * If the schema or relation is not found, return InvalidOid if flags contains
423 : * RVR_MISSING_OK, otherwise raise an error.
424 : *
425 : * If flags contains RVR_NOWAIT, throw an error if we'd have to wait for a
426 : * lock.
427 : *
428 : * If flags contains RVR_SKIP_LOCKED, return InvalidOid if we'd have to wait
429 : * for a lock.
430 : *
431 : * flags cannot contain both RVR_NOWAIT and RVR_SKIP_LOCKED.
432 : *
433 : * Note that if RVR_MISSING_OK and RVR_SKIP_LOCKED are both specified, a
434 : * return value of InvalidOid could either mean the relation is missing or it
435 : * could not be locked.
436 : *
437 : * Callback allows caller to check permissions or acquire additional locks
438 : * prior to grabbing the relation lock.
439 : */
440 : Oid
441 648274 : RangeVarGetRelidExtended(const RangeVar *relation, LOCKMODE lockmode,
442 : uint32 flags,
443 : RangeVarGetRelidCallback callback, void *callback_arg)
444 : {
445 : uint64 inval_count;
446 : Oid relId;
447 648274 : Oid oldRelId = InvalidOid;
448 648274 : bool retry = false;
449 648274 : bool missing_ok = (flags & RVR_MISSING_OK) != 0;
450 :
451 : /* verify that flags do no conflict */
452 : Assert(!((flags & RVR_NOWAIT) && (flags & RVR_SKIP_LOCKED)));
453 :
454 : /*
455 : * We check the catalog name and then ignore it.
456 : */
457 648274 : if (relation->catalogname)
458 : {
459 80 : if (strcmp(relation->catalogname, get_database_name(MyDatabaseId)) != 0)
460 80 : ereport(ERROR,
461 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
462 : errmsg("cross-database references are not implemented: \"%s.%s.%s\"",
463 : relation->catalogname, relation->schemaname,
464 : relation->relname)));
465 : }
466 :
467 : /*
468 : * DDL operations can change the results of a name lookup. Since all such
469 : * operations will generate invalidation messages, we keep track of
470 : * whether any such messages show up while we're performing the operation,
471 : * and retry until either (1) no more invalidation messages show up or (2)
472 : * the answer doesn't change.
473 : *
474 : * But if lockmode = NoLock, then we assume that either the caller is OK
475 : * with the answer changing under them, or that they already hold some
476 : * appropriate lock, and therefore return the first answer we get without
477 : * checking for invalidation messages. Also, if the requested lock is
478 : * already held, LockRelationOid will not AcceptInvalidationMessages, so
479 : * we may fail to notice a change. We could protect against that case by
480 : * calling AcceptInvalidationMessages() before beginning this loop, but
481 : * that would add a significant amount overhead, so for now we don't.
482 : */
483 : for (;;)
484 : {
485 : /*
486 : * Remember this value, so that, after looking up the relation name
487 : * and locking its OID, we can check whether any invalidation messages
488 : * have been processed that might require a do-over.
489 : */
490 652110 : inval_count = SharedInvalidMessageCounter;
491 :
492 : /*
493 : * Some non-default relpersistence value may have been specified. The
494 : * parser never generates such a RangeVar in simple DML, but it can
495 : * happen in contexts such as "CREATE TEMP TABLE foo (f1 int PRIMARY
496 : * KEY)". Such a command will generate an added CREATE INDEX
497 : * operation, which must be careful to find the temp table, even when
498 : * pg_temp is not first in the search path.
499 : */
500 652110 : if (relation->relpersistence == RELPERSISTENCE_TEMP)
501 : {
502 668 : if (!OidIsValid(myTempNamespace))
503 0 : relId = InvalidOid; /* this probably can't happen? */
504 : else
505 : {
506 668 : if (relation->schemaname)
507 : {
508 : Oid namespaceId;
509 :
510 12 : namespaceId = LookupExplicitNamespace(relation->schemaname, missing_ok);
511 :
512 : /*
513 : * For missing_ok, allow a non-existent schema name to
514 : * return InvalidOid.
515 : */
516 12 : if (namespaceId != myTempNamespace)
517 0 : ereport(ERROR,
518 : (errcode(ERRCODE_INVALID_TABLE_DEFINITION),
519 : errmsg("temporary tables cannot specify a schema name")));
520 : }
521 :
522 668 : relId = get_relname_relid(relation->relname, myTempNamespace);
523 : }
524 : }
525 651442 : else if (relation->schemaname)
526 : {
527 : Oid namespaceId;
528 :
529 : /* use exact schema given */
530 250808 : namespaceId = LookupExplicitNamespace(relation->schemaname, missing_ok);
531 250700 : if (missing_ok && !OidIsValid(namespaceId))
532 84 : relId = InvalidOid;
533 : else
534 250616 : relId = get_relname_relid(relation->relname, namespaceId);
535 : }
536 : else
537 : {
538 : /* search the namespace path */
539 400634 : relId = RelnameGetRelid(relation->relname);
540 : }
541 :
542 : /*
543 : * Invoke caller-supplied callback, if any.
544 : *
545 : * This callback is a good place to check permissions: we haven't
546 : * taken the table lock yet (and it's really best to check permissions
547 : * before locking anything!), but we've gotten far enough to know what
548 : * OID we think we should lock. Of course, concurrent DDL might
549 : * change things while we're waiting for the lock, but in that case
550 : * the callback will be invoked again for the new OID.
551 : */
552 652002 : if (callback)
553 82320 : callback(relation, relId, oldRelId, callback_arg);
554 :
555 : /*
556 : * If no lock requested, we assume the caller knows what they're
557 : * doing. They should have already acquired a heavyweight lock on
558 : * this relation earlier in the processing of this same statement, so
559 : * it wouldn't be appropriate to AcceptInvalidationMessages() here, as
560 : * that might pull the rug out from under them.
561 : */
562 651668 : if (lockmode == NoLock)
563 50638 : break;
564 :
565 : /*
566 : * If, upon retry, we get back the same OID we did last time, then the
567 : * invalidation messages we processed did not change the final answer.
568 : * So we're done.
569 : *
570 : * If we got a different OID, we've locked the relation that used to
571 : * have this name rather than the one that does now. So release the
572 : * lock.
573 : */
574 601030 : if (retry)
575 : {
576 3916 : if (relId == oldRelId)
577 3902 : break;
578 14 : if (OidIsValid(oldRelId))
579 14 : UnlockRelationOid(oldRelId, lockmode);
580 : }
581 :
582 : /*
583 : * Lock relation. This will also accept any pending invalidation
584 : * messages. If we got back InvalidOid, indicating not found, then
585 : * there's nothing to lock, but we accept invalidation messages
586 : * anyway, to flush any negative catcache entries that may be
587 : * lingering.
588 : */
589 597128 : if (!OidIsValid(relId))
590 1944 : AcceptInvalidationMessages();
591 595184 : else if (!(flags & (RVR_NOWAIT | RVR_SKIP_LOCKED)))
592 594560 : LockRelationOid(relId, lockmode);
593 624 : else if (!ConditionalLockRelationOid(relId, lockmode))
594 : {
595 16 : int elevel = (flags & RVR_SKIP_LOCKED) ? DEBUG1 : ERROR;
596 :
597 16 : if (relation->schemaname)
598 0 : ereport(elevel,
599 : (errcode(ERRCODE_LOCK_NOT_AVAILABLE),
600 : errmsg("could not obtain lock on relation \"%s.%s\"",
601 : relation->schemaname, relation->relname)));
602 : else
603 16 : ereport(elevel,
604 : (errcode(ERRCODE_LOCK_NOT_AVAILABLE),
605 : errmsg("could not obtain lock on relation \"%s\"",
606 : relation->relname)));
607 :
608 8 : return InvalidOid;
609 : }
610 :
611 : /*
612 : * If no invalidation message were processed, we're done!
613 : */
614 597096 : if (inval_count == SharedInvalidMessageCounter)
615 593180 : break;
616 :
617 : /*
618 : * Something may have changed. Let's repeat the name lookup, to make
619 : * sure this name still references the same relation it did
620 : * previously.
621 : */
622 3916 : retry = true;
623 3916 : oldRelId = relId;
624 : }
625 :
626 647720 : if (!OidIsValid(relId))
627 : {
628 2042 : int elevel = missing_ok ? DEBUG1 : ERROR;
629 :
630 2042 : if (relation->schemaname)
631 270 : ereport(elevel,
632 : (errcode(ERRCODE_UNDEFINED_TABLE),
633 : errmsg("relation \"%s.%s\" does not exist",
634 : relation->schemaname, relation->relname)));
635 : else
636 1772 : ereport(elevel,
637 : (errcode(ERRCODE_UNDEFINED_TABLE),
638 : errmsg("relation \"%s\" does not exist",
639 : relation->relname)));
640 : }
641 647268 : return relId;
642 : }
643 :
644 : /*
645 : * RangeVarGetCreationNamespace
646 : * Given a RangeVar describing a to-be-created relation,
647 : * choose which namespace to create it in.
648 : *
649 : * Note: calling this may result in a CommandCounterIncrement operation.
650 : * That will happen on the first request for a temp table in any particular
651 : * backend run; we will need to either create or clean out the temp schema.
652 : */
653 : Oid
654 117014 : RangeVarGetCreationNamespace(const RangeVar *newRelation)
655 : {
656 : Oid namespaceId;
657 :
658 : /*
659 : * We check the catalog name and then ignore it.
660 : */
661 117014 : if (newRelation->catalogname)
662 : {
663 0 : if (strcmp(newRelation->catalogname, get_database_name(MyDatabaseId)) != 0)
664 0 : ereport(ERROR,
665 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
666 : errmsg("cross-database references are not implemented: \"%s.%s.%s\"",
667 : newRelation->catalogname, newRelation->schemaname,
668 : newRelation->relname)));
669 : }
670 :
671 117014 : if (newRelation->schemaname)
672 : {
673 : /* check for pg_temp alias */
674 39932 : if (strcmp(newRelation->schemaname, "pg_temp") == 0)
675 : {
676 : /* Initialize temp namespace */
677 70 : AccessTempTableNamespace(false);
678 70 : return myTempNamespace;
679 : }
680 : /* use exact schema given */
681 39862 : namespaceId = get_namespace_oid(newRelation->schemaname, false);
682 : /* we do not check for USAGE rights here! */
683 : }
684 77082 : else if (newRelation->relpersistence == RELPERSISTENCE_TEMP)
685 : {
686 : /* Initialize temp namespace */
687 6202 : AccessTempTableNamespace(false);
688 6202 : return myTempNamespace;
689 : }
690 : else
691 : {
692 : /* use the default creation namespace */
693 70880 : recomputeNamespacePath();
694 70880 : if (activeTempCreationPending)
695 : {
696 : /* Need to initialize temp namespace */
697 0 : AccessTempTableNamespace(true);
698 0 : return myTempNamespace;
699 : }
700 70880 : namespaceId = activeCreationNamespace;
701 70880 : if (!OidIsValid(namespaceId))
702 0 : ereport(ERROR,
703 : (errcode(ERRCODE_UNDEFINED_SCHEMA),
704 : errmsg("no schema has been selected to create in")));
705 : }
706 :
707 : /* Note: callers will check for CREATE rights when appropriate */
708 :
709 110742 : return namespaceId;
710 : }
711 :
712 : /*
713 : * RangeVarGetAndCheckCreationNamespace
714 : *
715 : * This function returns the OID of the namespace in which a new relation
716 : * with a given name should be created. If the user does not have CREATE
717 : * permission on the target namespace, this function will instead signal
718 : * an ERROR.
719 : *
720 : * If non-NULL, *existing_relation_id is set to the OID of any existing relation
721 : * with the same name which already exists in that namespace, or to InvalidOid
722 : * if no such relation exists.
723 : *
724 : * If lockmode != NoLock, the specified lock mode is acquired on the existing
725 : * relation, if any, provided that the current user owns the target relation.
726 : * However, if lockmode != NoLock and the user does not own the target
727 : * relation, we throw an ERROR, as we must not try to lock relations the
728 : * user does not have permissions on.
729 : *
730 : * As a side effect, this function acquires AccessShareLock on the target
731 : * namespace. Without this, the namespace could be dropped before our
732 : * transaction commits, leaving behind relations with relnamespace pointing
733 : * to a no-longer-existent namespace.
734 : *
735 : * As a further side-effect, if the selected namespace is a temporary namespace,
736 : * we mark the RangeVar as RELPERSISTENCE_TEMP.
737 : */
738 : Oid
739 111920 : RangeVarGetAndCheckCreationNamespace(RangeVar *relation,
740 : LOCKMODE lockmode,
741 : Oid *existing_relation_id)
742 : {
743 : uint64 inval_count;
744 : Oid relid;
745 111920 : Oid oldrelid = InvalidOid;
746 : Oid nspid;
747 111920 : Oid oldnspid = InvalidOid;
748 111920 : bool retry = false;
749 :
750 : /*
751 : * We check the catalog name and then ignore it.
752 : */
753 111920 : if (relation->catalogname)
754 : {
755 0 : if (strcmp(relation->catalogname, get_database_name(MyDatabaseId)) != 0)
756 0 : ereport(ERROR,
757 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
758 : errmsg("cross-database references are not implemented: \"%s.%s.%s\"",
759 : relation->catalogname, relation->schemaname,
760 : relation->relname)));
761 : }
762 :
763 : /*
764 : * As in RangeVarGetRelidExtended(), we guard against concurrent DDL
765 : * operations by tracking whether any invalidation messages are processed
766 : * while we're doing the name lookups and acquiring locks. See comments
767 : * in that function for a more detailed explanation of this logic.
768 : */
769 : for (;;)
770 2222 : {
771 : AclResult aclresult;
772 :
773 114142 : inval_count = SharedInvalidMessageCounter;
774 :
775 : /* Look up creation namespace and check for existing relation. */
776 114142 : nspid = RangeVarGetCreationNamespace(relation);
777 : Assert(OidIsValid(nspid));
778 114142 : if (existing_relation_id != NULL)
779 52920 : relid = get_relname_relid(relation->relname, nspid);
780 : else
781 61222 : relid = InvalidOid;
782 :
783 : /*
784 : * In bootstrap processing mode, we don't bother with permissions or
785 : * locking. Permissions might not be working yet, and locking is
786 : * unnecessary.
787 : */
788 114142 : if (IsBootstrapProcessingMode())
789 0 : break;
790 :
791 : /* Check namespace permissions. */
792 114142 : aclresult = object_aclcheck(NamespaceRelationId, nspid, GetUserId(), ACL_CREATE);
793 114142 : if (aclresult != ACLCHECK_OK)
794 0 : aclcheck_error(aclresult, OBJECT_SCHEMA,
795 0 : get_namespace_name(nspid));
796 :
797 114142 : if (retry)
798 : {
799 : /* If nothing changed, we're done. */
800 2222 : if (relid == oldrelid && nspid == oldnspid)
801 2222 : break;
802 : /* If creation namespace has changed, give up old lock. */
803 0 : if (nspid != oldnspid)
804 0 : UnlockDatabaseObject(NamespaceRelationId, oldnspid, 0,
805 : AccessShareLock);
806 : /* If name points to something different, give up old lock. */
807 0 : if (relid != oldrelid && OidIsValid(oldrelid) && lockmode != NoLock)
808 0 : UnlockRelationOid(oldrelid, lockmode);
809 : }
810 :
811 : /* Lock namespace. */
812 111920 : if (nspid != oldnspid)
813 111920 : LockDatabaseObject(NamespaceRelationId, nspid, 0, AccessShareLock);
814 :
815 : /* Lock relation, if required if and we have permission. */
816 111920 : if (lockmode != NoLock && OidIsValid(relid))
817 : {
818 224 : if (!object_ownercheck(RelationRelationId, relid, GetUserId()))
819 0 : aclcheck_error(ACLCHECK_NOT_OWNER, get_relkind_objtype(get_rel_relkind(relid)),
820 0 : relation->relname);
821 224 : if (relid != oldrelid)
822 224 : LockRelationOid(relid, lockmode);
823 : }
824 :
825 : /* If no invalidation message were processed, we're done! */
826 111920 : if (inval_count == SharedInvalidMessageCounter)
827 109698 : break;
828 :
829 : /* Something may have changed, so recheck our work. */
830 2222 : retry = true;
831 2222 : oldrelid = relid;
832 2222 : oldnspid = nspid;
833 : }
834 :
835 111920 : RangeVarAdjustRelationPersistence(relation, nspid);
836 111896 : if (existing_relation_id != NULL)
837 51016 : *existing_relation_id = relid;
838 111896 : return nspid;
839 : }
840 :
841 : /*
842 : * Adjust the relpersistence for an about-to-be-created relation based on the
843 : * creation namespace, and throw an error for invalid combinations.
844 : */
845 : void
846 116400 : RangeVarAdjustRelationPersistence(RangeVar *newRelation, Oid nspid)
847 : {
848 116400 : switch (newRelation->relpersistence)
849 : {
850 5796 : case RELPERSISTENCE_TEMP:
851 5796 : if (!isTempOrTempToastNamespace(nspid))
852 : {
853 18 : if (isAnyTempNamespace(nspid))
854 0 : ereport(ERROR,
855 : (errcode(ERRCODE_INVALID_TABLE_DEFINITION),
856 : errmsg("cannot create relations in temporary schemas of other sessions")));
857 : else
858 18 : ereport(ERROR,
859 : (errcode(ERRCODE_INVALID_TABLE_DEFINITION),
860 : errmsg("cannot create temporary relation in non-temporary schema")));
861 : }
862 5778 : break;
863 110238 : case RELPERSISTENCE_PERMANENT:
864 110238 : if (isTempOrTempToastNamespace(nspid))
865 26 : newRelation->relpersistence = RELPERSISTENCE_TEMP;
866 110212 : else if (isAnyTempNamespace(nspid))
867 0 : ereport(ERROR,
868 : (errcode(ERRCODE_INVALID_TABLE_DEFINITION),
869 : errmsg("cannot create relations in temporary schemas of other sessions")));
870 110238 : break;
871 366 : default:
872 366 : if (isAnyTempNamespace(nspid))
873 6 : ereport(ERROR,
874 : (errcode(ERRCODE_INVALID_TABLE_DEFINITION),
875 : errmsg("only temporary relations may be created in temporary schemas")));
876 : }
877 116376 : }
878 :
879 : /*
880 : * RelnameGetRelid
881 : * Try to resolve an unqualified relation name.
882 : * Returns OID if relation found in search path, else InvalidOid.
883 : */
884 : Oid
885 400694 : RelnameGetRelid(const char *relname)
886 : {
887 : Oid relid;
888 : ListCell *l;
889 :
890 400694 : recomputeNamespacePath();
891 :
892 718756 : foreach(l, activeSearchPath)
893 : {
894 716944 : Oid namespaceId = lfirst_oid(l);
895 :
896 716944 : relid = get_relname_relid(relname, namespaceId);
897 716944 : if (OidIsValid(relid))
898 398882 : return relid;
899 : }
900 :
901 : /* Not found in path */
902 1812 : return InvalidOid;
903 : }
904 :
905 :
906 : /*
907 : * RelationIsVisible
908 : * Determine whether a relation (identified by OID) is visible in the
909 : * current search path. Visible means "would be found by searching
910 : * for the unqualified relation name".
911 : */
912 : bool
913 279796 : RelationIsVisible(Oid relid)
914 : {
915 279796 : return RelationIsVisibleExt(relid, NULL);
916 : }
917 :
918 : /*
919 : * RelationIsVisibleExt
920 : * As above, but if the relation isn't found and is_missing is not NULL,
921 : * then set *is_missing = true and return false instead of throwing
922 : * an error. (Caller must initialize *is_missing = false.)
923 : */
924 : static bool
925 300954 : RelationIsVisibleExt(Oid relid, bool *is_missing)
926 : {
927 : HeapTuple reltup;
928 : Form_pg_class relform;
929 : Oid relnamespace;
930 : bool visible;
931 :
932 300954 : reltup = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
933 300954 : if (!HeapTupleIsValid(reltup))
934 : {
935 6 : if (is_missing != NULL)
936 : {
937 6 : *is_missing = true;
938 6 : return false;
939 : }
940 0 : elog(ERROR, "cache lookup failed for relation %u", relid);
941 : }
942 300948 : relform = (Form_pg_class) GETSTRUCT(reltup);
943 :
944 300948 : recomputeNamespacePath();
945 :
946 : /*
947 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
948 : * the system namespace are surely in the path and so we needn't even do
949 : * list_member_oid() for them.
950 : */
951 300948 : relnamespace = relform->relnamespace;
952 300948 : if (relnamespace != PG_CATALOG_NAMESPACE &&
953 251002 : !list_member_oid(activeSearchPath, relnamespace))
954 106324 : visible = false;
955 : else
956 : {
957 : /*
958 : * If it is in the path, it might still not be visible; it could be
959 : * hidden by another relation of the same name earlier in the path. So
960 : * we must do a slow check for conflicting relations.
961 : */
962 194624 : char *relname = NameStr(relform->relname);
963 : ListCell *l;
964 :
965 194624 : visible = false;
966 472810 : foreach(l, activeSearchPath)
967 : {
968 472810 : Oid namespaceId = lfirst_oid(l);
969 :
970 472810 : if (namespaceId == relnamespace)
971 : {
972 : /* Found it first in path */
973 194564 : visible = true;
974 194564 : break;
975 : }
976 278246 : if (OidIsValid(get_relname_relid(relname, namespaceId)))
977 : {
978 : /* Found something else first in path */
979 60 : break;
980 : }
981 : }
982 : }
983 :
984 300948 : ReleaseSysCache(reltup);
985 :
986 300948 : return visible;
987 : }
988 :
989 :
990 : /*
991 : * TypenameGetTypid
992 : * Wrapper for binary compatibility.
993 : */
994 : Oid
995 0 : TypenameGetTypid(const char *typname)
996 : {
997 0 : return TypenameGetTypidExtended(typname, true);
998 : }
999 :
1000 : /*
1001 : * TypenameGetTypidExtended
1002 : * Try to resolve an unqualified datatype name.
1003 : * Returns OID if type found in search path, else InvalidOid.
1004 : *
1005 : * This is essentially the same as RelnameGetRelid.
1006 : */
1007 : Oid
1008 330558 : TypenameGetTypidExtended(const char *typname, bool temp_ok)
1009 : {
1010 : Oid typid;
1011 : ListCell *l;
1012 :
1013 330558 : recomputeNamespacePath();
1014 :
1015 550910 : foreach(l, activeSearchPath)
1016 : {
1017 502624 : Oid namespaceId = lfirst_oid(l);
1018 :
1019 502624 : if (!temp_ok && namespaceId == myTempNamespace)
1020 6692 : continue; /* do not look in temp namespace */
1021 :
1022 495932 : typid = GetSysCacheOid2(TYPENAMENSP, Anum_pg_type_oid,
1023 : PointerGetDatum(typname),
1024 : ObjectIdGetDatum(namespaceId));
1025 495932 : if (OidIsValid(typid))
1026 282272 : return typid;
1027 : }
1028 :
1029 : /* Not found in path */
1030 48286 : return InvalidOid;
1031 : }
1032 :
1033 : /*
1034 : * TypeIsVisible
1035 : * Determine whether a type (identified by OID) is visible in the
1036 : * current search path. Visible means "would be found by searching
1037 : * for the unqualified type name".
1038 : */
1039 : bool
1040 469856 : TypeIsVisible(Oid typid)
1041 : {
1042 469856 : return TypeIsVisibleExt(typid, NULL);
1043 : }
1044 :
1045 : /*
1046 : * TypeIsVisibleExt
1047 : * As above, but if the type isn't found and is_missing is not NULL,
1048 : * then set *is_missing = true and return false instead of throwing
1049 : * an error. (Caller must initialize *is_missing = false.)
1050 : */
1051 : static bool
1052 473798 : TypeIsVisibleExt(Oid typid, bool *is_missing)
1053 : {
1054 : HeapTuple typtup;
1055 : Form_pg_type typform;
1056 : Oid typnamespace;
1057 : bool visible;
1058 :
1059 473798 : typtup = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
1060 473798 : if (!HeapTupleIsValid(typtup))
1061 : {
1062 0 : if (is_missing != NULL)
1063 : {
1064 0 : *is_missing = true;
1065 0 : return false;
1066 : }
1067 0 : elog(ERROR, "cache lookup failed for type %u", typid);
1068 : }
1069 473798 : typform = (Form_pg_type) GETSTRUCT(typtup);
1070 :
1071 473798 : recomputeNamespacePath();
1072 :
1073 : /*
1074 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
1075 : * the system namespace are surely in the path and so we needn't even do
1076 : * list_member_oid() for them.
1077 : */
1078 473798 : typnamespace = typform->typnamespace;
1079 473798 : if (typnamespace != PG_CATALOG_NAMESPACE &&
1080 97300 : !list_member_oid(activeSearchPath, typnamespace))
1081 19188 : visible = false;
1082 : else
1083 : {
1084 : /*
1085 : * If it is in the path, it might still not be visible; it could be
1086 : * hidden by another type of the same name earlier in the path. So we
1087 : * must do a slow check for conflicting types.
1088 : */
1089 454610 : char *typname = NameStr(typform->typname);
1090 : ListCell *l;
1091 :
1092 454610 : visible = false;
1093 581686 : foreach(l, activeSearchPath)
1094 : {
1095 581686 : Oid namespaceId = lfirst_oid(l);
1096 :
1097 581686 : if (namespaceId == typnamespace)
1098 : {
1099 : /* Found it first in path */
1100 454598 : visible = true;
1101 454598 : break;
1102 : }
1103 127088 : if (SearchSysCacheExists2(TYPENAMENSP,
1104 : PointerGetDatum(typname),
1105 : ObjectIdGetDatum(namespaceId)))
1106 : {
1107 : /* Found something else first in path */
1108 12 : break;
1109 : }
1110 : }
1111 : }
1112 :
1113 473798 : ReleaseSysCache(typtup);
1114 :
1115 473798 : return visible;
1116 : }
1117 :
1118 :
1119 : /*
1120 : * FuncnameGetCandidates
1121 : * Given a possibly-qualified function name and argument count,
1122 : * retrieve a list of the possible matches.
1123 : *
1124 : * If nargs is -1, we return all functions matching the given name,
1125 : * regardless of argument count. (argnames must be NIL, and expand_variadic
1126 : * and expand_defaults must be false, in this case.)
1127 : *
1128 : * If argnames isn't NIL, we are considering a named- or mixed-notation call,
1129 : * and only functions having all the listed argument names will be returned.
1130 : * (We assume that length(argnames) <= nargs and all the passed-in names are
1131 : * distinct.) The returned structs will include an argnumbers array showing
1132 : * the actual argument index for each logical argument position.
1133 : *
1134 : * If expand_variadic is true, then variadic functions having the same number
1135 : * or fewer arguments will be retrieved, with the variadic argument and any
1136 : * additional argument positions filled with the variadic element type.
1137 : * nvargs in the returned struct is set to the number of such arguments.
1138 : * If expand_variadic is false, variadic arguments are not treated specially,
1139 : * and the returned nvargs will always be zero.
1140 : *
1141 : * If expand_defaults is true, functions that could match after insertion of
1142 : * default argument values will also be retrieved. In this case the returned
1143 : * structs could have nargs > passed-in nargs, and ndargs is set to the number
1144 : * of additional args (which can be retrieved from the function's
1145 : * proargdefaults entry).
1146 : *
1147 : * If include_out_arguments is true, then OUT-mode arguments are considered to
1148 : * be included in the argument list. Their types are included in the returned
1149 : * arrays, and argnumbers are indexes in proallargtypes not proargtypes.
1150 : * We also set nominalnargs to be the length of proallargtypes not proargtypes.
1151 : * Otherwise OUT-mode arguments are ignored.
1152 : *
1153 : * It is not possible for nvargs and ndargs to both be nonzero in the same
1154 : * list entry, since default insertion allows matches to functions with more
1155 : * than nargs arguments while the variadic transformation requires the same
1156 : * number or less.
1157 : *
1158 : * When argnames isn't NIL, the returned args[] type arrays are not ordered
1159 : * according to the functions' declarations, but rather according to the call:
1160 : * first any positional arguments, then the named arguments, then defaulted
1161 : * arguments (if needed and allowed by expand_defaults). The argnumbers[]
1162 : * array can be used to map this back to the catalog information.
1163 : * argnumbers[k] is set to the proargtypes or proallargtypes index of the
1164 : * k'th call argument.
1165 : *
1166 : * We search a single namespace if the function name is qualified, else
1167 : * all namespaces in the search path. In the multiple-namespace case,
1168 : * we arrange for entries in earlier namespaces to mask identical entries in
1169 : * later namespaces.
1170 : *
1171 : * When expanding variadics, we arrange for non-variadic functions to mask
1172 : * variadic ones if the expanded argument list is the same. It is still
1173 : * possible for there to be conflicts between different variadic functions,
1174 : * however.
1175 : *
1176 : * It is guaranteed that the return list will never contain multiple entries
1177 : * with identical argument lists. When expand_defaults is true, the entries
1178 : * could have more than nargs positions, but we still guarantee that they are
1179 : * distinct in the first nargs positions. However, if argnames isn't NIL or
1180 : * either expand_variadic or expand_defaults is true, there might be multiple
1181 : * candidate functions that expand to identical argument lists. Rather than
1182 : * throw error here, we report such situations by returning a single entry
1183 : * with oid = 0 that represents a set of such conflicting candidates.
1184 : * The caller might end up discarding such an entry anyway, but if it selects
1185 : * such an entry it should react as though the call were ambiguous.
1186 : *
1187 : * If missing_ok is true, an empty list (NULL) is returned if the name was
1188 : * schema-qualified with a schema that does not exist. Likewise if no
1189 : * candidate is found for other reasons.
1190 : */
1191 : FuncCandidateList
1192 532278 : FuncnameGetCandidates(List *names, int nargs, List *argnames,
1193 : bool expand_variadic, bool expand_defaults,
1194 : bool include_out_arguments, bool missing_ok)
1195 : {
1196 532278 : FuncCandidateList resultList = NULL;
1197 532278 : bool any_special = false;
1198 : char *schemaname;
1199 : char *funcname;
1200 : Oid namespaceId;
1201 : CatCList *catlist;
1202 : int i;
1203 :
1204 : /* check for caller error */
1205 : Assert(nargs >= 0 || !(expand_variadic | expand_defaults));
1206 :
1207 : /* deconstruct the name list */
1208 532278 : DeconstructQualifiedName(names, &schemaname, &funcname);
1209 :
1210 532242 : if (schemaname)
1211 : {
1212 : /* use exact schema given */
1213 131458 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
1214 131458 : if (!OidIsValid(namespaceId))
1215 48 : return NULL;
1216 : }
1217 : else
1218 : {
1219 : /* flag to indicate we need namespace search */
1220 400784 : namespaceId = InvalidOid;
1221 400784 : recomputeNamespacePath();
1222 : }
1223 :
1224 : /* Search syscache by name only */
1225 532194 : catlist = SearchSysCacheList1(PROCNAMEARGSNSP, CStringGetDatum(funcname));
1226 :
1227 1648990 : for (i = 0; i < catlist->n_members; i++)
1228 : {
1229 1116796 : HeapTuple proctup = &catlist->members[i]->tuple;
1230 1116796 : Form_pg_proc procform = (Form_pg_proc) GETSTRUCT(proctup);
1231 1116796 : Oid *proargtypes = procform->proargtypes.values;
1232 1116796 : int pronargs = procform->pronargs;
1233 : int effective_nargs;
1234 1116796 : int pathpos = 0;
1235 : bool variadic;
1236 : bool use_defaults;
1237 : Oid va_elem_type;
1238 1116796 : int *argnumbers = NULL;
1239 : FuncCandidateList newResult;
1240 :
1241 1116796 : if (OidIsValid(namespaceId))
1242 : {
1243 : /* Consider only procs in specified namespace */
1244 258786 : if (procform->pronamespace != namespaceId)
1245 283200 : continue;
1246 : }
1247 : else
1248 : {
1249 : /*
1250 : * Consider only procs that are in the search path and are not in
1251 : * the temp namespace.
1252 : */
1253 : ListCell *nsp;
1254 :
1255 1107942 : foreach(nsp, activeSearchPath)
1256 : {
1257 1105062 : if (procform->pronamespace == lfirst_oid(nsp) &&
1258 855166 : procform->pronamespace != myTempNamespace)
1259 855130 : break;
1260 249932 : pathpos++;
1261 : }
1262 858010 : if (nsp == NULL)
1263 2880 : continue; /* proc is not in search path */
1264 : }
1265 :
1266 : /*
1267 : * If we are asked to match to OUT arguments, then use the
1268 : * proallargtypes array (which includes those); otherwise use
1269 : * proargtypes (which doesn't). Of course, if proallargtypes is null,
1270 : * we always use proargtypes.
1271 : */
1272 1104152 : if (include_out_arguments)
1273 : {
1274 : Datum proallargtypes;
1275 : bool isNull;
1276 :
1277 696 : proallargtypes = SysCacheGetAttr(PROCNAMEARGSNSP, proctup,
1278 : Anum_pg_proc_proallargtypes,
1279 : &isNull);
1280 696 : if (!isNull)
1281 : {
1282 226 : ArrayType *arr = DatumGetArrayTypeP(proallargtypes);
1283 :
1284 226 : pronargs = ARR_DIMS(arr)[0];
1285 226 : if (ARR_NDIM(arr) != 1 ||
1286 226 : pronargs < 0 ||
1287 226 : ARR_HASNULL(arr) ||
1288 226 : ARR_ELEMTYPE(arr) != OIDOID)
1289 0 : elog(ERROR, "proallargtypes is not a 1-D Oid array or it contains nulls");
1290 : Assert(pronargs >= procform->pronargs);
1291 226 : proargtypes = (Oid *) ARR_DATA_PTR(arr);
1292 : }
1293 : }
1294 :
1295 1104152 : if (argnames != NIL)
1296 : {
1297 : /*
1298 : * Call uses named or mixed notation
1299 : *
1300 : * Named or mixed notation can match a variadic function only if
1301 : * expand_variadic is off; otherwise there is no way to match the
1302 : * presumed-nameless parameters expanded from the variadic array.
1303 : */
1304 32054 : if (OidIsValid(procform->provariadic) && expand_variadic)
1305 0 : continue;
1306 32054 : va_elem_type = InvalidOid;
1307 32054 : variadic = false;
1308 :
1309 : /*
1310 : * Check argument count.
1311 : */
1312 : Assert(nargs >= 0); /* -1 not supported with argnames */
1313 :
1314 32054 : if (pronargs > nargs && expand_defaults)
1315 : {
1316 : /* Ignore if not enough default expressions */
1317 13982 : if (nargs + procform->pronargdefaults < pronargs)
1318 6506 : continue;
1319 7476 : use_defaults = true;
1320 : }
1321 : else
1322 18072 : use_defaults = false;
1323 :
1324 : /* Ignore if it doesn't match requested argument count */
1325 25548 : if (pronargs != nargs && !use_defaults)
1326 9430 : continue;
1327 :
1328 : /* Check for argument name match, generate positional mapping */
1329 16118 : if (!MatchNamedCall(proctup, nargs, argnames,
1330 : include_out_arguments, pronargs,
1331 : &argnumbers))
1332 18 : continue;
1333 :
1334 : /* Named argument matching is always "special" */
1335 16100 : any_special = true;
1336 : }
1337 : else
1338 : {
1339 : /*
1340 : * Call uses positional notation
1341 : *
1342 : * Check if function is variadic, and get variadic element type if
1343 : * so. If expand_variadic is false, we should just ignore
1344 : * variadic-ness.
1345 : */
1346 1072098 : if (pronargs <= nargs && expand_variadic)
1347 : {
1348 665420 : va_elem_type = procform->provariadic;
1349 665420 : variadic = OidIsValid(va_elem_type);
1350 665420 : any_special |= variadic;
1351 : }
1352 : else
1353 : {
1354 406678 : va_elem_type = InvalidOid;
1355 406678 : variadic = false;
1356 : }
1357 :
1358 : /*
1359 : * Check if function can match by using parameter defaults.
1360 : */
1361 1072098 : if (pronargs > nargs && expand_defaults)
1362 : {
1363 : /* Ignore if not enough default expressions */
1364 197180 : if (nargs + procform->pronargdefaults < pronargs)
1365 190988 : continue;
1366 6192 : use_defaults = true;
1367 6192 : any_special = true;
1368 : }
1369 : else
1370 874918 : use_defaults = false;
1371 :
1372 : /* Ignore if it doesn't match requested argument count */
1373 881110 : if (nargs >= 0 && pronargs != nargs && !variadic && !use_defaults)
1374 61362 : continue;
1375 : }
1376 :
1377 : /*
1378 : * We must compute the effective argument list so that we can easily
1379 : * compare it to earlier results. We waste a palloc cycle if it gets
1380 : * masked by an earlier result, but really that's a pretty infrequent
1381 : * case so it's not worth worrying about.
1382 : */
1383 835848 : effective_nargs = Max(pronargs, nargs);
1384 : newResult = (FuncCandidateList)
1385 835848 : palloc(offsetof(struct _FuncCandidateList, args) +
1386 : effective_nargs * sizeof(Oid));
1387 835848 : newResult->pathpos = pathpos;
1388 835848 : newResult->oid = procform->oid;
1389 835848 : newResult->nominalnargs = pronargs;
1390 835848 : newResult->nargs = effective_nargs;
1391 835848 : newResult->argnumbers = argnumbers;
1392 835848 : if (argnumbers)
1393 : {
1394 : /* Re-order the argument types into call's logical order */
1395 82208 : for (int j = 0; j < pronargs; j++)
1396 66108 : newResult->args[j] = proargtypes[argnumbers[j]];
1397 : }
1398 : else
1399 : {
1400 : /* Simple positional case, just copy proargtypes as-is */
1401 819748 : memcpy(newResult->args, proargtypes, pronargs * sizeof(Oid));
1402 : }
1403 835848 : if (variadic)
1404 : {
1405 7890 : newResult->nvargs = effective_nargs - pronargs + 1;
1406 : /* Expand variadic argument into N copies of element type */
1407 24140 : for (int j = pronargs - 1; j < effective_nargs; j++)
1408 16250 : newResult->args[j] = va_elem_type;
1409 : }
1410 : else
1411 827958 : newResult->nvargs = 0;
1412 835848 : newResult->ndargs = use_defaults ? pronargs - nargs : 0;
1413 :
1414 : /*
1415 : * Does it have the same arguments as something we already accepted?
1416 : * If so, decide what to do to avoid returning duplicate argument
1417 : * lists. We can skip this check for the single-namespace case if no
1418 : * special (named, variadic or defaults) match has been made, since
1419 : * then the unique index on pg_proc guarantees all the matches have
1420 : * different argument lists.
1421 : */
1422 835848 : if (resultList != NULL &&
1423 306202 : (any_special || !OidIsValid(namespaceId)))
1424 : {
1425 : /*
1426 : * If we have an ordered list from SearchSysCacheList (the normal
1427 : * case), then any conflicting proc must immediately adjoin this
1428 : * one in the list, so we only need to look at the newest result
1429 : * item. If we have an unordered list, we have to scan the whole
1430 : * result list. Also, if either the current candidate or any
1431 : * previous candidate is a special match, we can't assume that
1432 : * conflicts are adjacent.
1433 : *
1434 : * We ignore defaulted arguments in deciding what is a match.
1435 : */
1436 : FuncCandidateList prevResult;
1437 :
1438 270164 : if (catlist->ordered && !any_special)
1439 : {
1440 : /* ndargs must be 0 if !any_special */
1441 267582 : if (effective_nargs == resultList->nargs &&
1442 267462 : memcmp(newResult->args,
1443 267462 : resultList->args,
1444 : effective_nargs * sizeof(Oid)) == 0)
1445 8 : prevResult = resultList;
1446 : else
1447 267574 : prevResult = NULL;
1448 : }
1449 : else
1450 : {
1451 2582 : int cmp_nargs = newResult->nargs - newResult->ndargs;
1452 :
1453 2842 : for (prevResult = resultList;
1454 : prevResult;
1455 260 : prevResult = prevResult->next)
1456 : {
1457 2582 : if (cmp_nargs == prevResult->nargs - prevResult->ndargs &&
1458 2582 : memcmp(newResult->args,
1459 2582 : prevResult->args,
1460 : cmp_nargs * sizeof(Oid)) == 0)
1461 2322 : break;
1462 : }
1463 : }
1464 :
1465 270164 : if (prevResult)
1466 : {
1467 : /*
1468 : * We have a match with a previous result. Decide which one
1469 : * to keep, or mark it ambiguous if we can't decide. The
1470 : * logic here is preference > 0 means prefer the old result,
1471 : * preference < 0 means prefer the new, preference = 0 means
1472 : * ambiguous.
1473 : */
1474 : int preference;
1475 :
1476 2330 : if (pathpos != prevResult->pathpos)
1477 : {
1478 : /*
1479 : * Prefer the one that's earlier in the search path.
1480 : */
1481 2 : preference = pathpos - prevResult->pathpos;
1482 : }
1483 2328 : else if (variadic && prevResult->nvargs == 0)
1484 : {
1485 : /*
1486 : * With variadic functions we could have, for example,
1487 : * both foo(numeric) and foo(variadic numeric[]) in the
1488 : * same namespace; if so we prefer the non-variadic match
1489 : * on efficiency grounds.
1490 : */
1491 2208 : preference = 1;
1492 : }
1493 120 : else if (!variadic && prevResult->nvargs > 0)
1494 : {
1495 78 : preference = -1;
1496 : }
1497 : else
1498 : {
1499 : /*----------
1500 : * We can't decide. This can happen with, for example,
1501 : * both foo(numeric, variadic numeric[]) and
1502 : * foo(variadic numeric[]) in the same namespace, or
1503 : * both foo(int) and foo (int, int default something)
1504 : * in the same namespace, or both foo(a int, b text)
1505 : * and foo(b text, a int) in the same namespace.
1506 : *----------
1507 : */
1508 42 : preference = 0;
1509 : }
1510 :
1511 2330 : if (preference > 0)
1512 : {
1513 : /* keep previous result */
1514 2210 : pfree(newResult);
1515 2210 : continue;
1516 : }
1517 120 : else if (preference < 0)
1518 : {
1519 : /* remove previous result from the list */
1520 78 : if (prevResult == resultList)
1521 78 : resultList = prevResult->next;
1522 : else
1523 : {
1524 : FuncCandidateList prevPrevResult;
1525 :
1526 0 : for (prevPrevResult = resultList;
1527 : prevPrevResult;
1528 0 : prevPrevResult = prevPrevResult->next)
1529 : {
1530 0 : if (prevResult == prevPrevResult->next)
1531 : {
1532 0 : prevPrevResult->next = prevResult->next;
1533 0 : break;
1534 : }
1535 : }
1536 : Assert(prevPrevResult); /* assert we found it */
1537 : }
1538 78 : pfree(prevResult);
1539 : /* fall through to add newResult to list */
1540 : }
1541 : else
1542 : {
1543 : /* mark old result as ambiguous, discard new */
1544 42 : prevResult->oid = InvalidOid;
1545 42 : pfree(newResult);
1546 42 : continue;
1547 : }
1548 : }
1549 : }
1550 :
1551 : /*
1552 : * Okay to add it to result list
1553 : */
1554 833596 : newResult->next = resultList;
1555 833596 : resultList = newResult;
1556 : }
1557 :
1558 532194 : ReleaseSysCacheList(catlist);
1559 :
1560 532194 : return resultList;
1561 : }
1562 :
1563 : /*
1564 : * MatchNamedCall
1565 : * Given a pg_proc heap tuple and a call's list of argument names,
1566 : * check whether the function could match the call.
1567 : *
1568 : * The call could match if all supplied argument names are accepted by
1569 : * the function, in positions after the last positional argument, and there
1570 : * are defaults for all unsupplied arguments.
1571 : *
1572 : * If include_out_arguments is true, we are treating OUT arguments as
1573 : * included in the argument list. pronargs is the number of arguments
1574 : * we're considering (the length of either proargtypes or proallargtypes).
1575 : *
1576 : * The number of positional arguments is nargs - list_length(argnames).
1577 : * Note caller has already done basic checks on argument count.
1578 : *
1579 : * On match, return true and fill *argnumbers with a palloc'd array showing
1580 : * the mapping from call argument positions to actual function argument
1581 : * numbers. Defaulted arguments are included in this map, at positions
1582 : * after the last supplied argument.
1583 : */
1584 : static bool
1585 16118 : MatchNamedCall(HeapTuple proctup, int nargs, List *argnames,
1586 : bool include_out_arguments, int pronargs,
1587 : int **argnumbers)
1588 : {
1589 16118 : Form_pg_proc procform = (Form_pg_proc) GETSTRUCT(proctup);
1590 16118 : int numposargs = nargs - list_length(argnames);
1591 : int pronallargs;
1592 : Oid *p_argtypes;
1593 : char **p_argnames;
1594 : char *p_argmodes;
1595 : bool arggiven[FUNC_MAX_ARGS];
1596 : bool isnull;
1597 : int ap; /* call args position */
1598 : int pp; /* proargs position */
1599 : ListCell *lc;
1600 :
1601 : Assert(argnames != NIL);
1602 : Assert(numposargs >= 0);
1603 : Assert(nargs <= pronargs);
1604 :
1605 : /* Ignore this function if its proargnames is null */
1606 16118 : (void) SysCacheGetAttr(PROCOID, proctup, Anum_pg_proc_proargnames,
1607 : &isnull);
1608 16118 : if (isnull)
1609 0 : return false;
1610 :
1611 : /* OK, let's extract the argument names and types */
1612 16118 : pronallargs = get_func_arg_info(proctup,
1613 : &p_argtypes, &p_argnames, &p_argmodes);
1614 : Assert(p_argnames != NULL);
1615 :
1616 : Assert(include_out_arguments ? (pronargs == pronallargs) : (pronargs <= pronallargs));
1617 :
1618 : /* initialize state for matching */
1619 16118 : *argnumbers = (int *) palloc(pronargs * sizeof(int));
1620 16118 : memset(arggiven, false, pronargs * sizeof(bool));
1621 :
1622 : /* there are numposargs positional args before the named args */
1623 18010 : for (ap = 0; ap < numposargs; ap++)
1624 : {
1625 1892 : (*argnumbers)[ap] = ap;
1626 1892 : arggiven[ap] = true;
1627 : }
1628 :
1629 : /* now examine the named args */
1630 64384 : foreach(lc, argnames)
1631 : {
1632 48278 : char *argname = (char *) lfirst(lc);
1633 : bool found;
1634 : int i;
1635 :
1636 48278 : pp = 0;
1637 48278 : found = false;
1638 114010 : for (i = 0; i < pronallargs; i++)
1639 : {
1640 : /* consider only input params, except with include_out_arguments */
1641 114004 : if (!include_out_arguments &&
1642 73782 : p_argmodes &&
1643 73782 : (p_argmodes[i] != FUNC_PARAM_IN &&
1644 48 : p_argmodes[i] != FUNC_PARAM_INOUT &&
1645 48 : p_argmodes[i] != FUNC_PARAM_VARIADIC))
1646 48 : continue;
1647 113956 : if (p_argnames[i] && strcmp(p_argnames[i], argname) == 0)
1648 : {
1649 : /* fail if argname matches a positional argument */
1650 48272 : if (arggiven[pp])
1651 12 : return false;
1652 48266 : arggiven[pp] = true;
1653 48266 : (*argnumbers)[ap] = pp;
1654 48266 : found = true;
1655 48266 : break;
1656 : }
1657 : /* increase pp only for considered parameters */
1658 65684 : pp++;
1659 : }
1660 : /* if name isn't in proargnames, fail */
1661 48272 : if (!found)
1662 6 : return false;
1663 48266 : ap++;
1664 : }
1665 :
1666 : Assert(ap == nargs); /* processed all actual parameters */
1667 :
1668 : /* Check for default arguments */
1669 16106 : if (nargs < pronargs)
1670 : {
1671 7464 : int first_arg_with_default = pronargs - procform->pronargdefaults;
1672 :
1673 51286 : for (pp = numposargs; pp < pronargs; pp++)
1674 : {
1675 43828 : if (arggiven[pp])
1676 27842 : continue;
1677 : /* fail if arg not given and no default available */
1678 15986 : if (pp < first_arg_with_default)
1679 6 : return false;
1680 15980 : (*argnumbers)[ap++] = pp;
1681 : }
1682 : }
1683 :
1684 : Assert(ap == pronargs); /* processed all function parameters */
1685 :
1686 16100 : return true;
1687 : }
1688 :
1689 : /*
1690 : * FunctionIsVisible
1691 : * Determine whether a function (identified by OID) is visible in the
1692 : * current search path. Visible means "would be found by searching
1693 : * for the unqualified function name with exact argument matches".
1694 : */
1695 : bool
1696 25326 : FunctionIsVisible(Oid funcid)
1697 : {
1698 25326 : return FunctionIsVisibleExt(funcid, NULL);
1699 : }
1700 :
1701 : /*
1702 : * FunctionIsVisibleExt
1703 : * As above, but if the function isn't found and is_missing is not NULL,
1704 : * then set *is_missing = true and return false instead of throwing
1705 : * an error. (Caller must initialize *is_missing = false.)
1706 : */
1707 : static bool
1708 32828 : FunctionIsVisibleExt(Oid funcid, bool *is_missing)
1709 : {
1710 : HeapTuple proctup;
1711 : Form_pg_proc procform;
1712 : Oid pronamespace;
1713 : bool visible;
1714 :
1715 32828 : proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1716 32828 : if (!HeapTupleIsValid(proctup))
1717 : {
1718 0 : if (is_missing != NULL)
1719 : {
1720 0 : *is_missing = true;
1721 0 : return false;
1722 : }
1723 0 : elog(ERROR, "cache lookup failed for function %u", funcid);
1724 : }
1725 32828 : procform = (Form_pg_proc) GETSTRUCT(proctup);
1726 :
1727 32828 : recomputeNamespacePath();
1728 :
1729 : /*
1730 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
1731 : * the system namespace are surely in the path and so we needn't even do
1732 : * list_member_oid() for them.
1733 : */
1734 32828 : pronamespace = procform->pronamespace;
1735 32828 : if (pronamespace != PG_CATALOG_NAMESPACE &&
1736 11798 : !list_member_oid(activeSearchPath, pronamespace))
1737 796 : visible = false;
1738 : else
1739 : {
1740 : /*
1741 : * If it is in the path, it might still not be visible; it could be
1742 : * hidden by another proc of the same name and arguments earlier in
1743 : * the path. So we must do a slow check to see if this is the same
1744 : * proc that would be found by FuncnameGetCandidates.
1745 : */
1746 32032 : char *proname = NameStr(procform->proname);
1747 32032 : int nargs = procform->pronargs;
1748 : FuncCandidateList clist;
1749 :
1750 32032 : visible = false;
1751 :
1752 32032 : clist = FuncnameGetCandidates(list_make1(makeString(proname)),
1753 : nargs, NIL, false, false, false, false);
1754 :
1755 41112 : for (; clist; clist = clist->next)
1756 : {
1757 41100 : if (memcmp(clist->args, procform->proargtypes.values,
1758 : nargs * sizeof(Oid)) == 0)
1759 : {
1760 : /* Found the expected entry; is it the right proc? */
1761 32020 : visible = (clist->oid == funcid);
1762 32020 : break;
1763 : }
1764 : }
1765 : }
1766 :
1767 32828 : ReleaseSysCache(proctup);
1768 :
1769 32828 : return visible;
1770 : }
1771 :
1772 :
1773 : /*
1774 : * OpernameGetOprid
1775 : * Given a possibly-qualified operator name and exact input datatypes,
1776 : * look up the operator. Returns InvalidOid if not found.
1777 : *
1778 : * Pass oprleft = InvalidOid for a prefix op.
1779 : *
1780 : * If the operator name is not schema-qualified, it is sought in the current
1781 : * namespace search path. If the name is schema-qualified and the given
1782 : * schema does not exist, InvalidOid is returned.
1783 : */
1784 : Oid
1785 86254 : OpernameGetOprid(List *names, Oid oprleft, Oid oprright)
1786 : {
1787 : char *schemaname;
1788 : char *opername;
1789 : CatCList *catlist;
1790 : ListCell *l;
1791 :
1792 : /* deconstruct the name list */
1793 86254 : DeconstructQualifiedName(names, &schemaname, &opername);
1794 :
1795 86254 : if (schemaname)
1796 : {
1797 : /* search only in exact schema given */
1798 : Oid namespaceId;
1799 :
1800 2494 : namespaceId = LookupExplicitNamespace(schemaname, true);
1801 2494 : if (OidIsValid(namespaceId))
1802 : {
1803 : HeapTuple opertup;
1804 :
1805 2470 : opertup = SearchSysCache4(OPERNAMENSP,
1806 : CStringGetDatum(opername),
1807 : ObjectIdGetDatum(oprleft),
1808 : ObjectIdGetDatum(oprright),
1809 : ObjectIdGetDatum(namespaceId));
1810 2470 : if (HeapTupleIsValid(opertup))
1811 : {
1812 1406 : Form_pg_operator operclass = (Form_pg_operator) GETSTRUCT(opertup);
1813 1406 : Oid result = operclass->oid;
1814 :
1815 1406 : ReleaseSysCache(opertup);
1816 1406 : return result;
1817 : }
1818 : }
1819 :
1820 1088 : return InvalidOid;
1821 : }
1822 :
1823 : /* Search syscache by name and argument types */
1824 83760 : catlist = SearchSysCacheList3(OPERNAMENSP,
1825 : CStringGetDatum(opername),
1826 : ObjectIdGetDatum(oprleft),
1827 : ObjectIdGetDatum(oprright));
1828 :
1829 83760 : if (catlist->n_members == 0)
1830 : {
1831 : /* no hope, fall out early */
1832 16198 : ReleaseSysCacheList(catlist);
1833 16198 : return InvalidOid;
1834 : }
1835 :
1836 : /*
1837 : * We have to find the list member that is first in the search path, if
1838 : * there's more than one. This doubly-nested loop looks ugly, but in
1839 : * practice there should usually be few catlist members.
1840 : */
1841 67562 : recomputeNamespacePath();
1842 :
1843 78942 : foreach(l, activeSearchPath)
1844 : {
1845 78930 : Oid namespaceId = lfirst_oid(l);
1846 : int i;
1847 :
1848 78930 : if (namespaceId == myTempNamespace)
1849 7070 : continue; /* do not look in temp namespace */
1850 :
1851 76206 : for (i = 0; i < catlist->n_members; i++)
1852 : {
1853 71896 : HeapTuple opertup = &catlist->members[i]->tuple;
1854 71896 : Form_pg_operator operform = (Form_pg_operator) GETSTRUCT(opertup);
1855 :
1856 71896 : if (operform->oprnamespace == namespaceId)
1857 : {
1858 67550 : Oid result = operform->oid;
1859 :
1860 67550 : ReleaseSysCacheList(catlist);
1861 67550 : return result;
1862 : }
1863 : }
1864 : }
1865 :
1866 12 : ReleaseSysCacheList(catlist);
1867 12 : return InvalidOid;
1868 : }
1869 :
1870 : /*
1871 : * OpernameGetCandidates
1872 : * Given a possibly-qualified operator name and operator kind,
1873 : * retrieve a list of the possible matches.
1874 : *
1875 : * If oprkind is '\0', we return all operators matching the given name,
1876 : * regardless of arguments.
1877 : *
1878 : * We search a single namespace if the operator name is qualified, else
1879 : * all namespaces in the search path. The return list will never contain
1880 : * multiple entries with identical argument lists --- in the multiple-
1881 : * namespace case, we arrange for entries in earlier namespaces to mask
1882 : * identical entries in later namespaces.
1883 : *
1884 : * The returned items always have two args[] entries --- the first will be
1885 : * InvalidOid for a prefix oprkind. nargs is always 2, too.
1886 : */
1887 : FuncCandidateList
1888 16256 : OpernameGetCandidates(List *names, char oprkind, bool missing_schema_ok)
1889 : {
1890 16256 : FuncCandidateList resultList = NULL;
1891 16256 : char *resultSpace = NULL;
1892 16256 : int nextResult = 0;
1893 : char *schemaname;
1894 : char *opername;
1895 : Oid namespaceId;
1896 : CatCList *catlist;
1897 : int i;
1898 :
1899 : /* deconstruct the name list */
1900 16256 : DeconstructQualifiedName(names, &schemaname, &opername);
1901 :
1902 16256 : if (schemaname)
1903 : {
1904 : /* use exact schema given */
1905 1084 : namespaceId = LookupExplicitNamespace(schemaname, missing_schema_ok);
1906 1084 : if (missing_schema_ok && !OidIsValid(namespaceId))
1907 18 : return NULL;
1908 : }
1909 : else
1910 : {
1911 : /* flag to indicate we need namespace search */
1912 15172 : namespaceId = InvalidOid;
1913 15172 : recomputeNamespacePath();
1914 : }
1915 :
1916 : /* Search syscache by name only */
1917 16238 : catlist = SearchSysCacheList1(OPERNAMENSP, CStringGetDatum(opername));
1918 :
1919 : /*
1920 : * In typical scenarios, most if not all of the operators found by the
1921 : * catcache search will end up getting returned; and there can be quite a
1922 : * few, for common operator names such as '=' or '+'. To reduce the time
1923 : * spent in palloc, we allocate the result space as an array large enough
1924 : * to hold all the operators. The original coding of this routine did a
1925 : * separate palloc for each operator, but profiling revealed that the
1926 : * pallocs used an unreasonably large fraction of parsing time.
1927 : */
1928 : #define SPACE_PER_OP MAXALIGN(offsetof(struct _FuncCandidateList, args) + \
1929 : 2 * sizeof(Oid))
1930 :
1931 16238 : if (catlist->n_members > 0)
1932 16226 : resultSpace = palloc(catlist->n_members * SPACE_PER_OP);
1933 :
1934 747790 : for (i = 0; i < catlist->n_members; i++)
1935 : {
1936 731552 : HeapTuple opertup = &catlist->members[i]->tuple;
1937 731552 : Form_pg_operator operform = (Form_pg_operator) GETSTRUCT(opertup);
1938 731552 : int pathpos = 0;
1939 : FuncCandidateList newResult;
1940 :
1941 : /* Ignore operators of wrong kind, if specific kind requested */
1942 731552 : if (oprkind && operform->oprkind != oprkind)
1943 9434 : continue;
1944 :
1945 722118 : if (OidIsValid(namespaceId))
1946 : {
1947 : /* Consider only opers in specified namespace */
1948 18980 : if (operform->oprnamespace != namespaceId)
1949 0 : continue;
1950 : /* No need to check args, they must all be different */
1951 : }
1952 : else
1953 : {
1954 : /*
1955 : * Consider only opers that are in the search path and are not in
1956 : * the temp namespace.
1957 : */
1958 : ListCell *nsp;
1959 :
1960 787634 : foreach(nsp, activeSearchPath)
1961 : {
1962 786774 : if (operform->oprnamespace == lfirst_oid(nsp) &&
1963 702278 : operform->oprnamespace != myTempNamespace)
1964 702278 : break;
1965 84496 : pathpos++;
1966 : }
1967 703138 : if (nsp == NULL)
1968 860 : continue; /* oper is not in search path */
1969 :
1970 : /*
1971 : * Okay, it's in the search path, but does it have the same
1972 : * arguments as something we already accepted? If so, keep only
1973 : * the one that appears earlier in the search path.
1974 : *
1975 : * If we have an ordered list from SearchSysCacheList (the normal
1976 : * case), then any conflicting oper must immediately adjoin this
1977 : * one in the list, so we only need to look at the newest result
1978 : * item. If we have an unordered list, we have to scan the whole
1979 : * result list.
1980 : */
1981 702278 : if (resultList)
1982 : {
1983 : FuncCandidateList prevResult;
1984 :
1985 687118 : if (catlist->ordered)
1986 : {
1987 687118 : if (operform->oprleft == resultList->args[0] &&
1988 198324 : operform->oprright == resultList->args[1])
1989 0 : prevResult = resultList;
1990 : else
1991 687118 : prevResult = NULL;
1992 : }
1993 : else
1994 : {
1995 0 : for (prevResult = resultList;
1996 : prevResult;
1997 0 : prevResult = prevResult->next)
1998 : {
1999 0 : if (operform->oprleft == prevResult->args[0] &&
2000 0 : operform->oprright == prevResult->args[1])
2001 0 : break;
2002 : }
2003 : }
2004 687118 : if (prevResult)
2005 : {
2006 : /* We have a match with a previous result */
2007 : Assert(pathpos != prevResult->pathpos);
2008 0 : if (pathpos > prevResult->pathpos)
2009 0 : continue; /* keep previous result */
2010 : /* replace previous result */
2011 0 : prevResult->pathpos = pathpos;
2012 0 : prevResult->oid = operform->oid;
2013 0 : continue; /* args are same, of course */
2014 : }
2015 : }
2016 : }
2017 :
2018 : /*
2019 : * Okay to add it to result list
2020 : */
2021 721258 : newResult = (FuncCandidateList) (resultSpace + nextResult);
2022 721258 : nextResult += SPACE_PER_OP;
2023 :
2024 721258 : newResult->pathpos = pathpos;
2025 721258 : newResult->oid = operform->oid;
2026 721258 : newResult->nominalnargs = 2;
2027 721258 : newResult->nargs = 2;
2028 721258 : newResult->nvargs = 0;
2029 721258 : newResult->ndargs = 0;
2030 721258 : newResult->argnumbers = NULL;
2031 721258 : newResult->args[0] = operform->oprleft;
2032 721258 : newResult->args[1] = operform->oprright;
2033 721258 : newResult->next = resultList;
2034 721258 : resultList = newResult;
2035 : }
2036 :
2037 16238 : ReleaseSysCacheList(catlist);
2038 :
2039 16238 : return resultList;
2040 : }
2041 :
2042 : /*
2043 : * OperatorIsVisible
2044 : * Determine whether an operator (identified by OID) is visible in the
2045 : * current search path. Visible means "would be found by searching
2046 : * for the unqualified operator name with exact argument matches".
2047 : */
2048 : bool
2049 2606 : OperatorIsVisible(Oid oprid)
2050 : {
2051 2606 : return OperatorIsVisibleExt(oprid, NULL);
2052 : }
2053 :
2054 : /*
2055 : * OperatorIsVisibleExt
2056 : * As above, but if the operator isn't found and is_missing is not NULL,
2057 : * then set *is_missing = true and return false instead of throwing
2058 : * an error. (Caller must initialize *is_missing = false.)
2059 : */
2060 : static bool
2061 4306 : OperatorIsVisibleExt(Oid oprid, bool *is_missing)
2062 : {
2063 : HeapTuple oprtup;
2064 : Form_pg_operator oprform;
2065 : Oid oprnamespace;
2066 : bool visible;
2067 :
2068 4306 : oprtup = SearchSysCache1(OPEROID, ObjectIdGetDatum(oprid));
2069 4306 : if (!HeapTupleIsValid(oprtup))
2070 : {
2071 0 : if (is_missing != NULL)
2072 : {
2073 0 : *is_missing = true;
2074 0 : return false;
2075 : }
2076 0 : elog(ERROR, "cache lookup failed for operator %u", oprid);
2077 : }
2078 4306 : oprform = (Form_pg_operator) GETSTRUCT(oprtup);
2079 :
2080 4306 : recomputeNamespacePath();
2081 :
2082 : /*
2083 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2084 : * the system namespace are surely in the path and so we needn't even do
2085 : * list_member_oid() for them.
2086 : */
2087 4306 : oprnamespace = oprform->oprnamespace;
2088 4306 : if (oprnamespace != PG_CATALOG_NAMESPACE &&
2089 1446 : !list_member_oid(activeSearchPath, oprnamespace))
2090 308 : visible = false;
2091 : else
2092 : {
2093 : /*
2094 : * If it is in the path, it might still not be visible; it could be
2095 : * hidden by another operator of the same name and arguments earlier
2096 : * in the path. So we must do a slow check to see if this is the same
2097 : * operator that would be found by OpernameGetOprid.
2098 : */
2099 3998 : char *oprname = NameStr(oprform->oprname);
2100 :
2101 3998 : visible = (OpernameGetOprid(list_make1(makeString(oprname)),
2102 : oprform->oprleft, oprform->oprright)
2103 : == oprid);
2104 : }
2105 :
2106 4306 : ReleaseSysCache(oprtup);
2107 :
2108 4306 : return visible;
2109 : }
2110 :
2111 :
2112 : /*
2113 : * OpclassnameGetOpcid
2114 : * Try to resolve an unqualified index opclass name.
2115 : * Returns OID if opclass found in search path, else InvalidOid.
2116 : *
2117 : * This is essentially the same as TypenameGetTypid, but we have to have
2118 : * an extra argument for the index AM OID.
2119 : */
2120 : Oid
2121 20560 : OpclassnameGetOpcid(Oid amid, const char *opcname)
2122 : {
2123 : Oid opcid;
2124 : ListCell *l;
2125 :
2126 20560 : recomputeNamespacePath();
2127 :
2128 21320 : foreach(l, activeSearchPath)
2129 : {
2130 21296 : Oid namespaceId = lfirst_oid(l);
2131 :
2132 21296 : if (namespaceId == myTempNamespace)
2133 278 : continue; /* do not look in temp namespace */
2134 :
2135 21018 : opcid = GetSysCacheOid3(CLAAMNAMENSP, Anum_pg_opclass_oid,
2136 : ObjectIdGetDatum(amid),
2137 : PointerGetDatum(opcname),
2138 : ObjectIdGetDatum(namespaceId));
2139 21018 : if (OidIsValid(opcid))
2140 20536 : return opcid;
2141 : }
2142 :
2143 : /* Not found in path */
2144 24 : return InvalidOid;
2145 : }
2146 :
2147 : /*
2148 : * OpclassIsVisible
2149 : * Determine whether an opclass (identified by OID) is visible in the
2150 : * current search path. Visible means "would be found by searching
2151 : * for the unqualified opclass name".
2152 : */
2153 : bool
2154 690 : OpclassIsVisible(Oid opcid)
2155 : {
2156 690 : return OpclassIsVisibleExt(opcid, NULL);
2157 : }
2158 :
2159 : /*
2160 : * OpclassIsVisibleExt
2161 : * As above, but if the opclass isn't found and is_missing is not NULL,
2162 : * then set *is_missing = true and return false instead of throwing
2163 : * an error. (Caller must initialize *is_missing = false.)
2164 : */
2165 : static bool
2166 708 : OpclassIsVisibleExt(Oid opcid, bool *is_missing)
2167 : {
2168 : HeapTuple opctup;
2169 : Form_pg_opclass opcform;
2170 : Oid opcnamespace;
2171 : bool visible;
2172 :
2173 708 : opctup = SearchSysCache1(CLAOID, ObjectIdGetDatum(opcid));
2174 708 : if (!HeapTupleIsValid(opctup))
2175 : {
2176 0 : if (is_missing != NULL)
2177 : {
2178 0 : *is_missing = true;
2179 0 : return false;
2180 : }
2181 0 : elog(ERROR, "cache lookup failed for opclass %u", opcid);
2182 : }
2183 708 : opcform = (Form_pg_opclass) GETSTRUCT(opctup);
2184 :
2185 708 : recomputeNamespacePath();
2186 :
2187 : /*
2188 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2189 : * the system namespace are surely in the path and so we needn't even do
2190 : * list_member_oid() for them.
2191 : */
2192 708 : opcnamespace = opcform->opcnamespace;
2193 708 : if (opcnamespace != PG_CATALOG_NAMESPACE &&
2194 136 : !list_member_oid(activeSearchPath, opcnamespace))
2195 28 : visible = false;
2196 : else
2197 : {
2198 : /*
2199 : * If it is in the path, it might still not be visible; it could be
2200 : * hidden by another opclass of the same name earlier in the path. So
2201 : * we must do a slow check to see if this opclass would be found by
2202 : * OpclassnameGetOpcid.
2203 : */
2204 680 : char *opcname = NameStr(opcform->opcname);
2205 :
2206 680 : visible = (OpclassnameGetOpcid(opcform->opcmethod, opcname) == opcid);
2207 : }
2208 :
2209 708 : ReleaseSysCache(opctup);
2210 :
2211 708 : return visible;
2212 : }
2213 :
2214 : /*
2215 : * OpfamilynameGetOpfid
2216 : * Try to resolve an unqualified index opfamily name.
2217 : * Returns OID if opfamily found in search path, else InvalidOid.
2218 : *
2219 : * This is essentially the same as TypenameGetTypid, but we have to have
2220 : * an extra argument for the index AM OID.
2221 : */
2222 : Oid
2223 2352 : OpfamilynameGetOpfid(Oid amid, const char *opfname)
2224 : {
2225 : Oid opfid;
2226 : ListCell *l;
2227 :
2228 2352 : recomputeNamespacePath();
2229 :
2230 4648 : foreach(l, activeSearchPath)
2231 : {
2232 4636 : Oid namespaceId = lfirst_oid(l);
2233 :
2234 4636 : if (namespaceId == myTempNamespace)
2235 402 : continue; /* do not look in temp namespace */
2236 :
2237 4234 : opfid = GetSysCacheOid3(OPFAMILYAMNAMENSP, Anum_pg_opfamily_oid,
2238 : ObjectIdGetDatum(amid),
2239 : PointerGetDatum(opfname),
2240 : ObjectIdGetDatum(namespaceId));
2241 4234 : if (OidIsValid(opfid))
2242 2340 : return opfid;
2243 : }
2244 :
2245 : /* Not found in path */
2246 12 : return InvalidOid;
2247 : }
2248 :
2249 : /*
2250 : * OpfamilyIsVisible
2251 : * Determine whether an opfamily (identified by OID) is visible in the
2252 : * current search path. Visible means "would be found by searching
2253 : * for the unqualified opfamily name".
2254 : */
2255 : bool
2256 1440 : OpfamilyIsVisible(Oid opfid)
2257 : {
2258 1440 : return OpfamilyIsVisibleExt(opfid, NULL);
2259 : }
2260 :
2261 : /*
2262 : * OpfamilyIsVisibleExt
2263 : * As above, but if the opfamily isn't found and is_missing is not NULL,
2264 : * then set *is_missing = true and return false instead of throwing
2265 : * an error. (Caller must initialize *is_missing = false.)
2266 : */
2267 : static bool
2268 1752 : OpfamilyIsVisibleExt(Oid opfid, bool *is_missing)
2269 : {
2270 : HeapTuple opftup;
2271 : Form_pg_opfamily opfform;
2272 : Oid opfnamespace;
2273 : bool visible;
2274 :
2275 1752 : opftup = SearchSysCache1(OPFAMILYOID, ObjectIdGetDatum(opfid));
2276 1752 : if (!HeapTupleIsValid(opftup))
2277 : {
2278 0 : if (is_missing != NULL)
2279 : {
2280 0 : *is_missing = true;
2281 0 : return false;
2282 : }
2283 0 : elog(ERROR, "cache lookup failed for opfamily %u", opfid);
2284 : }
2285 1752 : opfform = (Form_pg_opfamily) GETSTRUCT(opftup);
2286 :
2287 1752 : recomputeNamespacePath();
2288 :
2289 : /*
2290 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2291 : * the system namespace are surely in the path and so we needn't even do
2292 : * list_member_oid() for them.
2293 : */
2294 1752 : opfnamespace = opfform->opfnamespace;
2295 1752 : if (opfnamespace != PG_CATALOG_NAMESPACE &&
2296 1416 : !list_member_oid(activeSearchPath, opfnamespace))
2297 196 : visible = false;
2298 : else
2299 : {
2300 : /*
2301 : * If it is in the path, it might still not be visible; it could be
2302 : * hidden by another opfamily of the same name earlier in the path. So
2303 : * we must do a slow check to see if this opfamily would be found by
2304 : * OpfamilynameGetOpfid.
2305 : */
2306 1556 : char *opfname = NameStr(opfform->opfname);
2307 :
2308 1556 : visible = (OpfamilynameGetOpfid(opfform->opfmethod, opfname) == opfid);
2309 : }
2310 :
2311 1752 : ReleaseSysCache(opftup);
2312 :
2313 1752 : return visible;
2314 : }
2315 :
2316 : /*
2317 : * lookup_collation
2318 : * If there's a collation of the given name/namespace, and it works
2319 : * with the given encoding, return its OID. Else return InvalidOid.
2320 : */
2321 : static Oid
2322 11880 : lookup_collation(const char *collname, Oid collnamespace, int32 encoding)
2323 : {
2324 : Oid collid;
2325 : HeapTuple colltup;
2326 : Form_pg_collation collform;
2327 :
2328 : /* Check for encoding-specific entry (exact match) */
2329 11880 : collid = GetSysCacheOid3(COLLNAMEENCNSP, Anum_pg_collation_oid,
2330 : PointerGetDatum(collname),
2331 : Int32GetDatum(encoding),
2332 : ObjectIdGetDatum(collnamespace));
2333 11880 : if (OidIsValid(collid))
2334 296 : return collid;
2335 :
2336 : /*
2337 : * Check for any-encoding entry. This takes a bit more work: while libc
2338 : * collations with collencoding = -1 do work with all encodings, ICU
2339 : * collations only work with certain encodings, so we have to check that
2340 : * aspect before deciding it's a match.
2341 : */
2342 11584 : colltup = SearchSysCache3(COLLNAMEENCNSP,
2343 : PointerGetDatum(collname),
2344 : Int32GetDatum(-1),
2345 : ObjectIdGetDatum(collnamespace));
2346 11584 : if (!HeapTupleIsValid(colltup))
2347 1276 : return InvalidOid;
2348 10308 : collform = (Form_pg_collation) GETSTRUCT(colltup);
2349 10308 : if (collform->collprovider == COLLPROVIDER_ICU)
2350 : {
2351 1238 : if (is_encoding_supported_by_icu(encoding))
2352 1238 : collid = collform->oid;
2353 : else
2354 0 : collid = InvalidOid;
2355 : }
2356 : else
2357 : {
2358 9070 : collid = collform->oid;
2359 : }
2360 10308 : ReleaseSysCache(colltup);
2361 10308 : return collid;
2362 : }
2363 :
2364 : /*
2365 : * CollationGetCollid
2366 : * Try to resolve an unqualified collation name.
2367 : * Returns OID if collation found in search path, else InvalidOid.
2368 : *
2369 : * Note that this will only find collations that work with the current
2370 : * database's encoding.
2371 : */
2372 : Oid
2373 380 : CollationGetCollid(const char *collname)
2374 : {
2375 380 : int32 dbencoding = GetDatabaseEncoding();
2376 : ListCell *l;
2377 :
2378 380 : recomputeNamespacePath();
2379 :
2380 598 : foreach(l, activeSearchPath)
2381 : {
2382 598 : Oid namespaceId = lfirst_oid(l);
2383 : Oid collid;
2384 :
2385 598 : if (namespaceId == myTempNamespace)
2386 148 : continue; /* do not look in temp namespace */
2387 :
2388 450 : collid = lookup_collation(collname, namespaceId, dbencoding);
2389 450 : if (OidIsValid(collid))
2390 380 : return collid;
2391 : }
2392 :
2393 : /* Not found in path */
2394 0 : return InvalidOid;
2395 : }
2396 :
2397 : /*
2398 : * CollationIsVisible
2399 : * Determine whether a collation (identified by OID) is visible in the
2400 : * current search path. Visible means "would be found by searching
2401 : * for the unqualified collation name".
2402 : *
2403 : * Note that only collations that work with the current database's encoding
2404 : * will be considered visible.
2405 : */
2406 : bool
2407 380 : CollationIsVisible(Oid collid)
2408 : {
2409 380 : return CollationIsVisibleExt(collid, NULL);
2410 : }
2411 :
2412 : /*
2413 : * CollationIsVisibleExt
2414 : * As above, but if the collation isn't found and is_missing is not NULL,
2415 : * then set *is_missing = true and return false instead of throwing
2416 : * an error. (Caller must initialize *is_missing = false.)
2417 : */
2418 : static bool
2419 380 : CollationIsVisibleExt(Oid collid, bool *is_missing)
2420 : {
2421 : HeapTuple colltup;
2422 : Form_pg_collation collform;
2423 : Oid collnamespace;
2424 : bool visible;
2425 :
2426 380 : colltup = SearchSysCache1(COLLOID, ObjectIdGetDatum(collid));
2427 380 : if (!HeapTupleIsValid(colltup))
2428 : {
2429 0 : if (is_missing != NULL)
2430 : {
2431 0 : *is_missing = true;
2432 0 : return false;
2433 : }
2434 0 : elog(ERROR, "cache lookup failed for collation %u", collid);
2435 : }
2436 380 : collform = (Form_pg_collation) GETSTRUCT(colltup);
2437 :
2438 380 : recomputeNamespacePath();
2439 :
2440 : /*
2441 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2442 : * the system namespace are surely in the path and so we needn't even do
2443 : * list_member_oid() for them.
2444 : */
2445 380 : collnamespace = collform->collnamespace;
2446 380 : if (collnamespace != PG_CATALOG_NAMESPACE &&
2447 70 : !list_member_oid(activeSearchPath, collnamespace))
2448 0 : visible = false;
2449 : else
2450 : {
2451 : /*
2452 : * If it is in the path, it might still not be visible; it could be
2453 : * hidden by another collation of the same name earlier in the path,
2454 : * or it might not work with the current DB encoding. So we must do a
2455 : * slow check to see if this collation would be found by
2456 : * CollationGetCollid.
2457 : */
2458 380 : char *collname = NameStr(collform->collname);
2459 :
2460 380 : visible = (CollationGetCollid(collname) == collid);
2461 : }
2462 :
2463 380 : ReleaseSysCache(colltup);
2464 :
2465 380 : return visible;
2466 : }
2467 :
2468 :
2469 : /*
2470 : * ConversionGetConid
2471 : * Try to resolve an unqualified conversion name.
2472 : * Returns OID if conversion found in search path, else InvalidOid.
2473 : *
2474 : * This is essentially the same as RelnameGetRelid.
2475 : */
2476 : Oid
2477 18 : ConversionGetConid(const char *conname)
2478 : {
2479 : Oid conid;
2480 : ListCell *l;
2481 :
2482 18 : recomputeNamespacePath();
2483 :
2484 36 : foreach(l, activeSearchPath)
2485 : {
2486 36 : Oid namespaceId = lfirst_oid(l);
2487 :
2488 36 : if (namespaceId == myTempNamespace)
2489 0 : continue; /* do not look in temp namespace */
2490 :
2491 36 : conid = GetSysCacheOid2(CONNAMENSP, Anum_pg_conversion_oid,
2492 : PointerGetDatum(conname),
2493 : ObjectIdGetDatum(namespaceId));
2494 36 : if (OidIsValid(conid))
2495 18 : return conid;
2496 : }
2497 :
2498 : /* Not found in path */
2499 0 : return InvalidOid;
2500 : }
2501 :
2502 : /*
2503 : * ConversionIsVisible
2504 : * Determine whether a conversion (identified by OID) is visible in the
2505 : * current search path. Visible means "would be found by searching
2506 : * for the unqualified conversion name".
2507 : */
2508 : bool
2509 30 : ConversionIsVisible(Oid conid)
2510 : {
2511 30 : return ConversionIsVisibleExt(conid, NULL);
2512 : }
2513 :
2514 : /*
2515 : * ConversionIsVisibleExt
2516 : * As above, but if the conversion isn't found and is_missing is not NULL,
2517 : * then set *is_missing = true and return false instead of throwing
2518 : * an error. (Caller must initialize *is_missing = false.)
2519 : */
2520 : static bool
2521 30 : ConversionIsVisibleExt(Oid conid, bool *is_missing)
2522 : {
2523 : HeapTuple contup;
2524 : Form_pg_conversion conform;
2525 : Oid connamespace;
2526 : bool visible;
2527 :
2528 30 : contup = SearchSysCache1(CONVOID, ObjectIdGetDatum(conid));
2529 30 : if (!HeapTupleIsValid(contup))
2530 : {
2531 0 : if (is_missing != NULL)
2532 : {
2533 0 : *is_missing = true;
2534 0 : return false;
2535 : }
2536 0 : elog(ERROR, "cache lookup failed for conversion %u", conid);
2537 : }
2538 30 : conform = (Form_pg_conversion) GETSTRUCT(contup);
2539 :
2540 30 : recomputeNamespacePath();
2541 :
2542 : /*
2543 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2544 : * the system namespace are surely in the path and so we needn't even do
2545 : * list_member_oid() for them.
2546 : */
2547 30 : connamespace = conform->connamespace;
2548 30 : if (connamespace != PG_CATALOG_NAMESPACE &&
2549 30 : !list_member_oid(activeSearchPath, connamespace))
2550 12 : visible = false;
2551 : else
2552 : {
2553 : /*
2554 : * If it is in the path, it might still not be visible; it could be
2555 : * hidden by another conversion of the same name earlier in the path.
2556 : * So we must do a slow check to see if this conversion would be found
2557 : * by ConversionGetConid.
2558 : */
2559 18 : char *conname = NameStr(conform->conname);
2560 :
2561 18 : visible = (ConversionGetConid(conname) == conid);
2562 : }
2563 :
2564 30 : ReleaseSysCache(contup);
2565 :
2566 30 : return visible;
2567 : }
2568 :
2569 : /*
2570 : * get_statistics_object_oid - find a statistics object by possibly qualified name
2571 : *
2572 : * If not found, returns InvalidOid if missing_ok, else throws error
2573 : */
2574 : Oid
2575 322 : get_statistics_object_oid(List *names, bool missing_ok)
2576 : {
2577 : char *schemaname;
2578 : char *stats_name;
2579 : Oid namespaceId;
2580 322 : Oid stats_oid = InvalidOid;
2581 : ListCell *l;
2582 :
2583 : /* deconstruct the name list */
2584 322 : DeconstructQualifiedName(names, &schemaname, &stats_name);
2585 :
2586 322 : if (schemaname)
2587 : {
2588 : /* use exact schema given */
2589 28 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
2590 28 : if (missing_ok && !OidIsValid(namespaceId))
2591 0 : stats_oid = InvalidOid;
2592 : else
2593 28 : stats_oid = GetSysCacheOid2(STATEXTNAMENSP, Anum_pg_statistic_ext_oid,
2594 : PointerGetDatum(stats_name),
2595 : ObjectIdGetDatum(namespaceId));
2596 : }
2597 : else
2598 : {
2599 : /* search for it in search path */
2600 294 : recomputeNamespacePath();
2601 :
2602 600 : foreach(l, activeSearchPath)
2603 : {
2604 588 : namespaceId = lfirst_oid(l);
2605 :
2606 588 : if (namespaceId == myTempNamespace)
2607 0 : continue; /* do not look in temp namespace */
2608 588 : stats_oid = GetSysCacheOid2(STATEXTNAMENSP, Anum_pg_statistic_ext_oid,
2609 : PointerGetDatum(stats_name),
2610 : ObjectIdGetDatum(namespaceId));
2611 588 : if (OidIsValid(stats_oid))
2612 282 : break;
2613 : }
2614 : }
2615 :
2616 322 : if (!OidIsValid(stats_oid) && !missing_ok)
2617 6 : ereport(ERROR,
2618 : (errcode(ERRCODE_UNDEFINED_OBJECT),
2619 : errmsg("statistics object \"%s\" does not exist",
2620 : NameListToString(names))));
2621 :
2622 316 : return stats_oid;
2623 : }
2624 :
2625 : /*
2626 : * StatisticsObjIsVisible
2627 : * Determine whether a statistics object (identified by OID) is visible in
2628 : * the current search path. Visible means "would be found by searching
2629 : * for the unqualified statistics object name".
2630 : */
2631 : bool
2632 322 : StatisticsObjIsVisible(Oid stxid)
2633 : {
2634 322 : return StatisticsObjIsVisibleExt(stxid, NULL);
2635 : }
2636 :
2637 : /*
2638 : * StatisticsObjIsVisibleExt
2639 : * As above, but if the statistics object isn't found and is_missing is
2640 : * not NULL, then set *is_missing = true and return false instead of
2641 : * throwing an error. (Caller must initialize *is_missing = false.)
2642 : */
2643 : static bool
2644 688 : StatisticsObjIsVisibleExt(Oid stxid, bool *is_missing)
2645 : {
2646 : HeapTuple stxtup;
2647 : Form_pg_statistic_ext stxform;
2648 : Oid stxnamespace;
2649 : bool visible;
2650 :
2651 688 : stxtup = SearchSysCache1(STATEXTOID, ObjectIdGetDatum(stxid));
2652 688 : if (!HeapTupleIsValid(stxtup))
2653 : {
2654 0 : if (is_missing != NULL)
2655 : {
2656 0 : *is_missing = true;
2657 0 : return false;
2658 : }
2659 0 : elog(ERROR, "cache lookup failed for statistics object %u", stxid);
2660 : }
2661 688 : stxform = (Form_pg_statistic_ext) GETSTRUCT(stxtup);
2662 :
2663 688 : recomputeNamespacePath();
2664 :
2665 : /*
2666 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2667 : * the system namespace are surely in the path and so we needn't even do
2668 : * list_member_oid() for them.
2669 : */
2670 688 : stxnamespace = stxform->stxnamespace;
2671 688 : if (stxnamespace != PG_CATALOG_NAMESPACE &&
2672 688 : !list_member_oid(activeSearchPath, stxnamespace))
2673 78 : visible = false;
2674 : else
2675 : {
2676 : /*
2677 : * If it is in the path, it might still not be visible; it could be
2678 : * hidden by another statistics object of the same name earlier in the
2679 : * path. So we must do a slow check for conflicting objects.
2680 : */
2681 610 : char *stxname = NameStr(stxform->stxname);
2682 : ListCell *l;
2683 :
2684 610 : visible = false;
2685 1324 : foreach(l, activeSearchPath)
2686 : {
2687 1324 : Oid namespaceId = lfirst_oid(l);
2688 :
2689 1324 : if (namespaceId == stxnamespace)
2690 : {
2691 : /* Found it first in path */
2692 610 : visible = true;
2693 610 : break;
2694 : }
2695 714 : if (SearchSysCacheExists2(STATEXTNAMENSP,
2696 : PointerGetDatum(stxname),
2697 : ObjectIdGetDatum(namespaceId)))
2698 : {
2699 : /* Found something else first in path */
2700 0 : break;
2701 : }
2702 : }
2703 : }
2704 :
2705 688 : ReleaseSysCache(stxtup);
2706 :
2707 688 : return visible;
2708 : }
2709 :
2710 : /*
2711 : * get_ts_parser_oid - find a TS parser by possibly qualified name
2712 : *
2713 : * If not found, returns InvalidOid if missing_ok, else throws error
2714 : */
2715 : Oid
2716 2536 : get_ts_parser_oid(List *names, bool missing_ok)
2717 : {
2718 : char *schemaname;
2719 : char *parser_name;
2720 : Oid namespaceId;
2721 2536 : Oid prsoid = InvalidOid;
2722 : ListCell *l;
2723 :
2724 : /* deconstruct the name list */
2725 2536 : DeconstructQualifiedName(names, &schemaname, &parser_name);
2726 :
2727 2524 : if (schemaname)
2728 : {
2729 : /* use exact schema given */
2730 46 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
2731 46 : if (missing_ok && !OidIsValid(namespaceId))
2732 6 : prsoid = InvalidOid;
2733 : else
2734 40 : prsoid = GetSysCacheOid2(TSPARSERNAMENSP, Anum_pg_ts_parser_oid,
2735 : PointerGetDatum(parser_name),
2736 : ObjectIdGetDatum(namespaceId));
2737 : }
2738 : else
2739 : {
2740 : /* search for it in search path */
2741 2478 : recomputeNamespacePath();
2742 :
2743 2560 : foreach(l, activeSearchPath)
2744 : {
2745 2536 : namespaceId = lfirst_oid(l);
2746 :
2747 2536 : if (namespaceId == myTempNamespace)
2748 0 : continue; /* do not look in temp namespace */
2749 :
2750 2536 : prsoid = GetSysCacheOid2(TSPARSERNAMENSP, Anum_pg_ts_parser_oid,
2751 : PointerGetDatum(parser_name),
2752 : ObjectIdGetDatum(namespaceId));
2753 2536 : if (OidIsValid(prsoid))
2754 2454 : break;
2755 : }
2756 : }
2757 :
2758 2524 : if (!OidIsValid(prsoid) && !missing_ok)
2759 30 : ereport(ERROR,
2760 : (errcode(ERRCODE_UNDEFINED_OBJECT),
2761 : errmsg("text search parser \"%s\" does not exist",
2762 : NameListToString(names))));
2763 :
2764 2494 : return prsoid;
2765 : }
2766 :
2767 : /*
2768 : * TSParserIsVisible
2769 : * Determine whether a parser (identified by OID) is visible in the
2770 : * current search path. Visible means "would be found by searching
2771 : * for the unqualified parser name".
2772 : */
2773 : bool
2774 30 : TSParserIsVisible(Oid prsId)
2775 : {
2776 30 : return TSParserIsVisibleExt(prsId, NULL);
2777 : }
2778 :
2779 : /*
2780 : * TSParserIsVisibleExt
2781 : * As above, but if the parser isn't found and is_missing is not NULL,
2782 : * then set *is_missing = true and return false instead of throwing
2783 : * an error. (Caller must initialize *is_missing = false.)
2784 : */
2785 : static bool
2786 30 : TSParserIsVisibleExt(Oid prsId, bool *is_missing)
2787 : {
2788 : HeapTuple tup;
2789 : Form_pg_ts_parser form;
2790 : Oid namespace;
2791 : bool visible;
2792 :
2793 30 : tup = SearchSysCache1(TSPARSEROID, ObjectIdGetDatum(prsId));
2794 30 : if (!HeapTupleIsValid(tup))
2795 : {
2796 0 : if (is_missing != NULL)
2797 : {
2798 0 : *is_missing = true;
2799 0 : return false;
2800 : }
2801 0 : elog(ERROR, "cache lookup failed for text search parser %u", prsId);
2802 : }
2803 30 : form = (Form_pg_ts_parser) GETSTRUCT(tup);
2804 :
2805 30 : recomputeNamespacePath();
2806 :
2807 : /*
2808 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2809 : * the system namespace are surely in the path and so we needn't even do
2810 : * list_member_oid() for them.
2811 : */
2812 30 : namespace = form->prsnamespace;
2813 30 : if (namespace != PG_CATALOG_NAMESPACE &&
2814 30 : !list_member_oid(activeSearchPath, namespace))
2815 12 : visible = false;
2816 : else
2817 : {
2818 : /*
2819 : * If it is in the path, it might still not be visible; it could be
2820 : * hidden by another parser of the same name earlier in the path. So
2821 : * we must do a slow check for conflicting parsers.
2822 : */
2823 18 : char *name = NameStr(form->prsname);
2824 : ListCell *l;
2825 :
2826 18 : visible = false;
2827 36 : foreach(l, activeSearchPath)
2828 : {
2829 36 : Oid namespaceId = lfirst_oid(l);
2830 :
2831 36 : if (namespaceId == myTempNamespace)
2832 0 : continue; /* do not look in temp namespace */
2833 :
2834 36 : if (namespaceId == namespace)
2835 : {
2836 : /* Found it first in path */
2837 18 : visible = true;
2838 18 : break;
2839 : }
2840 18 : if (SearchSysCacheExists2(TSPARSERNAMENSP,
2841 : PointerGetDatum(name),
2842 : ObjectIdGetDatum(namespaceId)))
2843 : {
2844 : /* Found something else first in path */
2845 0 : break;
2846 : }
2847 : }
2848 : }
2849 :
2850 30 : ReleaseSysCache(tup);
2851 :
2852 30 : return visible;
2853 : }
2854 :
2855 : /*
2856 : * get_ts_dict_oid - find a TS dictionary by possibly qualified name
2857 : *
2858 : * If not found, returns InvalidOid if missing_ok, else throws error
2859 : */
2860 : Oid
2861 10796 : get_ts_dict_oid(List *names, bool missing_ok)
2862 : {
2863 : char *schemaname;
2864 : char *dict_name;
2865 : Oid namespaceId;
2866 10796 : Oid dictoid = InvalidOid;
2867 : ListCell *l;
2868 :
2869 : /* deconstruct the name list */
2870 10796 : DeconstructQualifiedName(names, &schemaname, &dict_name);
2871 :
2872 10784 : if (schemaname)
2873 : {
2874 : /* use exact schema given */
2875 104 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
2876 104 : if (missing_ok && !OidIsValid(namespaceId))
2877 6 : dictoid = InvalidOid;
2878 : else
2879 98 : dictoid = GetSysCacheOid2(TSDICTNAMENSP, Anum_pg_ts_dict_oid,
2880 : PointerGetDatum(dict_name),
2881 : ObjectIdGetDatum(namespaceId));
2882 : }
2883 : else
2884 : {
2885 : /* search for it in search path */
2886 10680 : recomputeNamespacePath();
2887 :
2888 11558 : foreach(l, activeSearchPath)
2889 : {
2890 11528 : namespaceId = lfirst_oid(l);
2891 :
2892 11528 : if (namespaceId == myTempNamespace)
2893 0 : continue; /* do not look in temp namespace */
2894 :
2895 11528 : dictoid = GetSysCacheOid2(TSDICTNAMENSP, Anum_pg_ts_dict_oid,
2896 : PointerGetDatum(dict_name),
2897 : ObjectIdGetDatum(namespaceId));
2898 11528 : if (OidIsValid(dictoid))
2899 10650 : break;
2900 : }
2901 : }
2902 :
2903 10784 : if (!OidIsValid(dictoid) && !missing_ok)
2904 30 : ereport(ERROR,
2905 : (errcode(ERRCODE_UNDEFINED_OBJECT),
2906 : errmsg("text search dictionary \"%s\" does not exist",
2907 : NameListToString(names))));
2908 :
2909 10754 : return dictoid;
2910 : }
2911 :
2912 : /*
2913 : * TSDictionaryIsVisible
2914 : * Determine whether a dictionary (identified by OID) is visible in the
2915 : * current search path. Visible means "would be found by searching
2916 : * for the unqualified dictionary name".
2917 : */
2918 : bool
2919 5830 : TSDictionaryIsVisible(Oid dictId)
2920 : {
2921 5830 : return TSDictionaryIsVisibleExt(dictId, NULL);
2922 : }
2923 :
2924 : /*
2925 : * TSDictionaryIsVisibleExt
2926 : * As above, but if the dictionary isn't found and is_missing is not NULL,
2927 : * then set *is_missing = true and return false instead of throwing
2928 : * an error. (Caller must initialize *is_missing = false.)
2929 : */
2930 : static bool
2931 5830 : TSDictionaryIsVisibleExt(Oid dictId, bool *is_missing)
2932 : {
2933 : HeapTuple tup;
2934 : Form_pg_ts_dict form;
2935 : Oid namespace;
2936 : bool visible;
2937 :
2938 5830 : tup = SearchSysCache1(TSDICTOID, ObjectIdGetDatum(dictId));
2939 5830 : if (!HeapTupleIsValid(tup))
2940 : {
2941 0 : if (is_missing != NULL)
2942 : {
2943 0 : *is_missing = true;
2944 0 : return false;
2945 : }
2946 0 : elog(ERROR, "cache lookup failed for text search dictionary %u",
2947 : dictId);
2948 : }
2949 5830 : form = (Form_pg_ts_dict) GETSTRUCT(tup);
2950 :
2951 5830 : recomputeNamespacePath();
2952 :
2953 : /*
2954 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
2955 : * the system namespace are surely in the path and so we needn't even do
2956 : * list_member_oid() for them.
2957 : */
2958 5830 : namespace = form->dictnamespace;
2959 5830 : if (namespace != PG_CATALOG_NAMESPACE &&
2960 306 : !list_member_oid(activeSearchPath, namespace))
2961 282 : visible = false;
2962 : else
2963 : {
2964 : /*
2965 : * If it is in the path, it might still not be visible; it could be
2966 : * hidden by another dictionary of the same name earlier in the path.
2967 : * So we must do a slow check for conflicting dictionaries.
2968 : */
2969 5548 : char *name = NameStr(form->dictname);
2970 : ListCell *l;
2971 :
2972 5548 : visible = false;
2973 5572 : foreach(l, activeSearchPath)
2974 : {
2975 5572 : Oid namespaceId = lfirst_oid(l);
2976 :
2977 5572 : if (namespaceId == myTempNamespace)
2978 0 : continue; /* do not look in temp namespace */
2979 :
2980 5572 : if (namespaceId == namespace)
2981 : {
2982 : /* Found it first in path */
2983 5548 : visible = true;
2984 5548 : break;
2985 : }
2986 24 : if (SearchSysCacheExists2(TSDICTNAMENSP,
2987 : PointerGetDatum(name),
2988 : ObjectIdGetDatum(namespaceId)))
2989 : {
2990 : /* Found something else first in path */
2991 0 : break;
2992 : }
2993 : }
2994 : }
2995 :
2996 5830 : ReleaseSysCache(tup);
2997 :
2998 5830 : return visible;
2999 : }
3000 :
3001 : /*
3002 : * get_ts_template_oid - find a TS template by possibly qualified name
3003 : *
3004 : * If not found, returns InvalidOid if missing_ok, else throws error
3005 : */
3006 : Oid
3007 2738 : get_ts_template_oid(List *names, bool missing_ok)
3008 : {
3009 : char *schemaname;
3010 : char *template_name;
3011 : Oid namespaceId;
3012 2738 : Oid tmploid = InvalidOid;
3013 : ListCell *l;
3014 :
3015 : /* deconstruct the name list */
3016 2738 : DeconstructQualifiedName(names, &schemaname, &template_name);
3017 :
3018 2726 : if (schemaname)
3019 : {
3020 : /* use exact schema given */
3021 56 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
3022 56 : if (missing_ok && !OidIsValid(namespaceId))
3023 6 : tmploid = InvalidOid;
3024 : else
3025 50 : tmploid = GetSysCacheOid2(TSTEMPLATENAMENSP, Anum_pg_ts_template_oid,
3026 : PointerGetDatum(template_name),
3027 : ObjectIdGetDatum(namespaceId));
3028 : }
3029 : else
3030 : {
3031 : /* search for it in search path */
3032 2670 : recomputeNamespacePath();
3033 :
3034 2754 : foreach(l, activeSearchPath)
3035 : {
3036 2730 : namespaceId = lfirst_oid(l);
3037 :
3038 2730 : if (namespaceId == myTempNamespace)
3039 0 : continue; /* do not look in temp namespace */
3040 :
3041 2730 : tmploid = GetSysCacheOid2(TSTEMPLATENAMENSP, Anum_pg_ts_template_oid,
3042 : PointerGetDatum(template_name),
3043 : ObjectIdGetDatum(namespaceId));
3044 2730 : if (OidIsValid(tmploid))
3045 2646 : break;
3046 : }
3047 : }
3048 :
3049 2726 : if (!OidIsValid(tmploid) && !missing_ok)
3050 30 : ereport(ERROR,
3051 : (errcode(ERRCODE_UNDEFINED_OBJECT),
3052 : errmsg("text search template \"%s\" does not exist",
3053 : NameListToString(names))));
3054 :
3055 2696 : return tmploid;
3056 : }
3057 :
3058 : /*
3059 : * TSTemplateIsVisible
3060 : * Determine whether a template (identified by OID) is visible in the
3061 : * current search path. Visible means "would be found by searching
3062 : * for the unqualified template name".
3063 : */
3064 : bool
3065 30 : TSTemplateIsVisible(Oid tmplId)
3066 : {
3067 30 : return TSTemplateIsVisibleExt(tmplId, NULL);
3068 : }
3069 :
3070 : /*
3071 : * TSTemplateIsVisibleExt
3072 : * As above, but if the template isn't found and is_missing is not NULL,
3073 : * then set *is_missing = true and return false instead of throwing
3074 : * an error. (Caller must initialize *is_missing = false.)
3075 : */
3076 : static bool
3077 30 : TSTemplateIsVisibleExt(Oid tmplId, bool *is_missing)
3078 : {
3079 : HeapTuple tup;
3080 : Form_pg_ts_template form;
3081 : Oid namespace;
3082 : bool visible;
3083 :
3084 30 : tup = SearchSysCache1(TSTEMPLATEOID, ObjectIdGetDatum(tmplId));
3085 30 : if (!HeapTupleIsValid(tup))
3086 : {
3087 0 : if (is_missing != NULL)
3088 : {
3089 0 : *is_missing = true;
3090 0 : return false;
3091 : }
3092 0 : elog(ERROR, "cache lookup failed for text search template %u", tmplId);
3093 : }
3094 30 : form = (Form_pg_ts_template) GETSTRUCT(tup);
3095 :
3096 30 : recomputeNamespacePath();
3097 :
3098 : /*
3099 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
3100 : * the system namespace are surely in the path and so we needn't even do
3101 : * list_member_oid() for them.
3102 : */
3103 30 : namespace = form->tmplnamespace;
3104 30 : if (namespace != PG_CATALOG_NAMESPACE &&
3105 30 : !list_member_oid(activeSearchPath, namespace))
3106 12 : visible = false;
3107 : else
3108 : {
3109 : /*
3110 : * If it is in the path, it might still not be visible; it could be
3111 : * hidden by another template of the same name earlier in the path. So
3112 : * we must do a slow check for conflicting templates.
3113 : */
3114 18 : char *name = NameStr(form->tmplname);
3115 : ListCell *l;
3116 :
3117 18 : visible = false;
3118 36 : foreach(l, activeSearchPath)
3119 : {
3120 36 : Oid namespaceId = lfirst_oid(l);
3121 :
3122 36 : if (namespaceId == myTempNamespace)
3123 0 : continue; /* do not look in temp namespace */
3124 :
3125 36 : if (namespaceId == namespace)
3126 : {
3127 : /* Found it first in path */
3128 18 : visible = true;
3129 18 : break;
3130 : }
3131 18 : if (SearchSysCacheExists2(TSTEMPLATENAMENSP,
3132 : PointerGetDatum(name),
3133 : ObjectIdGetDatum(namespaceId)))
3134 : {
3135 : /* Found something else first in path */
3136 0 : break;
3137 : }
3138 : }
3139 : }
3140 :
3141 30 : ReleaseSysCache(tup);
3142 :
3143 30 : return visible;
3144 : }
3145 :
3146 : /*
3147 : * get_ts_config_oid - find a TS config by possibly qualified name
3148 : *
3149 : * If not found, returns InvalidOid if missing_ok, else throws error
3150 : */
3151 : Oid
3152 17568 : get_ts_config_oid(List *names, bool missing_ok)
3153 : {
3154 : char *schemaname;
3155 : char *config_name;
3156 : Oid namespaceId;
3157 17568 : Oid cfgoid = InvalidOid;
3158 : ListCell *l;
3159 :
3160 : /* deconstruct the name list */
3161 17568 : DeconstructQualifiedName(names, &schemaname, &config_name);
3162 :
3163 17556 : if (schemaname)
3164 : {
3165 : /* use exact schema given */
3166 5682 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
3167 5682 : if (missing_ok && !OidIsValid(namespaceId))
3168 6 : cfgoid = InvalidOid;
3169 : else
3170 5676 : cfgoid = GetSysCacheOid2(TSCONFIGNAMENSP, Anum_pg_ts_config_oid,
3171 : PointerGetDatum(config_name),
3172 : ObjectIdGetDatum(namespaceId));
3173 : }
3174 : else
3175 : {
3176 : /* search for it in search path */
3177 11874 : recomputeNamespacePath();
3178 :
3179 13026 : foreach(l, activeSearchPath)
3180 : {
3181 12970 : namespaceId = lfirst_oid(l);
3182 :
3183 12970 : if (namespaceId == myTempNamespace)
3184 702 : continue; /* do not look in temp namespace */
3185 :
3186 12268 : cfgoid = GetSysCacheOid2(TSCONFIGNAMENSP, Anum_pg_ts_config_oid,
3187 : PointerGetDatum(config_name),
3188 : ObjectIdGetDatum(namespaceId));
3189 12268 : if (OidIsValid(cfgoid))
3190 11818 : break;
3191 : }
3192 : }
3193 :
3194 17556 : if (!OidIsValid(cfgoid) && !missing_ok)
3195 30 : ereport(ERROR,
3196 : (errcode(ERRCODE_UNDEFINED_OBJECT),
3197 : errmsg("text search configuration \"%s\" does not exist",
3198 : NameListToString(names))));
3199 :
3200 17526 : return cfgoid;
3201 : }
3202 :
3203 : /*
3204 : * TSConfigIsVisible
3205 : * Determine whether a text search configuration (identified by OID)
3206 : * is visible in the current search path. Visible means "would be found
3207 : * by searching for the unqualified text search configuration name".
3208 : */
3209 : bool
3210 46 : TSConfigIsVisible(Oid cfgid)
3211 : {
3212 46 : return TSConfigIsVisibleExt(cfgid, NULL);
3213 : }
3214 :
3215 : /*
3216 : * TSConfigIsVisibleExt
3217 : * As above, but if the configuration isn't found and is_missing is not
3218 : * NULL, then set *is_missing = true and return false instead of throwing
3219 : * an error. (Caller must initialize *is_missing = false.)
3220 : */
3221 : static bool
3222 46 : TSConfigIsVisibleExt(Oid cfgid, bool *is_missing)
3223 : {
3224 : HeapTuple tup;
3225 : Form_pg_ts_config form;
3226 : Oid namespace;
3227 : bool visible;
3228 :
3229 46 : tup = SearchSysCache1(TSCONFIGOID, ObjectIdGetDatum(cfgid));
3230 46 : if (!HeapTupleIsValid(tup))
3231 : {
3232 0 : if (is_missing != NULL)
3233 : {
3234 0 : *is_missing = true;
3235 0 : return false;
3236 : }
3237 0 : elog(ERROR, "cache lookup failed for text search configuration %u",
3238 : cfgid);
3239 : }
3240 46 : form = (Form_pg_ts_config) GETSTRUCT(tup);
3241 :
3242 46 : recomputeNamespacePath();
3243 :
3244 : /*
3245 : * Quick check: if it ain't in the path at all, it ain't visible. Items in
3246 : * the system namespace are surely in the path and so we needn't even do
3247 : * list_member_oid() for them.
3248 : */
3249 46 : namespace = form->cfgnamespace;
3250 46 : if (namespace != PG_CATALOG_NAMESPACE &&
3251 46 : !list_member_oid(activeSearchPath, namespace))
3252 16 : visible = false;
3253 : else
3254 : {
3255 : /*
3256 : * If it is in the path, it might still not be visible; it could be
3257 : * hidden by another configuration of the same name earlier in the
3258 : * path. So we must do a slow check for conflicting configurations.
3259 : */
3260 30 : char *name = NameStr(form->cfgname);
3261 : ListCell *l;
3262 :
3263 30 : visible = false;
3264 60 : foreach(l, activeSearchPath)
3265 : {
3266 60 : Oid namespaceId = lfirst_oid(l);
3267 :
3268 60 : if (namespaceId == myTempNamespace)
3269 0 : continue; /* do not look in temp namespace */
3270 :
3271 60 : if (namespaceId == namespace)
3272 : {
3273 : /* Found it first in path */
3274 30 : visible = true;
3275 30 : break;
3276 : }
3277 30 : if (SearchSysCacheExists2(TSCONFIGNAMENSP,
3278 : PointerGetDatum(name),
3279 : ObjectIdGetDatum(namespaceId)))
3280 : {
3281 : /* Found something else first in path */
3282 0 : break;
3283 : }
3284 : }
3285 : }
3286 :
3287 46 : ReleaseSysCache(tup);
3288 :
3289 46 : return visible;
3290 : }
3291 :
3292 :
3293 : /*
3294 : * DeconstructQualifiedName
3295 : * Given a possibly-qualified name expressed as a list of String nodes,
3296 : * extract the schema name and object name.
3297 : *
3298 : * *nspname_p is set to NULL if there is no explicit schema name.
3299 : */
3300 : void
3301 1996146 : DeconstructQualifiedName(const List *names,
3302 : char **nspname_p,
3303 : char **objname_p)
3304 : {
3305 : char *catalogname;
3306 1996146 : char *schemaname = NULL;
3307 1996146 : char *objname = NULL;
3308 :
3309 1996146 : switch (list_length(names))
3310 : {
3311 1585088 : case 1:
3312 1585088 : objname = strVal(linitial(names));
3313 1585088 : break;
3314 410950 : case 2:
3315 410950 : schemaname = strVal(linitial(names));
3316 410950 : objname = strVal(lsecond(names));
3317 410950 : break;
3318 102 : case 3:
3319 102 : catalogname = strVal(linitial(names));
3320 102 : schemaname = strVal(lsecond(names));
3321 102 : objname = strVal(lthird(names));
3322 :
3323 : /*
3324 : * We check the catalog name and then ignore it.
3325 : */
3326 102 : if (strcmp(catalogname, get_database_name(MyDatabaseId)) != 0)
3327 102 : ereport(ERROR,
3328 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
3329 : errmsg("cross-database references are not implemented: %s",
3330 : NameListToString(names))));
3331 0 : break;
3332 6 : default:
3333 6 : ereport(ERROR,
3334 : (errcode(ERRCODE_SYNTAX_ERROR),
3335 : errmsg("improper qualified name (too many dotted names): %s",
3336 : NameListToString(names))));
3337 : break;
3338 : }
3339 :
3340 1996038 : *nspname_p = schemaname;
3341 1996038 : *objname_p = objname;
3342 1996038 : }
3343 :
3344 : /*
3345 : * LookupNamespaceNoError
3346 : * Look up a schema name.
3347 : *
3348 : * Returns the namespace OID, or InvalidOid if not found.
3349 : *
3350 : * Note this does NOT perform any permissions check --- callers are
3351 : * responsible for being sure that an appropriate check is made.
3352 : * In the majority of cases LookupExplicitNamespace is preferable.
3353 : */
3354 : Oid
3355 342 : LookupNamespaceNoError(const char *nspname)
3356 : {
3357 : /* check for pg_temp alias */
3358 342 : if (strcmp(nspname, "pg_temp") == 0)
3359 : {
3360 0 : if (OidIsValid(myTempNamespace))
3361 : {
3362 0 : InvokeNamespaceSearchHook(myTempNamespace, true);
3363 0 : return myTempNamespace;
3364 : }
3365 :
3366 : /*
3367 : * Since this is used only for looking up existing objects, there is
3368 : * no point in trying to initialize the temp namespace here; and doing
3369 : * so might create problems for some callers. Just report "not found".
3370 : */
3371 0 : return InvalidOid;
3372 : }
3373 :
3374 342 : return get_namespace_oid(nspname, true);
3375 : }
3376 :
3377 : /*
3378 : * LookupExplicitNamespace
3379 : * Process an explicitly-specified schema name: look up the schema
3380 : * and verify we have USAGE (lookup) rights in it.
3381 : *
3382 : * Returns the namespace OID
3383 : */
3384 : Oid
3385 660276 : LookupExplicitNamespace(const char *nspname, bool missing_ok)
3386 : {
3387 : Oid namespaceId;
3388 : AclResult aclresult;
3389 :
3390 : /* check for pg_temp alias */
3391 660276 : if (strcmp(nspname, "pg_temp") == 0)
3392 : {
3393 318 : if (OidIsValid(myTempNamespace))
3394 318 : return myTempNamespace;
3395 :
3396 : /*
3397 : * Since this is used only for looking up existing objects, there is
3398 : * no point in trying to initialize the temp namespace here; and doing
3399 : * so might create problems for some callers --- just fall through.
3400 : */
3401 : }
3402 :
3403 659958 : namespaceId = get_namespace_oid(nspname, missing_ok);
3404 659846 : if (missing_ok && !OidIsValid(namespaceId))
3405 336 : return InvalidOid;
3406 :
3407 659510 : aclresult = object_aclcheck(NamespaceRelationId, namespaceId, GetUserId(), ACL_USAGE);
3408 659510 : if (aclresult != ACLCHECK_OK)
3409 8 : aclcheck_error(aclresult, OBJECT_SCHEMA,
3410 : nspname);
3411 : /* Schema search hook for this lookup */
3412 659502 : InvokeNamespaceSearchHook(namespaceId, true);
3413 :
3414 659502 : return namespaceId;
3415 : }
3416 :
3417 : /*
3418 : * LookupCreationNamespace
3419 : * Look up the schema and verify we have CREATE rights on it.
3420 : *
3421 : * This is just like LookupExplicitNamespace except for the different
3422 : * permission check, and that we are willing to create pg_temp if needed.
3423 : *
3424 : * Note: calling this may result in a CommandCounterIncrement operation,
3425 : * if we have to create or clean out the temp namespace.
3426 : */
3427 : Oid
3428 486 : LookupCreationNamespace(const char *nspname)
3429 : {
3430 : Oid namespaceId;
3431 : AclResult aclresult;
3432 :
3433 : /* check for pg_temp alias */
3434 486 : if (strcmp(nspname, "pg_temp") == 0)
3435 : {
3436 : /* Initialize temp namespace */
3437 152 : AccessTempTableNamespace(false);
3438 152 : return myTempNamespace;
3439 : }
3440 :
3441 334 : namespaceId = get_namespace_oid(nspname, false);
3442 :
3443 332 : aclresult = object_aclcheck(NamespaceRelationId, namespaceId, GetUserId(), ACL_CREATE);
3444 332 : if (aclresult != ACLCHECK_OK)
3445 0 : aclcheck_error(aclresult, OBJECT_SCHEMA,
3446 : nspname);
3447 :
3448 332 : return namespaceId;
3449 : }
3450 :
3451 : /*
3452 : * Common checks on switching namespaces.
3453 : *
3454 : * We complain if either the old or new namespaces is a temporary schema
3455 : * (or temporary toast schema), or if either the old or new namespaces is the
3456 : * TOAST schema.
3457 : */
3458 : void
3459 540 : CheckSetNamespace(Oid oldNspOid, Oid nspOid)
3460 : {
3461 : /* disallow renaming into or out of temp schemas */
3462 540 : if (isAnyTempNamespace(nspOid) || isAnyTempNamespace(oldNspOid))
3463 0 : ereport(ERROR,
3464 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
3465 : errmsg("cannot move objects into or out of temporary schemas")));
3466 :
3467 : /* same for TOAST schema */
3468 540 : if (nspOid == PG_TOAST_NAMESPACE || oldNspOid == PG_TOAST_NAMESPACE)
3469 0 : ereport(ERROR,
3470 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
3471 : errmsg("cannot move objects into or out of TOAST schema")));
3472 540 : }
3473 :
3474 : /*
3475 : * QualifiedNameGetCreationNamespace
3476 : * Given a possibly-qualified name for an object (in List-of-Strings
3477 : * format), determine what namespace the object should be created in.
3478 : * Also extract and return the object name (last component of list).
3479 : *
3480 : * Note: this does not apply any permissions check. Callers must check
3481 : * for CREATE rights on the selected namespace when appropriate.
3482 : *
3483 : * Note: calling this may result in a CommandCounterIncrement operation,
3484 : * if we have to create or clean out the temp namespace.
3485 : */
3486 : Oid
3487 34746 : QualifiedNameGetCreationNamespace(const List *names, char **objname_p)
3488 : {
3489 : char *schemaname;
3490 : Oid namespaceId;
3491 :
3492 : /* deconstruct the name list */
3493 34746 : DeconstructQualifiedName(names, &schemaname, objname_p);
3494 :
3495 34746 : if (schemaname)
3496 : {
3497 : /* check for pg_temp alias */
3498 1572 : if (strcmp(schemaname, "pg_temp") == 0)
3499 : {
3500 : /* Initialize temp namespace */
3501 300 : AccessTempTableNamespace(false);
3502 300 : return myTempNamespace;
3503 : }
3504 : /* use exact schema given */
3505 1272 : namespaceId = get_namespace_oid(schemaname, false);
3506 : /* we do not check for USAGE rights here! */
3507 : }
3508 : else
3509 : {
3510 : /* use the default creation namespace */
3511 33174 : recomputeNamespacePath();
3512 33174 : if (activeTempCreationPending)
3513 : {
3514 : /* Need to initialize temp namespace */
3515 0 : AccessTempTableNamespace(true);
3516 0 : return myTempNamespace;
3517 : }
3518 33174 : namespaceId = activeCreationNamespace;
3519 33174 : if (!OidIsValid(namespaceId))
3520 0 : ereport(ERROR,
3521 : (errcode(ERRCODE_UNDEFINED_SCHEMA),
3522 : errmsg("no schema has been selected to create in")));
3523 : }
3524 :
3525 34446 : return namespaceId;
3526 : }
3527 :
3528 : /*
3529 : * get_namespace_oid - given a namespace name, look up the OID
3530 : *
3531 : * If missing_ok is false, throw an error if namespace name not found. If
3532 : * true, just return InvalidOid.
3533 : */
3534 : Oid
3535 766196 : get_namespace_oid(const char *nspname, bool missing_ok)
3536 : {
3537 : Oid oid;
3538 :
3539 766196 : oid = GetSysCacheOid1(NAMESPACENAME, Anum_pg_namespace_oid,
3540 : CStringGetDatum(nspname));
3541 766196 : if (!OidIsValid(oid) && !missing_ok)
3542 178 : ereport(ERROR,
3543 : (errcode(ERRCODE_UNDEFINED_SCHEMA),
3544 : errmsg("schema \"%s\" does not exist", nspname)));
3545 :
3546 766018 : return oid;
3547 : }
3548 :
3549 : /*
3550 : * makeRangeVarFromNameList
3551 : * Utility routine to convert a qualified-name list into RangeVar form.
3552 : */
3553 : RangeVar *
3554 58188 : makeRangeVarFromNameList(const List *names)
3555 : {
3556 58188 : RangeVar *rel = makeRangeVar(NULL, NULL, -1);
3557 :
3558 58188 : switch (list_length(names))
3559 : {
3560 38610 : case 1:
3561 38610 : rel->relname = strVal(linitial(names));
3562 38610 : break;
3563 19498 : case 2:
3564 19498 : rel->schemaname = strVal(linitial(names));
3565 19498 : rel->relname = strVal(lsecond(names));
3566 19498 : break;
3567 80 : case 3:
3568 80 : rel->catalogname = strVal(linitial(names));
3569 80 : rel->schemaname = strVal(lsecond(names));
3570 80 : rel->relname = strVal(lthird(names));
3571 80 : break;
3572 0 : default:
3573 0 : ereport(ERROR,
3574 : (errcode(ERRCODE_SYNTAX_ERROR),
3575 : errmsg("improper relation name (too many dotted names): %s",
3576 : NameListToString(names))));
3577 : break;
3578 : }
3579 :
3580 58188 : return rel;
3581 : }
3582 :
3583 : /*
3584 : * NameListToString
3585 : * Utility routine to convert a qualified-name list into a string.
3586 : *
3587 : * This is used primarily to form error messages, and so we do not quote
3588 : * the list elements, for the sake of legibility.
3589 : *
3590 : * In most scenarios the list elements should always be String values,
3591 : * but we also allow A_Star for the convenience of ColumnRef processing.
3592 : */
3593 : char *
3594 1734 : NameListToString(const List *names)
3595 : {
3596 : StringInfoData string;
3597 : ListCell *l;
3598 :
3599 1734 : initStringInfo(&string);
3600 :
3601 3912 : foreach(l, names)
3602 : {
3603 2178 : Node *name = (Node *) lfirst(l);
3604 :
3605 2178 : if (l != list_head(names))
3606 444 : appendStringInfoChar(&string, '.');
3607 :
3608 2178 : if (IsA(name, String))
3609 2178 : appendStringInfoString(&string, strVal(name));
3610 0 : else if (IsA(name, A_Star))
3611 0 : appendStringInfoChar(&string, '*');
3612 : else
3613 0 : elog(ERROR, "unexpected node type in name list: %d",
3614 : (int) nodeTag(name));
3615 : }
3616 :
3617 1734 : return string.data;
3618 : }
3619 :
3620 : /*
3621 : * NameListToQuotedString
3622 : * Utility routine to convert a qualified-name list into a string.
3623 : *
3624 : * Same as above except that names will be double-quoted where necessary,
3625 : * so the string could be re-parsed (eg, by textToQualifiedNameList).
3626 : */
3627 : char *
3628 0 : NameListToQuotedString(const List *names)
3629 : {
3630 : StringInfoData string;
3631 : ListCell *l;
3632 :
3633 0 : initStringInfo(&string);
3634 :
3635 0 : foreach(l, names)
3636 : {
3637 0 : if (l != list_head(names))
3638 0 : appendStringInfoChar(&string, '.');
3639 0 : appendStringInfoString(&string, quote_identifier(strVal(lfirst(l))));
3640 : }
3641 :
3642 0 : return string.data;
3643 : }
3644 :
3645 : /*
3646 : * isTempNamespace - is the given namespace my temporary-table namespace?
3647 : */
3648 : bool
3649 66806 : isTempNamespace(Oid namespaceId)
3650 : {
3651 66806 : if (OidIsValid(myTempNamespace) && myTempNamespace == namespaceId)
3652 904 : return true;
3653 65902 : return false;
3654 : }
3655 :
3656 : /*
3657 : * isTempToastNamespace - is the given namespace my temporary-toast-table
3658 : * namespace?
3659 : */
3660 : bool
3661 6207962 : isTempToastNamespace(Oid namespaceId)
3662 : {
3663 6207962 : if (OidIsValid(myTempToastNamespace) && myTempToastNamespace == namespaceId)
3664 3584 : return true;
3665 6204378 : return false;
3666 : }
3667 :
3668 : /*
3669 : * isTempOrTempToastNamespace - is the given namespace my temporary-table
3670 : * namespace or my temporary-toast-table namespace?
3671 : */
3672 : bool
3673 174704 : isTempOrTempToastNamespace(Oid namespaceId)
3674 : {
3675 174704 : if (OidIsValid(myTempNamespace) &&
3676 68526 : (myTempNamespace == namespaceId || myTempToastNamespace == namespaceId))
3677 34696 : return true;
3678 140008 : return false;
3679 : }
3680 :
3681 : /*
3682 : * isAnyTempNamespace - is the given namespace a temporary-table namespace
3683 : * (either my own, or another backend's)? Temporary-toast-table namespaces
3684 : * are included, too.
3685 : */
3686 : bool
3687 128888 : isAnyTempNamespace(Oid namespaceId)
3688 : {
3689 : bool result;
3690 : char *nspname;
3691 :
3692 : /* True if the namespace name starts with "pg_temp_" or "pg_toast_temp_" */
3693 128888 : nspname = get_namespace_name(namespaceId);
3694 128888 : if (!nspname)
3695 0 : return false; /* no such namespace? */
3696 253048 : result = (strncmp(nspname, "pg_temp_", 8) == 0) ||
3697 124160 : (strncmp(nspname, "pg_toast_temp_", 14) == 0);
3698 128888 : pfree(nspname);
3699 128888 : return result;
3700 : }
3701 :
3702 : /*
3703 : * isOtherTempNamespace - is the given namespace some other backend's
3704 : * temporary-table namespace (including temporary-toast-table namespaces)?
3705 : *
3706 : * Note: for most purposes in the C code, this function is obsolete. Use
3707 : * RELATION_IS_OTHER_TEMP() instead to detect non-local temp relations.
3708 : */
3709 : bool
3710 14192 : isOtherTempNamespace(Oid namespaceId)
3711 : {
3712 : /* If it's my own temp namespace, say "false" */
3713 14192 : if (isTempOrTempToastNamespace(namespaceId))
3714 252 : return false;
3715 : /* Else, if it's any temp namespace, say "true" */
3716 13940 : return isAnyTempNamespace(namespaceId);
3717 : }
3718 :
3719 : /*
3720 : * checkTempNamespaceStatus - is the given namespace owned and actively used
3721 : * by a backend?
3722 : *
3723 : * Note: this can be used while scanning relations in pg_class to detect
3724 : * orphaned temporary tables or namespaces with a backend connected to a
3725 : * given database. The result may be out of date quickly, so the caller
3726 : * must be careful how to handle this information.
3727 : */
3728 : TempNamespaceStatus
3729 2 : checkTempNamespaceStatus(Oid namespaceId)
3730 : {
3731 : PGPROC *proc;
3732 : ProcNumber procNumber;
3733 :
3734 : Assert(OidIsValid(MyDatabaseId));
3735 :
3736 2 : procNumber = GetTempNamespaceProcNumber(namespaceId);
3737 :
3738 : /* No such namespace, or its name shows it's not temp? */
3739 2 : if (procNumber == INVALID_PROC_NUMBER)
3740 0 : return TEMP_NAMESPACE_NOT_TEMP;
3741 :
3742 : /* Is the backend alive? */
3743 2 : proc = ProcNumberGetProc(procNumber);
3744 2 : if (proc == NULL)
3745 0 : return TEMP_NAMESPACE_IDLE;
3746 :
3747 : /* Is the backend connected to the same database we are looking at? */
3748 2 : if (proc->databaseId != MyDatabaseId)
3749 0 : return TEMP_NAMESPACE_IDLE;
3750 :
3751 : /* Does the backend own the temporary namespace? */
3752 2 : if (proc->tempNamespaceId != namespaceId)
3753 0 : return TEMP_NAMESPACE_IDLE;
3754 :
3755 : /* Yup, so namespace is busy */
3756 2 : return TEMP_NAMESPACE_IN_USE;
3757 : }
3758 :
3759 : /*
3760 : * GetTempNamespaceProcNumber - if the given namespace is a temporary-table
3761 : * namespace (either my own, or another backend's), return the proc number
3762 : * that owns it. Temporary-toast-table namespaces are included, too.
3763 : * If it isn't a temp namespace, return INVALID_PROC_NUMBER.
3764 : */
3765 : ProcNumber
3766 38 : GetTempNamespaceProcNumber(Oid namespaceId)
3767 : {
3768 : int result;
3769 : char *nspname;
3770 :
3771 : /* See if the namespace name starts with "pg_temp_" or "pg_toast_temp_" */
3772 38 : nspname = get_namespace_name(namespaceId);
3773 38 : if (!nspname)
3774 0 : return INVALID_PROC_NUMBER; /* no such namespace? */
3775 38 : if (strncmp(nspname, "pg_temp_", 8) == 0)
3776 38 : result = atoi(nspname + 8);
3777 0 : else if (strncmp(nspname, "pg_toast_temp_", 14) == 0)
3778 0 : result = atoi(nspname + 14);
3779 : else
3780 0 : result = INVALID_PROC_NUMBER;
3781 38 : pfree(nspname);
3782 38 : return result;
3783 : }
3784 :
3785 : /*
3786 : * GetTempToastNamespace - get the OID of my temporary-toast-table namespace,
3787 : * which must already be assigned. (This is only used when creating a toast
3788 : * table for a temp table, so we must have already done InitTempTableNamespace)
3789 : */
3790 : Oid
3791 916 : GetTempToastNamespace(void)
3792 : {
3793 : Assert(OidIsValid(myTempToastNamespace));
3794 916 : return myTempToastNamespace;
3795 : }
3796 :
3797 :
3798 : /*
3799 : * GetTempNamespaceState - fetch status of session's temporary namespace
3800 : *
3801 : * This is used for conveying state to a parallel worker, and is not meant
3802 : * for general-purpose access.
3803 : */
3804 : void
3805 892 : GetTempNamespaceState(Oid *tempNamespaceId, Oid *tempToastNamespaceId)
3806 : {
3807 : /* Return namespace OIDs, or 0 if session has not created temp namespace */
3808 892 : *tempNamespaceId = myTempNamespace;
3809 892 : *tempToastNamespaceId = myTempToastNamespace;
3810 892 : }
3811 :
3812 : /*
3813 : * SetTempNamespaceState - set status of session's temporary namespace
3814 : *
3815 : * This is used for conveying state to a parallel worker, and is not meant for
3816 : * general-purpose access. By transferring these namespace OIDs to workers,
3817 : * we ensure they will have the same notion of the search path as their leader
3818 : * does.
3819 : */
3820 : void
3821 2714 : SetTempNamespaceState(Oid tempNamespaceId, Oid tempToastNamespaceId)
3822 : {
3823 : /* Worker should not have created its own namespaces ... */
3824 : Assert(myTempNamespace == InvalidOid);
3825 : Assert(myTempToastNamespace == InvalidOid);
3826 : Assert(myTempNamespaceSubID == InvalidSubTransactionId);
3827 :
3828 : /* Assign same namespace OIDs that leader has */
3829 2714 : myTempNamespace = tempNamespaceId;
3830 2714 : myTempToastNamespace = tempToastNamespaceId;
3831 :
3832 : /*
3833 : * It's fine to leave myTempNamespaceSubID == InvalidSubTransactionId.
3834 : * Even if the namespace is new so far as the leader is concerned, it's
3835 : * not new to the worker, and we certainly wouldn't want the worker trying
3836 : * to destroy it.
3837 : */
3838 :
3839 2714 : baseSearchPathValid = false; /* may need to rebuild list */
3840 2714 : searchPathCacheValid = false;
3841 2714 : }
3842 :
3843 :
3844 : /*
3845 : * GetSearchPathMatcher - fetch current search path definition.
3846 : *
3847 : * The result structure is allocated in the specified memory context
3848 : * (which might or might not be equal to CurrentMemoryContext); but any
3849 : * junk created by revalidation calculations will be in CurrentMemoryContext.
3850 : */
3851 : SearchPathMatcher *
3852 58352 : GetSearchPathMatcher(MemoryContext context)
3853 : {
3854 : SearchPathMatcher *result;
3855 : List *schemas;
3856 : MemoryContext oldcxt;
3857 :
3858 58352 : recomputeNamespacePath();
3859 :
3860 58352 : oldcxt = MemoryContextSwitchTo(context);
3861 :
3862 58352 : result = (SearchPathMatcher *) palloc0(sizeof(SearchPathMatcher));
3863 58352 : schemas = list_copy(activeSearchPath);
3864 125038 : while (schemas && linitial_oid(schemas) != activeCreationNamespace)
3865 : {
3866 66686 : if (linitial_oid(schemas) == myTempNamespace)
3867 10634 : result->addTemp = true;
3868 : else
3869 : {
3870 : Assert(linitial_oid(schemas) == PG_CATALOG_NAMESPACE);
3871 56052 : result->addCatalog = true;
3872 : }
3873 66686 : schemas = list_delete_first(schemas);
3874 : }
3875 58352 : result->schemas = schemas;
3876 58352 : result->generation = activePathGeneration;
3877 :
3878 58352 : MemoryContextSwitchTo(oldcxt);
3879 :
3880 58352 : return result;
3881 : }
3882 :
3883 : /*
3884 : * CopySearchPathMatcher - copy the specified SearchPathMatcher.
3885 : *
3886 : * The result structure is allocated in CurrentMemoryContext.
3887 : */
3888 : SearchPathMatcher *
3889 0 : CopySearchPathMatcher(SearchPathMatcher *path)
3890 : {
3891 : SearchPathMatcher *result;
3892 :
3893 0 : result = (SearchPathMatcher *) palloc(sizeof(SearchPathMatcher));
3894 0 : result->schemas = list_copy(path->schemas);
3895 0 : result->addCatalog = path->addCatalog;
3896 0 : result->addTemp = path->addTemp;
3897 0 : result->generation = path->generation;
3898 :
3899 0 : return result;
3900 : }
3901 :
3902 : /*
3903 : * SearchPathMatchesCurrentEnvironment - does path match current environment?
3904 : *
3905 : * This is tested over and over in some common code paths, and in the typical
3906 : * scenario where the active search path seldom changes, it'll always succeed.
3907 : * We make that case fast by keeping a generation counter that is advanced
3908 : * whenever the active search path changes.
3909 : */
3910 : bool
3911 450270 : SearchPathMatchesCurrentEnvironment(SearchPathMatcher *path)
3912 : {
3913 : ListCell *lc,
3914 : *lcp;
3915 :
3916 450270 : recomputeNamespacePath();
3917 :
3918 : /* Quick out if already known equal to active path. */
3919 450270 : if (path->generation == activePathGeneration)
3920 450032 : return true;
3921 :
3922 : /* We scan down the activeSearchPath to see if it matches the input. */
3923 238 : lc = list_head(activeSearchPath);
3924 :
3925 : /* If path->addTemp, first item should be my temp namespace. */
3926 238 : if (path->addTemp)
3927 : {
3928 20 : if (lc && lfirst_oid(lc) == myTempNamespace)
3929 20 : lc = lnext(activeSearchPath, lc);
3930 : else
3931 0 : return false;
3932 : }
3933 : /* If path->addCatalog, next item should be pg_catalog. */
3934 238 : if (path->addCatalog)
3935 : {
3936 138 : if (lc && lfirst_oid(lc) == PG_CATALOG_NAMESPACE)
3937 100 : lc = lnext(activeSearchPath, lc);
3938 : else
3939 38 : return false;
3940 : }
3941 : /* We should now be looking at the activeCreationNamespace. */
3942 200 : if (activeCreationNamespace != (lc ? lfirst_oid(lc) : InvalidOid))
3943 0 : return false;
3944 : /* The remainder of activeSearchPath should match path->schemas. */
3945 382 : foreach(lcp, path->schemas)
3946 : {
3947 220 : if (lc && lfirst_oid(lc) == lfirst_oid(lcp))
3948 182 : lc = lnext(activeSearchPath, lc);
3949 : else
3950 38 : return false;
3951 : }
3952 162 : if (lc)
3953 12 : return false;
3954 :
3955 : /*
3956 : * Update path->generation so that future tests will return quickly, so
3957 : * long as the active search path doesn't change.
3958 : */
3959 150 : path->generation = activePathGeneration;
3960 :
3961 150 : return true;
3962 : }
3963 :
3964 : /*
3965 : * get_collation_oid - find a collation by possibly qualified name
3966 : *
3967 : * Note that this will only find collations that work with the current
3968 : * database's encoding.
3969 : */
3970 : Oid
3971 10298 : get_collation_oid(List *collname, bool missing_ok)
3972 : {
3973 : char *schemaname;
3974 : char *collation_name;
3975 10298 : int32 dbencoding = GetDatabaseEncoding();
3976 : Oid namespaceId;
3977 : Oid colloid;
3978 : ListCell *l;
3979 :
3980 : /* deconstruct the name list */
3981 10298 : DeconstructQualifiedName(collname, &schemaname, &collation_name);
3982 :
3983 10298 : if (schemaname)
3984 : {
3985 : /* use exact schema given */
3986 6646 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
3987 6646 : if (missing_ok && !OidIsValid(namespaceId))
3988 24 : return InvalidOid;
3989 :
3990 6622 : colloid = lookup_collation(collation_name, namespaceId, dbencoding);
3991 6622 : if (OidIsValid(colloid))
3992 6622 : return colloid;
3993 : }
3994 : else
3995 : {
3996 : /* search for it in search path */
3997 3652 : recomputeNamespacePath();
3998 :
3999 6086 : foreach(l, activeSearchPath)
4000 : {
4001 6036 : namespaceId = lfirst_oid(l);
4002 :
4003 6036 : if (namespaceId == myTempNamespace)
4004 1228 : continue; /* do not look in temp namespace */
4005 :
4006 4808 : colloid = lookup_collation(collation_name, namespaceId, dbencoding);
4007 4808 : if (OidIsValid(colloid))
4008 3602 : return colloid;
4009 : }
4010 : }
4011 :
4012 : /* Not found in path */
4013 50 : if (!missing_ok)
4014 32 : ereport(ERROR,
4015 : (errcode(ERRCODE_UNDEFINED_OBJECT),
4016 : errmsg("collation \"%s\" for encoding \"%s\" does not exist",
4017 : NameListToString(collname), GetDatabaseEncodingName())));
4018 18 : return InvalidOid;
4019 : }
4020 :
4021 : /*
4022 : * get_conversion_oid - find a conversion by possibly qualified name
4023 : */
4024 : Oid
4025 178 : get_conversion_oid(List *conname, bool missing_ok)
4026 : {
4027 : char *schemaname;
4028 : char *conversion_name;
4029 : Oid namespaceId;
4030 178 : Oid conoid = InvalidOid;
4031 : ListCell *l;
4032 :
4033 : /* deconstruct the name list */
4034 178 : DeconstructQualifiedName(conname, &schemaname, &conversion_name);
4035 :
4036 166 : if (schemaname)
4037 : {
4038 : /* use exact schema given */
4039 38 : namespaceId = LookupExplicitNamespace(schemaname, missing_ok);
4040 38 : if (missing_ok && !OidIsValid(namespaceId))
4041 6 : conoid = InvalidOid;
4042 : else
4043 32 : conoid = GetSysCacheOid2(CONNAMENSP, Anum_pg_conversion_oid,
4044 : PointerGetDatum(conversion_name),
4045 : ObjectIdGetDatum(namespaceId));
4046 : }
4047 : else
4048 : {
4049 : /* search for it in search path */
4050 128 : recomputeNamespacePath();
4051 :
4052 286 : foreach(l, activeSearchPath)
4053 : {
4054 256 : namespaceId = lfirst_oid(l);
4055 :
4056 256 : if (namespaceId == myTempNamespace)
4057 0 : continue; /* do not look in temp namespace */
4058 :
4059 256 : conoid = GetSysCacheOid2(CONNAMENSP, Anum_pg_conversion_oid,
4060 : PointerGetDatum(conversion_name),
4061 : ObjectIdGetDatum(namespaceId));
4062 256 : if (OidIsValid(conoid))
4063 98 : return conoid;
4064 : }
4065 : }
4066 :
4067 : /* Not found in path */
4068 68 : if (!OidIsValid(conoid) && !missing_ok)
4069 36 : ereport(ERROR,
4070 : (errcode(ERRCODE_UNDEFINED_OBJECT),
4071 : errmsg("conversion \"%s\" does not exist",
4072 : NameListToString(conname))));
4073 32 : return conoid;
4074 : }
4075 :
4076 : /*
4077 : * FindDefaultConversionProc - find default encoding conversion proc
4078 : */
4079 : Oid
4080 6736 : FindDefaultConversionProc(int32 for_encoding, int32 to_encoding)
4081 : {
4082 : Oid proc;
4083 : ListCell *l;
4084 :
4085 6736 : recomputeNamespacePath();
4086 :
4087 6736 : foreach(l, activeSearchPath)
4088 : {
4089 6736 : Oid namespaceId = lfirst_oid(l);
4090 :
4091 6736 : if (namespaceId == myTempNamespace)
4092 0 : continue; /* do not look in temp namespace */
4093 :
4094 6736 : proc = FindDefaultConversion(namespaceId, for_encoding, to_encoding);
4095 6736 : if (OidIsValid(proc))
4096 6736 : return proc;
4097 : }
4098 :
4099 : /* Not found in path */
4100 0 : return InvalidOid;
4101 : }
4102 :
4103 : /*
4104 : * Look up namespace IDs and perform ACL checks. Return newly-allocated list.
4105 : */
4106 : static List *
4107 42332 : preprocessNamespacePath(const char *searchPath, Oid roleid,
4108 : bool *temp_missing)
4109 : {
4110 : char *rawname;
4111 : List *namelist;
4112 : List *oidlist;
4113 : ListCell *l;
4114 :
4115 : /* Need a modifiable copy */
4116 42332 : rawname = pstrdup(searchPath);
4117 :
4118 : /* Parse string into list of identifiers */
4119 42332 : if (!SplitIdentifierString(rawname, ',', &namelist))
4120 : {
4121 : /* syntax error in name list */
4122 : /* this should not happen if GUC checked check_search_path */
4123 0 : elog(ERROR, "invalid list syntax");
4124 : }
4125 :
4126 : /*
4127 : * Convert the list of names to a list of OIDs. If any names are not
4128 : * recognizable or we don't have read access, just leave them out of the
4129 : * list. (We can't raise an error, since the search_path setting has
4130 : * already been accepted.) Don't make duplicate entries, either.
4131 : */
4132 42332 : oidlist = NIL;
4133 42332 : *temp_missing = false;
4134 104104 : foreach(l, namelist)
4135 : {
4136 61772 : char *curname = (char *) lfirst(l);
4137 : Oid namespaceId;
4138 :
4139 61772 : if (strcmp(curname, "$user") == 0)
4140 : {
4141 : /* $user --- substitute namespace matching user name, if any */
4142 : HeapTuple tuple;
4143 :
4144 28468 : tuple = SearchSysCache1(AUTHOID, ObjectIdGetDatum(roleid));
4145 28468 : if (HeapTupleIsValid(tuple))
4146 : {
4147 : char *rname;
4148 :
4149 28468 : rname = NameStr(((Form_pg_authid) GETSTRUCT(tuple))->rolname);
4150 28468 : namespaceId = get_namespace_oid(rname, true);
4151 28468 : ReleaseSysCache(tuple);
4152 28476 : if (OidIsValid(namespaceId) &&
4153 8 : object_aclcheck(NamespaceRelationId, namespaceId, roleid,
4154 : ACL_USAGE) == ACLCHECK_OK)
4155 8 : oidlist = lappend_oid(oidlist, namespaceId);
4156 : }
4157 : }
4158 33304 : else if (strcmp(curname, "pg_temp") == 0)
4159 : {
4160 : /* pg_temp --- substitute temp namespace, if any */
4161 1034 : if (OidIsValid(myTempNamespace))
4162 260 : oidlist = lappend_oid(oidlist, myTempNamespace);
4163 : else
4164 : {
4165 : /* If it ought to be the creation namespace, set flag */
4166 774 : if (oidlist == NIL)
4167 6 : *temp_missing = true;
4168 : }
4169 : }
4170 : else
4171 : {
4172 : /* normal namespace reference */
4173 32270 : namespaceId = get_namespace_oid(curname, true);
4174 64474 : if (OidIsValid(namespaceId) &&
4175 32204 : object_aclcheck(NamespaceRelationId, namespaceId, roleid,
4176 : ACL_USAGE) == ACLCHECK_OK)
4177 32198 : oidlist = lappend_oid(oidlist, namespaceId);
4178 : }
4179 : }
4180 :
4181 42332 : pfree(rawname);
4182 42332 : list_free(namelist);
4183 :
4184 42332 : return oidlist;
4185 : }
4186 :
4187 : /*
4188 : * Remove duplicates, run namespace search hooks, and prepend
4189 : * implicitly-searched namespaces. Return newly-allocated list.
4190 : *
4191 : * If an object_access_hook is present, this must always be recalculated. It
4192 : * may seem that duplicate elimination is not dependent on the result of the
4193 : * hook, but if a hook returns different results on different calls for the
4194 : * same namespace ID, then it could affect the order in which that namespace
4195 : * appears in the final list.
4196 : */
4197 : static List *
4198 33712 : finalNamespacePath(List *oidlist, Oid *firstNS)
4199 : {
4200 33712 : List *finalPath = NIL;
4201 : ListCell *lc;
4202 :
4203 66186 : foreach(lc, oidlist)
4204 : {
4205 32474 : Oid namespaceId = lfirst_oid(lc);
4206 :
4207 32474 : if (!list_member_oid(finalPath, namespaceId))
4208 : {
4209 32460 : if (InvokeNamespaceSearchHook(namespaceId, false))
4210 32460 : finalPath = lappend_oid(finalPath, namespaceId);
4211 : }
4212 : }
4213 :
4214 : /*
4215 : * Remember the first member of the explicit list. (Note: this is
4216 : * nominally wrong if temp_missing, but we need it anyway to distinguish
4217 : * explicit from implicit mention of pg_catalog.)
4218 : */
4219 33712 : if (finalPath == NIL)
4220 1892 : *firstNS = InvalidOid;
4221 : else
4222 31820 : *firstNS = linitial_oid(finalPath);
4223 :
4224 : /*
4225 : * Add any implicitly-searched namespaces to the list. Note these go on
4226 : * the front, not the back; also notice that we do not check USAGE
4227 : * permissions for these.
4228 : */
4229 33712 : if (!list_member_oid(finalPath, PG_CATALOG_NAMESPACE))
4230 32628 : finalPath = lcons_oid(PG_CATALOG_NAMESPACE, finalPath);
4231 :
4232 33712 : if (OidIsValid(myTempNamespace) &&
4233 3644 : !list_member_oid(finalPath, myTempNamespace))
4234 3384 : finalPath = lcons_oid(myTempNamespace, finalPath);
4235 :
4236 33712 : return finalPath;
4237 : }
4238 :
4239 : /*
4240 : * Retrieve search path information from the cache; or if not there, fill
4241 : * it. The returned entry is valid only until the next call to this function.
4242 : */
4243 : static const SearchPathCacheEntry *
4244 75946 : cachedNamespacePath(const char *searchPath, Oid roleid)
4245 : {
4246 : MemoryContext oldcxt;
4247 : SearchPathCacheEntry *entry;
4248 :
4249 75946 : spcache_init();
4250 :
4251 75946 : entry = spcache_insert(searchPath, roleid);
4252 :
4253 : /*
4254 : * An OOM may have resulted in a cache entry with missing 'oidlist' or
4255 : * 'finalPath', so just compute whatever is missing.
4256 : */
4257 :
4258 75946 : if (entry->oidlist == NIL)
4259 : {
4260 42332 : oldcxt = MemoryContextSwitchTo(SearchPathCacheContext);
4261 42332 : entry->oidlist = preprocessNamespacePath(searchPath, roleid,
4262 : &entry->temp_missing);
4263 42332 : MemoryContextSwitchTo(oldcxt);
4264 : }
4265 :
4266 : /*
4267 : * If a hook is set, we must recompute finalPath from the oidlist each
4268 : * time, because the hook may affect the result. This is still much faster
4269 : * than recomputing from the string (and doing catalog lookups and ACL
4270 : * checks).
4271 : */
4272 75946 : if (entry->finalPath == NIL || object_access_hook ||
4273 42234 : entry->forceRecompute)
4274 : {
4275 33712 : list_free(entry->finalPath);
4276 33712 : entry->finalPath = NIL;
4277 :
4278 33712 : oldcxt = MemoryContextSwitchTo(SearchPathCacheContext);
4279 33712 : entry->finalPath = finalNamespacePath(entry->oidlist,
4280 : &entry->firstNS);
4281 33712 : MemoryContextSwitchTo(oldcxt);
4282 :
4283 : /*
4284 : * If an object_access_hook is set when finalPath is calculated, the
4285 : * result may be affected by the hook. Force recomputation of
4286 : * finalPath the next time this cache entry is used, even if the
4287 : * object_access_hook is not set at that time.
4288 : */
4289 33712 : entry->forceRecompute = object_access_hook ? true : false;
4290 : }
4291 :
4292 75946 : return entry;
4293 : }
4294 :
4295 : /*
4296 : * recomputeNamespacePath - recompute path derived variables if needed.
4297 : */
4298 : static void
4299 3380716 : recomputeNamespacePath(void)
4300 : {
4301 3380716 : Oid roleid = GetUserId();
4302 : bool pathChanged;
4303 : const SearchPathCacheEntry *entry;
4304 :
4305 : /* Do nothing if path is already valid. */
4306 3380716 : if (baseSearchPathValid && namespaceUser == roleid)
4307 3304770 : return;
4308 :
4309 75946 : entry = cachedNamespacePath(namespace_search_path, roleid);
4310 :
4311 75946 : if (baseCreationNamespace == entry->firstNS &&
4312 108122 : baseTempCreationPending == entry->temp_missing &&
4313 54058 : equal(entry->finalPath, baseSearchPath))
4314 : {
4315 51632 : pathChanged = false;
4316 : }
4317 : else
4318 : {
4319 : MemoryContext oldcxt;
4320 : List *newpath;
4321 :
4322 24314 : pathChanged = true;
4323 :
4324 : /* Must save OID list in permanent storage. */
4325 24314 : oldcxt = MemoryContextSwitchTo(TopMemoryContext);
4326 24314 : newpath = list_copy(entry->finalPath);
4327 24314 : MemoryContextSwitchTo(oldcxt);
4328 :
4329 : /* Now safe to assign to state variables. */
4330 24314 : list_free(baseSearchPath);
4331 24314 : baseSearchPath = newpath;
4332 24314 : baseCreationNamespace = entry->firstNS;
4333 24314 : baseTempCreationPending = entry->temp_missing;
4334 : }
4335 :
4336 : /* Mark the path valid. */
4337 75946 : baseSearchPathValid = true;
4338 75946 : namespaceUser = roleid;
4339 :
4340 : /* And make it active. */
4341 75946 : activeSearchPath = baseSearchPath;
4342 75946 : activeCreationNamespace = baseCreationNamespace;
4343 75946 : activeTempCreationPending = baseTempCreationPending;
4344 :
4345 : /*
4346 : * Bump the generation only if something actually changed. (Notice that
4347 : * what we compared to was the old state of the base path variables.)
4348 : */
4349 75946 : if (pathChanged)
4350 24314 : activePathGeneration++;
4351 : }
4352 :
4353 : /*
4354 : * AccessTempTableNamespace
4355 : * Provide access to a temporary namespace, potentially creating it
4356 : * if not present yet. This routine registers if the namespace gets
4357 : * in use in this transaction. 'force' can be set to true to allow
4358 : * the caller to enforce the creation of the temporary namespace for
4359 : * use in this backend, which happens if its creation is pending.
4360 : */
4361 : static void
4362 6730 : AccessTempTableNamespace(bool force)
4363 : {
4364 : /*
4365 : * Make note that this temporary namespace has been accessed in this
4366 : * transaction.
4367 : */
4368 6730 : MyXactFlags |= XACT_FLAGS_ACCESSEDTEMPNAMESPACE;
4369 :
4370 : /*
4371 : * If the caller attempting to access a temporary schema expects the
4372 : * creation of the namespace to be pending and should be enforced, then go
4373 : * through the creation.
4374 : */
4375 6730 : if (!force && OidIsValid(myTempNamespace))
4376 6132 : return;
4377 :
4378 : /*
4379 : * The temporary tablespace does not exist yet and is wanted, so
4380 : * initialize it.
4381 : */
4382 598 : InitTempTableNamespace();
4383 : }
4384 :
4385 : /*
4386 : * InitTempTableNamespace
4387 : * Initialize temp table namespace on first use in a particular backend
4388 : */
4389 : static void
4390 598 : InitTempTableNamespace(void)
4391 : {
4392 : char namespaceName[NAMEDATALEN];
4393 : Oid namespaceId;
4394 : Oid toastspaceId;
4395 :
4396 : Assert(!OidIsValid(myTempNamespace));
4397 :
4398 : /*
4399 : * First, do permission check to see if we are authorized to make temp
4400 : * tables. We use a nonstandard error message here since "databasename:
4401 : * permission denied" might be a tad cryptic.
4402 : *
4403 : * Note that ACL_CREATE_TEMP rights are rechecked in pg_namespace_aclmask;
4404 : * that's necessary since current user ID could change during the session.
4405 : * But there's no need to make the namespace in the first place until a
4406 : * temp table creation request is made by someone with appropriate rights.
4407 : */
4408 598 : if (object_aclcheck(DatabaseRelationId, MyDatabaseId, GetUserId(),
4409 : ACL_CREATE_TEMP) != ACLCHECK_OK)
4410 0 : ereport(ERROR,
4411 : (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
4412 : errmsg("permission denied to create temporary tables in database \"%s\"",
4413 : get_database_name(MyDatabaseId))));
4414 :
4415 : /*
4416 : * Do not allow a Hot Standby session to make temp tables. Aside from
4417 : * problems with modifying the system catalogs, there is a naming
4418 : * conflict: pg_temp_N belongs to the session with proc number N on the
4419 : * primary, not to a hot standby session with the same proc number. We
4420 : * should not be able to get here anyway due to XactReadOnly checks, but
4421 : * let's just make real sure. Note that this also backstops various
4422 : * operations that allow XactReadOnly transactions to modify temp tables;
4423 : * they'd need RecoveryInProgress checks if not for this.
4424 : */
4425 598 : if (RecoveryInProgress())
4426 0 : ereport(ERROR,
4427 : (errcode(ERRCODE_READ_ONLY_SQL_TRANSACTION),
4428 : errmsg("cannot create temporary tables during recovery")));
4429 :
4430 : /* Parallel workers can't create temporary tables, either. */
4431 598 : if (IsParallelWorker())
4432 0 : ereport(ERROR,
4433 : (errcode(ERRCODE_READ_ONLY_SQL_TRANSACTION),
4434 : errmsg("cannot create temporary tables during a parallel operation")));
4435 :
4436 598 : snprintf(namespaceName, sizeof(namespaceName), "pg_temp_%d", MyProcNumber);
4437 :
4438 598 : namespaceId = get_namespace_oid(namespaceName, true);
4439 598 : if (!OidIsValid(namespaceId))
4440 : {
4441 : /*
4442 : * First use of this temp namespace in this database; create it. The
4443 : * temp namespaces are always owned by the superuser. We leave their
4444 : * permissions at default --- i.e., no access except to superuser ---
4445 : * to ensure that unprivileged users can't peek at other backends'
4446 : * temp tables. This works because the places that access the temp
4447 : * namespace for my own backend skip permissions checks on it.
4448 : */
4449 394 : namespaceId = NamespaceCreate(namespaceName, BOOTSTRAP_SUPERUSERID,
4450 : true);
4451 : /* Advance command counter to make namespace visible */
4452 394 : CommandCounterIncrement();
4453 : }
4454 : else
4455 : {
4456 : /*
4457 : * If the namespace already exists, clean it out (in case the former
4458 : * owner crashed without doing so).
4459 : */
4460 204 : RemoveTempRelations(namespaceId);
4461 : }
4462 :
4463 : /*
4464 : * If the corresponding toast-table namespace doesn't exist yet, create
4465 : * it. (We assume there is no need to clean it out if it does exist, since
4466 : * dropping a parent table should make its toast table go away.)
4467 : */
4468 598 : snprintf(namespaceName, sizeof(namespaceName), "pg_toast_temp_%d",
4469 : MyProcNumber);
4470 :
4471 598 : toastspaceId = get_namespace_oid(namespaceName, true);
4472 598 : if (!OidIsValid(toastspaceId))
4473 : {
4474 394 : toastspaceId = NamespaceCreate(namespaceName, BOOTSTRAP_SUPERUSERID,
4475 : true);
4476 : /* Advance command counter to make namespace visible */
4477 394 : CommandCounterIncrement();
4478 : }
4479 :
4480 : /*
4481 : * Okay, we've prepared the temp namespace ... but it's not committed yet,
4482 : * so all our work could be undone by transaction rollback. Set flag for
4483 : * AtEOXact_Namespace to know what to do.
4484 : */
4485 598 : myTempNamespace = namespaceId;
4486 598 : myTempToastNamespace = toastspaceId;
4487 :
4488 : /*
4489 : * Mark MyProc as owning this namespace which other processes can use to
4490 : * decide if a temporary namespace is in use or not. We assume that
4491 : * assignment of namespaceId is an atomic operation. Even if it is not,
4492 : * the temporary relation which resulted in the creation of this temporary
4493 : * namespace is still locked until the current transaction commits, and
4494 : * its pg_namespace row is not visible yet. However it does not matter:
4495 : * this flag makes the namespace as being in use, so no objects created on
4496 : * it would be removed concurrently.
4497 : */
4498 598 : MyProc->tempNamespaceId = namespaceId;
4499 :
4500 : /* It should not be done already. */
4501 : Assert(myTempNamespaceSubID == InvalidSubTransactionId);
4502 598 : myTempNamespaceSubID = GetCurrentSubTransactionId();
4503 :
4504 598 : baseSearchPathValid = false; /* need to rebuild list */
4505 598 : searchPathCacheValid = false;
4506 598 : }
4507 :
4508 : /*
4509 : * End-of-transaction cleanup for namespaces.
4510 : */
4511 : void
4512 790138 : AtEOXact_Namespace(bool isCommit, bool parallel)
4513 : {
4514 : /*
4515 : * If we abort the transaction in which a temp namespace was selected,
4516 : * we'll have to do any creation or cleanout work over again. So, just
4517 : * forget the namespace entirely until next time. On the other hand, if
4518 : * we commit then register an exit callback to clean out the temp tables
4519 : * at backend shutdown. (We only want to register the callback once per
4520 : * session, so this is a good place to do it.)
4521 : */
4522 790138 : if (myTempNamespaceSubID != InvalidSubTransactionId && !parallel)
4523 : {
4524 596 : if (isCommit)
4525 576 : before_shmem_exit(RemoveTempRelationsCallback, 0);
4526 : else
4527 : {
4528 20 : myTempNamespace = InvalidOid;
4529 20 : myTempToastNamespace = InvalidOid;
4530 20 : baseSearchPathValid = false; /* need to rebuild list */
4531 20 : searchPathCacheValid = false;
4532 :
4533 : /*
4534 : * Reset the temporary namespace flag in MyProc. We assume that
4535 : * this operation is atomic.
4536 : *
4537 : * Because this transaction is aborting, the pg_namespace row is
4538 : * not visible to anyone else anyway, but that doesn't matter:
4539 : * it's not a problem if objects contained in this namespace are
4540 : * removed concurrently.
4541 : */
4542 20 : MyProc->tempNamespaceId = InvalidOid;
4543 : }
4544 596 : myTempNamespaceSubID = InvalidSubTransactionId;
4545 : }
4546 :
4547 790138 : }
4548 :
4549 : /*
4550 : * AtEOSubXact_Namespace
4551 : *
4552 : * At subtransaction commit, propagate the temp-namespace-creation
4553 : * flag to the parent subtransaction.
4554 : *
4555 : * At subtransaction abort, forget the flag if we set it up.
4556 : */
4557 : void
4558 20004 : AtEOSubXact_Namespace(bool isCommit, SubTransactionId mySubid,
4559 : SubTransactionId parentSubid)
4560 : {
4561 :
4562 20004 : if (myTempNamespaceSubID == mySubid)
4563 : {
4564 8 : if (isCommit)
4565 6 : myTempNamespaceSubID = parentSubid;
4566 : else
4567 : {
4568 2 : myTempNamespaceSubID = InvalidSubTransactionId;
4569 : /* TEMP namespace creation failed, so reset state */
4570 2 : myTempNamespace = InvalidOid;
4571 2 : myTempToastNamespace = InvalidOid;
4572 2 : baseSearchPathValid = false; /* need to rebuild list */
4573 2 : searchPathCacheValid = false;
4574 :
4575 : /*
4576 : * Reset the temporary namespace flag in MyProc. We assume that
4577 : * this operation is atomic.
4578 : *
4579 : * Because this subtransaction is aborting, the pg_namespace row
4580 : * is not visible to anyone else anyway, but that doesn't matter:
4581 : * it's not a problem if objects contained in this namespace are
4582 : * removed concurrently.
4583 : */
4584 2 : MyProc->tempNamespaceId = InvalidOid;
4585 : }
4586 : }
4587 20004 : }
4588 :
4589 : /*
4590 : * Remove all relations in the specified temp namespace.
4591 : *
4592 : * This is called at backend shutdown (if we made any temp relations).
4593 : * It is also called when we begin using a pre-existing temp namespace,
4594 : * in order to clean out any relations that might have been created by
4595 : * a crashed backend.
4596 : */
4597 : static void
4598 794 : RemoveTempRelations(Oid tempNamespaceId)
4599 : {
4600 : ObjectAddress object;
4601 :
4602 : /*
4603 : * We want to get rid of everything in the target namespace, but not the
4604 : * namespace itself (deleting it only to recreate it later would be a
4605 : * waste of cycles). Hence, specify SKIP_ORIGINAL. It's also an INTERNAL
4606 : * deletion, and we want to not drop any extensions that might happen to
4607 : * own temp objects.
4608 : */
4609 794 : object.classId = NamespaceRelationId;
4610 794 : object.objectId = tempNamespaceId;
4611 794 : object.objectSubId = 0;
4612 :
4613 794 : performDeletion(&object, DROP_CASCADE,
4614 : PERFORM_DELETION_INTERNAL |
4615 : PERFORM_DELETION_QUIETLY |
4616 : PERFORM_DELETION_SKIP_ORIGINAL |
4617 : PERFORM_DELETION_SKIP_EXTENSIONS);
4618 794 : }
4619 :
4620 : /*
4621 : * Callback to remove temp relations at backend exit.
4622 : */
4623 : static void
4624 576 : RemoveTempRelationsCallback(int code, Datum arg)
4625 : {
4626 576 : if (OidIsValid(myTempNamespace)) /* should always be true */
4627 : {
4628 : /* Need to ensure we have a usable transaction. */
4629 576 : AbortOutOfAnyTransaction();
4630 576 : StartTransactionCommand();
4631 576 : PushActiveSnapshot(GetTransactionSnapshot());
4632 :
4633 576 : RemoveTempRelations(myTempNamespace);
4634 :
4635 576 : PopActiveSnapshot();
4636 576 : CommitTransactionCommand();
4637 : }
4638 576 : }
4639 :
4640 : /*
4641 : * Remove all temp tables from the temporary namespace.
4642 : */
4643 : void
4644 14 : ResetTempTableNamespace(void)
4645 : {
4646 14 : if (OidIsValid(myTempNamespace))
4647 14 : RemoveTempRelations(myTempNamespace);
4648 14 : }
4649 :
4650 :
4651 : /*
4652 : * Routines for handling the GUC variable 'search_path'.
4653 : */
4654 :
4655 : /* check_hook: validate new search_path value */
4656 : bool
4657 259932 : check_search_path(char **newval, void **extra, GucSource source)
4658 : {
4659 259932 : Oid roleid = InvalidOid;
4660 259932 : const char *searchPath = *newval;
4661 : char *rawname;
4662 : List *namelist;
4663 259932 : bool use_cache = (SearchPathCacheContext != NULL);
4664 :
4665 : /*
4666 : * We used to try to check that the named schemas exist, but there are
4667 : * many valid use-cases for having search_path settings that include
4668 : * schemas that don't exist; and often, we are not inside a transaction
4669 : * here and so can't consult the system catalogs anyway. So now, the only
4670 : * requirement is syntactic validity of the identifier list.
4671 : *
4672 : * Checking only the syntactic validity also allows us to use the search
4673 : * path cache (if available) to avoid calling SplitIdentifierString() on
4674 : * the same string repeatedly.
4675 : */
4676 259932 : if (use_cache)
4677 : {
4678 138752 : spcache_init();
4679 :
4680 138752 : roleid = GetUserId();
4681 :
4682 138752 : if (spcache_lookup(searchPath, roleid) != NULL)
4683 130798 : return true;
4684 : }
4685 :
4686 : /*
4687 : * Ensure validity check succeeds before creating cache entry.
4688 : */
4689 :
4690 129134 : rawname = pstrdup(searchPath); /* need a modifiable copy */
4691 :
4692 : /* Parse string into list of identifiers */
4693 129134 : if (!SplitIdentifierString(rawname, ',', &namelist))
4694 : {
4695 : /* syntax error in name list */
4696 0 : GUC_check_errdetail("List syntax is invalid.");
4697 0 : pfree(rawname);
4698 0 : list_free(namelist);
4699 0 : return false;
4700 : }
4701 129134 : pfree(rawname);
4702 129134 : list_free(namelist);
4703 :
4704 : /* OK to create empty cache entry */
4705 129134 : if (use_cache)
4706 7954 : (void) spcache_insert(searchPath, roleid);
4707 :
4708 129134 : return true;
4709 : }
4710 :
4711 : /* assign_hook: do extra actions as needed */
4712 : void
4713 509790 : assign_search_path(const char *newval, void *extra)
4714 : {
4715 : /* don't access search_path during bootstrap */
4716 : Assert(!IsBootstrapProcessingMode());
4717 :
4718 : /*
4719 : * We mark the path as needing recomputation, but don't do anything until
4720 : * it's needed. This avoids trying to do database access during GUC
4721 : * initialization, or outside a transaction.
4722 : *
4723 : * This does not invalidate the search path cache, so if this value had
4724 : * been previously set and no syscache invalidations happened,
4725 : * recomputation may not be necessary.
4726 : */
4727 509790 : baseSearchPathValid = false;
4728 509790 : }
4729 :
4730 : /*
4731 : * InitializeSearchPath: initialize module during InitPostgres.
4732 : *
4733 : * This is called after we are up enough to be able to do catalog lookups.
4734 : */
4735 : void
4736 30204 : InitializeSearchPath(void)
4737 : {
4738 30204 : if (IsBootstrapProcessingMode())
4739 : {
4740 : /*
4741 : * In bootstrap mode, the search path must be 'pg_catalog' so that
4742 : * tables are created in the proper namespace; ignore the GUC setting.
4743 : */
4744 : MemoryContext oldcxt;
4745 :
4746 90 : oldcxt = MemoryContextSwitchTo(TopMemoryContext);
4747 90 : baseSearchPath = list_make1_oid(PG_CATALOG_NAMESPACE);
4748 90 : MemoryContextSwitchTo(oldcxt);
4749 90 : baseCreationNamespace = PG_CATALOG_NAMESPACE;
4750 90 : baseTempCreationPending = false;
4751 90 : baseSearchPathValid = true;
4752 90 : namespaceUser = GetUserId();
4753 90 : activeSearchPath = baseSearchPath;
4754 90 : activeCreationNamespace = baseCreationNamespace;
4755 90 : activeTempCreationPending = baseTempCreationPending;
4756 90 : activePathGeneration++; /* pro forma */
4757 : }
4758 : else
4759 : {
4760 : /*
4761 : * In normal mode, arrange for a callback on any syscache invalidation
4762 : * that will affect the search_path cache.
4763 : */
4764 :
4765 : /* namespace name or ACLs may have changed */
4766 30114 : CacheRegisterSyscacheCallback(NAMESPACEOID,
4767 : InvalidationCallback,
4768 : (Datum) 0);
4769 :
4770 : /* role name may affect the meaning of "$user" */
4771 30114 : CacheRegisterSyscacheCallback(AUTHOID,
4772 : InvalidationCallback,
4773 : (Datum) 0);
4774 :
4775 : /* role membership may affect ACLs */
4776 30114 : CacheRegisterSyscacheCallback(AUTHMEMROLEMEM,
4777 : InvalidationCallback,
4778 : (Datum) 0);
4779 :
4780 : /* database owner may affect ACLs */
4781 30114 : CacheRegisterSyscacheCallback(DATABASEOID,
4782 : InvalidationCallback,
4783 : (Datum) 0);
4784 :
4785 : /* Force search path to be recomputed on next use */
4786 30114 : baseSearchPathValid = false;
4787 30114 : searchPathCacheValid = false;
4788 : }
4789 30204 : }
4790 :
4791 : /*
4792 : * InvalidationCallback
4793 : * Syscache inval callback function
4794 : */
4795 : static void
4796 55488 : InvalidationCallback(Datum arg, int cacheid, uint32 hashvalue)
4797 : {
4798 : /*
4799 : * Force search path to be recomputed on next use, also invalidating the
4800 : * search path cache (because namespace names, ACLs, or role names may
4801 : * have changed).
4802 : */
4803 55488 : baseSearchPathValid = false;
4804 55488 : searchPathCacheValid = false;
4805 55488 : }
4806 :
4807 : /*
4808 : * Fetch the active search path. The return value is a palloc'ed list
4809 : * of OIDs; the caller is responsible for freeing this storage as
4810 : * appropriate.
4811 : *
4812 : * The returned list includes the implicitly-prepended namespaces only if
4813 : * includeImplicit is true.
4814 : *
4815 : * Note: calling this may result in a CommandCounterIncrement operation,
4816 : * if we have to create or clean out the temp namespace.
4817 : */
4818 : List *
4819 886 : fetch_search_path(bool includeImplicit)
4820 : {
4821 : List *result;
4822 :
4823 886 : recomputeNamespacePath();
4824 :
4825 : /*
4826 : * If the temp namespace should be first, force it to exist. This is so
4827 : * that callers can trust the result to reflect the actual default
4828 : * creation namespace. It's a bit bogus to do this here, since
4829 : * current_schema() is supposedly a stable function without side-effects,
4830 : * but the alternatives seem worse.
4831 : */
4832 886 : if (activeTempCreationPending)
4833 : {
4834 6 : AccessTempTableNamespace(true);
4835 6 : recomputeNamespacePath();
4836 : }
4837 :
4838 886 : result = list_copy(activeSearchPath);
4839 886 : if (!includeImplicit)
4840 : {
4841 1472 : while (result && linitial_oid(result) != activeCreationNamespace)
4842 772 : result = list_delete_first(result);
4843 : }
4844 :
4845 886 : return result;
4846 : }
4847 :
4848 : /*
4849 : * Fetch the active search path into a caller-allocated array of OIDs.
4850 : * Returns the number of path entries. (If this is more than sarray_len,
4851 : * then the data didn't fit and is not all stored.)
4852 : *
4853 : * The returned list always includes the implicitly-prepended namespaces,
4854 : * but never includes the temp namespace. (This is suitable for existing
4855 : * users, which would want to ignore the temp namespace anyway.) This
4856 : * definition allows us to not worry about initializing the temp namespace.
4857 : */
4858 : int
4859 669182 : fetch_search_path_array(Oid *sarray, int sarray_len)
4860 : {
4861 669182 : int count = 0;
4862 : ListCell *l;
4863 :
4864 669182 : recomputeNamespacePath();
4865 :
4866 1932478 : foreach(l, activeSearchPath)
4867 : {
4868 1263296 : Oid namespaceId = lfirst_oid(l);
4869 :
4870 1263296 : if (namespaceId == myTempNamespace)
4871 154120 : continue; /* do not include temp namespace */
4872 :
4873 1109176 : if (count < sarray_len)
4874 1109176 : sarray[count] = namespaceId;
4875 1109176 : count++;
4876 : }
4877 :
4878 669182 : return count;
4879 : }
4880 :
4881 :
4882 : /*
4883 : * Export the FooIsVisible functions as SQL-callable functions.
4884 : *
4885 : * Note: as of Postgres 8.4, these will silently return NULL if called on
4886 : * a nonexistent object OID, rather than failing. This is to avoid race
4887 : * condition errors when a query that's scanning a catalog using an MVCC
4888 : * snapshot uses one of these functions. The underlying IsVisible functions
4889 : * always use an up-to-date snapshot and so might see the object as already
4890 : * gone when it's still visible to the transaction snapshot.
4891 : */
4892 :
4893 : Datum
4894 21158 : pg_table_is_visible(PG_FUNCTION_ARGS)
4895 : {
4896 21158 : Oid oid = PG_GETARG_OID(0);
4897 : bool result;
4898 21158 : bool is_missing = false;
4899 :
4900 21158 : result = RelationIsVisibleExt(oid, &is_missing);
4901 :
4902 21158 : if (is_missing)
4903 6 : PG_RETURN_NULL();
4904 21152 : PG_RETURN_BOOL(result);
4905 : }
4906 :
4907 : Datum
4908 3942 : pg_type_is_visible(PG_FUNCTION_ARGS)
4909 : {
4910 3942 : Oid oid = PG_GETARG_OID(0);
4911 : bool result;
4912 3942 : bool is_missing = false;
4913 :
4914 3942 : result = TypeIsVisibleExt(oid, &is_missing);
4915 :
4916 3942 : if (is_missing)
4917 0 : PG_RETURN_NULL();
4918 3942 : PG_RETURN_BOOL(result);
4919 : }
4920 :
4921 : Datum
4922 7502 : pg_function_is_visible(PG_FUNCTION_ARGS)
4923 : {
4924 7502 : Oid oid = PG_GETARG_OID(0);
4925 : bool result;
4926 7502 : bool is_missing = false;
4927 :
4928 7502 : result = FunctionIsVisibleExt(oid, &is_missing);
4929 :
4930 7502 : if (is_missing)
4931 0 : PG_RETURN_NULL();
4932 7502 : PG_RETURN_BOOL(result);
4933 : }
4934 :
4935 : Datum
4936 1700 : pg_operator_is_visible(PG_FUNCTION_ARGS)
4937 : {
4938 1700 : Oid oid = PG_GETARG_OID(0);
4939 : bool result;
4940 1700 : bool is_missing = false;
4941 :
4942 1700 : result = OperatorIsVisibleExt(oid, &is_missing);
4943 :
4944 1700 : if (is_missing)
4945 0 : PG_RETURN_NULL();
4946 1700 : PG_RETURN_BOOL(result);
4947 : }
4948 :
4949 : Datum
4950 18 : pg_opclass_is_visible(PG_FUNCTION_ARGS)
4951 : {
4952 18 : Oid oid = PG_GETARG_OID(0);
4953 : bool result;
4954 18 : bool is_missing = false;
4955 :
4956 18 : result = OpclassIsVisibleExt(oid, &is_missing);
4957 :
4958 18 : if (is_missing)
4959 0 : PG_RETURN_NULL();
4960 18 : PG_RETURN_BOOL(result);
4961 : }
4962 :
4963 : Datum
4964 312 : pg_opfamily_is_visible(PG_FUNCTION_ARGS)
4965 : {
4966 312 : Oid oid = PG_GETARG_OID(0);
4967 : bool result;
4968 312 : bool is_missing = false;
4969 :
4970 312 : result = OpfamilyIsVisibleExt(oid, &is_missing);
4971 :
4972 312 : if (is_missing)
4973 0 : PG_RETURN_NULL();
4974 312 : PG_RETURN_BOOL(result);
4975 : }
4976 :
4977 : Datum
4978 0 : pg_collation_is_visible(PG_FUNCTION_ARGS)
4979 : {
4980 0 : Oid oid = PG_GETARG_OID(0);
4981 : bool result;
4982 0 : bool is_missing = false;
4983 :
4984 0 : result = CollationIsVisibleExt(oid, &is_missing);
4985 :
4986 0 : if (is_missing)
4987 0 : PG_RETURN_NULL();
4988 0 : PG_RETURN_BOOL(result);
4989 : }
4990 :
4991 : Datum
4992 0 : pg_conversion_is_visible(PG_FUNCTION_ARGS)
4993 : {
4994 0 : Oid oid = PG_GETARG_OID(0);
4995 : bool result;
4996 0 : bool is_missing = false;
4997 :
4998 0 : result = ConversionIsVisibleExt(oid, &is_missing);
4999 :
5000 0 : if (is_missing)
5001 0 : PG_RETURN_NULL();
5002 0 : PG_RETURN_BOOL(result);
5003 : }
5004 :
5005 : Datum
5006 366 : pg_statistics_obj_is_visible(PG_FUNCTION_ARGS)
5007 : {
5008 366 : Oid oid = PG_GETARG_OID(0);
5009 : bool result;
5010 366 : bool is_missing = false;
5011 :
5012 366 : result = StatisticsObjIsVisibleExt(oid, &is_missing);
5013 :
5014 366 : if (is_missing)
5015 0 : PG_RETURN_NULL();
5016 366 : PG_RETURN_BOOL(result);
5017 : }
5018 :
5019 : Datum
5020 0 : pg_ts_parser_is_visible(PG_FUNCTION_ARGS)
5021 : {
5022 0 : Oid oid = PG_GETARG_OID(0);
5023 : bool result;
5024 0 : bool is_missing = false;
5025 :
5026 0 : result = TSParserIsVisibleExt(oid, &is_missing);
5027 :
5028 0 : if (is_missing)
5029 0 : PG_RETURN_NULL();
5030 0 : PG_RETURN_BOOL(result);
5031 : }
5032 :
5033 : Datum
5034 0 : pg_ts_dict_is_visible(PG_FUNCTION_ARGS)
5035 : {
5036 0 : Oid oid = PG_GETARG_OID(0);
5037 : bool result;
5038 0 : bool is_missing = false;
5039 :
5040 0 : result = TSDictionaryIsVisibleExt(oid, &is_missing);
5041 :
5042 0 : if (is_missing)
5043 0 : PG_RETURN_NULL();
5044 0 : PG_RETURN_BOOL(result);
5045 : }
5046 :
5047 : Datum
5048 0 : pg_ts_template_is_visible(PG_FUNCTION_ARGS)
5049 : {
5050 0 : Oid oid = PG_GETARG_OID(0);
5051 : bool result;
5052 0 : bool is_missing = false;
5053 :
5054 0 : result = TSTemplateIsVisibleExt(oid, &is_missing);
5055 :
5056 0 : if (is_missing)
5057 0 : PG_RETURN_NULL();
5058 0 : PG_RETURN_BOOL(result);
5059 : }
5060 :
5061 : Datum
5062 0 : pg_ts_config_is_visible(PG_FUNCTION_ARGS)
5063 : {
5064 0 : Oid oid = PG_GETARG_OID(0);
5065 : bool result;
5066 0 : bool is_missing = false;
5067 :
5068 0 : result = TSConfigIsVisibleExt(oid, &is_missing);
5069 :
5070 0 : if (is_missing)
5071 0 : PG_RETURN_NULL();
5072 0 : PG_RETURN_BOOL(result);
5073 : }
5074 :
5075 : Datum
5076 2168 : pg_my_temp_schema(PG_FUNCTION_ARGS)
5077 : {
5078 2168 : PG_RETURN_OID(myTempNamespace);
5079 : }
5080 :
5081 : Datum
5082 14192 : pg_is_other_temp_schema(PG_FUNCTION_ARGS)
5083 : {
5084 14192 : Oid oid = PG_GETARG_OID(0);
5085 :
5086 14192 : PG_RETURN_BOOL(isOtherTempNamespace(oid));
5087 : }
|