2008-12-19 17:25:19 +01:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
*
|
|
|
|
* foreign.c
|
2009-06-11 16:49:15 +02:00
|
|
|
* support for foreign-data wrappers, servers and user mappings.
|
2008-12-19 17:25:19 +01:00
|
|
|
*
|
2013-01-01 23:15:01 +01:00
|
|
|
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
|
2008-12-19 17:25:19 +01:00
|
|
|
*
|
|
|
|
* IDENTIFICATION
|
2010-09-20 22:08:53 +02:00
|
|
|
* src/backend/foreign/foreign.c
|
2008-12-19 17:25:19 +01:00
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
#include "postgres.h"
|
|
|
|
|
2012-08-30 22:15:44 +02:00
|
|
|
#include "access/htup_details.h"
|
2008-12-19 17:25:19 +01:00
|
|
|
#include "access/reloptions.h"
|
|
|
|
#include "catalog/pg_foreign_data_wrapper.h"
|
|
|
|
#include "catalog/pg_foreign_server.h"
|
2011-02-20 06:17:18 +01:00
|
|
|
#include "catalog/pg_foreign_table.h"
|
2008-12-19 17:25:19 +01:00
|
|
|
#include "catalog/pg_user_mapping.h"
|
2011-02-20 06:17:18 +01:00
|
|
|
#include "foreign/fdwapi.h"
|
2008-12-19 17:25:19 +01:00
|
|
|
#include "foreign/foreign.h"
|
2012-08-29 01:02:00 +02:00
|
|
|
#include "lib/stringinfo.h"
|
2008-12-19 17:25:19 +01:00
|
|
|
#include "miscadmin.h"
|
|
|
|
#include "utils/builtins.h"
|
2013-03-07 05:47:38 +01:00
|
|
|
#include "utils/memutils.h"
|
|
|
|
#include "utils/rel.h"
|
2008-12-19 17:25:19 +01:00
|
|
|
#include "utils/syscache.h"
|
|
|
|
|
|
|
|
|
|
|
|
extern Datum pg_options_to_table(PG_FUNCTION_ARGS);
|
2009-02-24 11:06:36 +01:00
|
|
|
extern Datum postgresql_fdw_validator(PG_FUNCTION_ARGS);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
2009-06-11 16:49:15 +02:00
|
|
|
* GetForeignDataWrapper - look up the foreign-data wrapper by OID.
|
2008-12-19 17:25:19 +01:00
|
|
|
*/
|
|
|
|
ForeignDataWrapper *
|
|
|
|
GetForeignDataWrapper(Oid fdwid)
|
|
|
|
{
|
2009-06-11 16:49:15 +02:00
|
|
|
Form_pg_foreign_data_wrapper fdwform;
|
|
|
|
ForeignDataWrapper *fdw;
|
|
|
|
Datum datum;
|
|
|
|
HeapTuple tp;
|
|
|
|
bool isnull;
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2010-02-14 19:42:19 +01:00
|
|
|
tp = SearchSysCache1(FOREIGNDATAWRAPPEROID, ObjectIdGetDatum(fdwid));
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for foreign-data wrapper %u", fdwid);
|
|
|
|
|
|
|
|
fdwform = (Form_pg_foreign_data_wrapper) GETSTRUCT(tp);
|
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
fdw = (ForeignDataWrapper *) palloc(sizeof(ForeignDataWrapper));
|
2008-12-19 17:25:19 +01:00
|
|
|
fdw->fdwid = fdwid;
|
|
|
|
fdw->owner = fdwform->fdwowner;
|
|
|
|
fdw->fdwname = pstrdup(NameStr(fdwform->fdwname));
|
2011-02-19 06:06:18 +01:00
|
|
|
fdw->fdwhandler = fdwform->fdwhandler;
|
2009-02-24 11:06:36 +01:00
|
|
|
fdw->fdwvalidator = fdwform->fdwvalidator;
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
/* Extract the fdwoptions */
|
2008-12-19 17:25:19 +01:00
|
|
|
datum = SysCacheGetAttr(FOREIGNDATAWRAPPEROID,
|
|
|
|
tp,
|
|
|
|
Anum_pg_foreign_data_wrapper_fdwoptions,
|
|
|
|
&isnull);
|
2011-02-20 06:17:18 +01:00
|
|
|
if (isnull)
|
|
|
|
fdw->options = NIL;
|
|
|
|
else
|
|
|
|
fdw->options = untransformRelOptions(datum);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
return fdw;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GetForeignDataWrapperByName - look up the foreign-data wrapper
|
|
|
|
* definition by name.
|
|
|
|
*/
|
|
|
|
ForeignDataWrapper *
|
|
|
|
GetForeignDataWrapperByName(const char *fdwname, bool missing_ok)
|
|
|
|
{
|
2011-04-01 17:28:28 +02:00
|
|
|
Oid fdwId = get_foreign_data_wrapper_oid(fdwname, missing_ok);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
if (!OidIsValid(fdwId))
|
2008-12-19 17:25:19 +01:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return GetForeignDataWrapper(fdwId);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GetForeignServer - look up the foreign server definition.
|
|
|
|
*/
|
|
|
|
ForeignServer *
|
|
|
|
GetForeignServer(Oid serverid)
|
|
|
|
{
|
2009-06-11 16:49:15 +02:00
|
|
|
Form_pg_foreign_server serverform;
|
|
|
|
ForeignServer *server;
|
|
|
|
HeapTuple tp;
|
|
|
|
Datum datum;
|
|
|
|
bool isnull;
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2010-02-14 19:42:19 +01:00
|
|
|
tp = SearchSysCache1(FOREIGNSERVEROID, ObjectIdGetDatum(serverid));
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for foreign server %u", serverid);
|
|
|
|
|
|
|
|
serverform = (Form_pg_foreign_server) GETSTRUCT(tp);
|
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
server = (ForeignServer *) palloc(sizeof(ForeignServer));
|
2008-12-19 17:25:19 +01:00
|
|
|
server->serverid = serverid;
|
|
|
|
server->servername = pstrdup(NameStr(serverform->srvname));
|
|
|
|
server->owner = serverform->srvowner;
|
|
|
|
server->fdwid = serverform->srvfdw;
|
|
|
|
|
|
|
|
/* Extract server type */
|
|
|
|
datum = SysCacheGetAttr(FOREIGNSERVEROID,
|
|
|
|
tp,
|
|
|
|
Anum_pg_foreign_server_srvtype,
|
|
|
|
&isnull);
|
|
|
|
server->servertype = isnull ? NULL : pstrdup(TextDatumGetCString(datum));
|
|
|
|
|
|
|
|
/* Extract server version */
|
|
|
|
datum = SysCacheGetAttr(FOREIGNSERVEROID,
|
|
|
|
tp,
|
|
|
|
Anum_pg_foreign_server_srvversion,
|
|
|
|
&isnull);
|
|
|
|
server->serverversion = isnull ? NULL : pstrdup(TextDatumGetCString(datum));
|
|
|
|
|
|
|
|
/* Extract the srvoptions */
|
|
|
|
datum = SysCacheGetAttr(FOREIGNSERVEROID,
|
|
|
|
tp,
|
|
|
|
Anum_pg_foreign_server_srvoptions,
|
|
|
|
&isnull);
|
2011-02-20 06:17:18 +01:00
|
|
|
if (isnull)
|
|
|
|
server->options = NIL;
|
|
|
|
else
|
|
|
|
server->options = untransformRelOptions(datum);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
return server;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GetForeignServerByName - look up the foreign server definition by name.
|
|
|
|
*/
|
|
|
|
ForeignServer *
|
|
|
|
GetForeignServerByName(const char *srvname, bool missing_ok)
|
|
|
|
{
|
2011-04-01 17:28:28 +02:00
|
|
|
Oid serverid = get_foreign_server_oid(srvname, missing_ok);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
if (!OidIsValid(serverid))
|
2008-12-19 17:25:19 +01:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return GetForeignServer(serverid);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GetUserMapping - look up the user mapping.
|
|
|
|
*
|
|
|
|
* If no mapping is found for the supplied user, we also look for
|
|
|
|
* PUBLIC mappings (userid == InvalidOid).
|
|
|
|
*/
|
|
|
|
UserMapping *
|
|
|
|
GetUserMapping(Oid userid, Oid serverid)
|
|
|
|
{
|
|
|
|
Datum datum;
|
2009-06-11 16:49:15 +02:00
|
|
|
HeapTuple tp;
|
2008-12-19 17:25:19 +01:00
|
|
|
bool isnull;
|
2009-06-11 16:49:15 +02:00
|
|
|
UserMapping *um;
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2010-02-14 19:42:19 +01:00
|
|
|
tp = SearchSysCache2(USERMAPPINGUSERSERVER,
|
|
|
|
ObjectIdGetDatum(userid),
|
|
|
|
ObjectIdGetDatum(serverid));
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
{
|
|
|
|
/* Not found for the specific user -- try PUBLIC */
|
2010-02-14 19:42:19 +01:00
|
|
|
tp = SearchSysCache2(USERMAPPINGUSERSERVER,
|
|
|
|
ObjectIdGetDatum(InvalidOid),
|
|
|
|
ObjectIdGetDatum(serverid));
|
2008-12-19 17:25:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
2009-06-11 16:49:15 +02:00
|
|
|
errmsg("user mapping not found for \"%s\"",
|
|
|
|
MappingUserName(userid))));
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
um = (UserMapping *) palloc(sizeof(UserMapping));
|
|
|
|
um->userid = userid;
|
|
|
|
um->serverid = serverid;
|
|
|
|
|
2008-12-19 17:25:19 +01:00
|
|
|
/* Extract the umoptions */
|
|
|
|
datum = SysCacheGetAttr(USERMAPPINGUSERSERVER,
|
|
|
|
tp,
|
|
|
|
Anum_pg_user_mapping_umoptions,
|
|
|
|
&isnull);
|
2011-02-20 06:17:18 +01:00
|
|
|
if (isnull)
|
|
|
|
um->options = NIL;
|
|
|
|
else
|
|
|
|
um->options = untransformRelOptions(datum);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
return um;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
/*
|
|
|
|
* GetForeignTable - look up the foreign table definition by relation oid.
|
|
|
|
*/
|
|
|
|
ForeignTable *
|
|
|
|
GetForeignTable(Oid relid)
|
|
|
|
{
|
|
|
|
Form_pg_foreign_table tableform;
|
|
|
|
ForeignTable *ft;
|
|
|
|
HeapTuple tp;
|
|
|
|
Datum datum;
|
|
|
|
bool isnull;
|
|
|
|
|
|
|
|
tp = SearchSysCache1(FOREIGNTABLEREL, ObjectIdGetDatum(relid));
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for foreign table %u", relid);
|
|
|
|
tableform = (Form_pg_foreign_table) GETSTRUCT(tp);
|
|
|
|
|
|
|
|
ft = (ForeignTable *) palloc(sizeof(ForeignTable));
|
|
|
|
ft->relid = relid;
|
|
|
|
ft->serverid = tableform->ftserver;
|
|
|
|
|
|
|
|
/* Extract the ftoptions */
|
|
|
|
datum = SysCacheGetAttr(FOREIGNTABLEREL,
|
|
|
|
tp,
|
|
|
|
Anum_pg_foreign_table_ftoptions,
|
|
|
|
&isnull);
|
|
|
|
if (isnull)
|
|
|
|
ft->options = NIL;
|
|
|
|
else
|
|
|
|
ft->options = untransformRelOptions(datum);
|
|
|
|
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
return ft;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-08 00:20:58 +01:00
|
|
|
/*
|
|
|
|
* GetForeignColumnOptions - Get attfdwoptions of given relation/attnum
|
|
|
|
* as list of DefElem.
|
|
|
|
*/
|
|
|
|
List *
|
|
|
|
GetForeignColumnOptions(Oid relid, AttrNumber attnum)
|
|
|
|
{
|
|
|
|
List *options;
|
|
|
|
HeapTuple tp;
|
|
|
|
Datum datum;
|
|
|
|
bool isnull;
|
|
|
|
|
|
|
|
tp = SearchSysCache2(ATTNUM,
|
|
|
|
ObjectIdGetDatum(relid),
|
|
|
|
Int16GetDatum(attnum));
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for attribute %d of relation %u",
|
|
|
|
attnum, relid);
|
|
|
|
datum = SysCacheGetAttr(ATTNUM,
|
|
|
|
tp,
|
|
|
|
Anum_pg_attribute_attfdwoptions,
|
|
|
|
&isnull);
|
|
|
|
if (isnull)
|
|
|
|
options = NIL;
|
|
|
|
else
|
|
|
|
options = untransformRelOptions(datum);
|
|
|
|
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
/*
|
|
|
|
* GetFdwRoutine - call the specified foreign-data wrapper handler routine
|
|
|
|
* to get its FdwRoutine struct.
|
|
|
|
*/
|
|
|
|
FdwRoutine *
|
|
|
|
GetFdwRoutine(Oid fdwhandler)
|
|
|
|
{
|
|
|
|
Datum datum;
|
|
|
|
FdwRoutine *routine;
|
|
|
|
|
|
|
|
datum = OidFunctionCall0(fdwhandler);
|
|
|
|
routine = (FdwRoutine *) DatumGetPointer(datum);
|
|
|
|
|
|
|
|
if (routine == NULL || !IsA(routine, FdwRoutine))
|
|
|
|
elog(ERROR, "foreign-data wrapper handler function %u did not return an FdwRoutine struct",
|
|
|
|
fdwhandler);
|
|
|
|
|
|
|
|
return routine;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GetFdwRoutineByRelId - look up the handler of the foreign-data wrapper
|
|
|
|
* for the given foreign table, and retrieve its FdwRoutine struct.
|
|
|
|
*/
|
|
|
|
FdwRoutine *
|
|
|
|
GetFdwRoutineByRelId(Oid relid)
|
|
|
|
{
|
|
|
|
HeapTuple tp;
|
|
|
|
Form_pg_foreign_data_wrapper fdwform;
|
|
|
|
Form_pg_foreign_server serverform;
|
|
|
|
Form_pg_foreign_table tableform;
|
|
|
|
Oid serverid;
|
|
|
|
Oid fdwid;
|
|
|
|
Oid fdwhandler;
|
|
|
|
|
|
|
|
/* Get server OID for the foreign table. */
|
|
|
|
tp = SearchSysCache1(FOREIGNTABLEREL, ObjectIdGetDatum(relid));
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for foreign table %u", relid);
|
|
|
|
tableform = (Form_pg_foreign_table) GETSTRUCT(tp);
|
|
|
|
serverid = tableform->ftserver;
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
/* Get foreign-data wrapper OID for the server. */
|
|
|
|
tp = SearchSysCache1(FOREIGNSERVEROID, ObjectIdGetDatum(serverid));
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for foreign server %u", serverid);
|
|
|
|
serverform = (Form_pg_foreign_server) GETSTRUCT(tp);
|
|
|
|
fdwid = serverform->srvfdw;
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
/* Get handler function OID for the FDW. */
|
|
|
|
tp = SearchSysCache1(FOREIGNDATAWRAPPEROID, ObjectIdGetDatum(fdwid));
|
|
|
|
if (!HeapTupleIsValid(tp))
|
|
|
|
elog(ERROR, "cache lookup failed for foreign-data wrapper %u", fdwid);
|
|
|
|
fdwform = (Form_pg_foreign_data_wrapper) GETSTRUCT(tp);
|
|
|
|
fdwhandler = fdwform->fdwhandler;
|
|
|
|
|
|
|
|
/* Complain if FDW has been set to NO HANDLER. */
|
|
|
|
if (!OidIsValid(fdwhandler))
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
|
|
|
|
errmsg("foreign-data wrapper \"%s\" has no handler",
|
|
|
|
NameStr(fdwform->fdwname))));
|
|
|
|
|
|
|
|
ReleaseSysCache(tp);
|
|
|
|
|
|
|
|
/* And finally, call the handler function. */
|
|
|
|
return GetFdwRoutine(fdwhandler);
|
|
|
|
}
|
|
|
|
|
2013-03-07 05:47:38 +01:00
|
|
|
/*
|
|
|
|
* GetFdwRoutineForRelation - look up the handler of the foreign-data wrapper
|
|
|
|
* for the given foreign table, and retrieve its FdwRoutine struct.
|
|
|
|
*
|
|
|
|
* This function is preferred over GetFdwRoutineByRelId because it caches
|
|
|
|
* the data in the relcache entry, saving a number of catalog lookups.
|
|
|
|
*
|
|
|
|
* If makecopy is true then the returned data is freshly palloc'd in the
|
|
|
|
* caller's memory context. Otherwise, it's a pointer to the relcache data,
|
|
|
|
* which will be lost in any relcache reset --- so don't rely on it long.
|
|
|
|
*/
|
|
|
|
FdwRoutine *
|
|
|
|
GetFdwRoutineForRelation(Relation relation, bool makecopy)
|
|
|
|
{
|
|
|
|
FdwRoutine *fdwroutine;
|
|
|
|
FdwRoutine *cfdwroutine;
|
|
|
|
|
|
|
|
if (relation->rd_fdwroutine == NULL)
|
|
|
|
{
|
|
|
|
/* Get the info by consulting the catalogs and the FDW code */
|
|
|
|
fdwroutine = GetFdwRoutineByRelId(RelationGetRelid(relation));
|
|
|
|
|
|
|
|
/* Save the data for later reuse in CacheMemoryContext */
|
|
|
|
cfdwroutine = (FdwRoutine *) MemoryContextAlloc(CacheMemoryContext,
|
|
|
|
sizeof(FdwRoutine));
|
|
|
|
memcpy(cfdwroutine, fdwroutine, sizeof(FdwRoutine));
|
|
|
|
relation->rd_fdwroutine = cfdwroutine;
|
|
|
|
|
|
|
|
/* Give back the locally palloc'd copy regardless of makecopy */
|
|
|
|
return fdwroutine;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have valid cached data --- does the caller want a copy? */
|
|
|
|
if (makecopy)
|
|
|
|
{
|
|
|
|
fdwroutine = (FdwRoutine *) palloc(sizeof(FdwRoutine));
|
|
|
|
memcpy(fdwroutine, relation->rd_fdwroutine, sizeof(FdwRoutine));
|
|
|
|
return fdwroutine;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only a short-lived reference is needed, so just hand back cached copy */
|
|
|
|
return relation->rd_fdwroutine;
|
|
|
|
}
|
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
|
2008-12-19 17:25:19 +01:00
|
|
|
/*
|
|
|
|
* deflist_to_tuplestore - Helper function to convert DefElem list to
|
|
|
|
* tuplestore usable in SRF.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
deflist_to_tuplestore(ReturnSetInfo *rsinfo, List *options)
|
|
|
|
{
|
|
|
|
ListCell *cell;
|
|
|
|
TupleDesc tupdesc;
|
|
|
|
Tuplestorestate *tupstore;
|
|
|
|
Datum values[2];
|
2011-02-20 06:17:18 +01:00
|
|
|
bool nulls[2];
|
2008-12-19 17:25:19 +01:00
|
|
|
MemoryContext per_query_ctx;
|
|
|
|
MemoryContext oldcontext;
|
|
|
|
|
|
|
|
/* check to see if caller supports us returning a tuplestore */
|
|
|
|
if (rsinfo == NULL || !IsA(rsinfo, ReturnSetInfo))
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
|
|
|
|
errmsg("set-valued function called in context that cannot accept a set")));
|
2009-06-11 18:14:18 +02:00
|
|
|
if (!(rsinfo->allowedModes & SFRM_Materialize) ||
|
|
|
|
rsinfo->expectedDesc == NULL)
|
2008-12-19 17:25:19 +01:00
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
|
|
|
|
errmsg("materialize mode required, but it is not allowed in this context")));
|
|
|
|
|
|
|
|
per_query_ctx = rsinfo->econtext->ecxt_per_query_memory;
|
|
|
|
oldcontext = MemoryContextSwitchTo(per_query_ctx);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now prepare the result set.
|
|
|
|
*/
|
|
|
|
tupdesc = CreateTupleDescCopy(rsinfo->expectedDesc);
|
|
|
|
tupstore = tuplestore_begin_heap(true, false, work_mem);
|
|
|
|
rsinfo->returnMode = SFRM_Materialize;
|
|
|
|
rsinfo->setResult = tupstore;
|
|
|
|
rsinfo->setDesc = tupdesc;
|
|
|
|
|
2009-06-11 16:49:15 +02:00
|
|
|
foreach(cell, options)
|
2008-12-19 17:25:19 +01:00
|
|
|
{
|
2009-06-11 16:49:15 +02:00
|
|
|
DefElem *def = lfirst(cell);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
values[0] = CStringGetTextDatum(def->defname);
|
2011-09-13 17:36:49 +02:00
|
|
|
nulls[0] = false;
|
|
|
|
if (def->arg)
|
|
|
|
{
|
|
|
|
values[1] = CStringGetTextDatum(((Value *) (def->arg))->val.str);
|
|
|
|
nulls[1] = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
values[1] = (Datum) 0;
|
|
|
|
nulls[1] = true;
|
|
|
|
}
|
2008-12-19 17:25:19 +01:00
|
|
|
tuplestore_putvalues(tupstore, tupdesc, values, nulls);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* clean up and return the tuplestore */
|
|
|
|
tuplestore_donestoring(tupstore);
|
|
|
|
|
|
|
|
MemoryContextSwitchTo(oldcontext);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert options array to name/value table. Useful for information
|
|
|
|
* schema and pg_dump.
|
|
|
|
*/
|
|
|
|
Datum
|
|
|
|
pg_options_to_table(PG_FUNCTION_ARGS)
|
|
|
|
{
|
2009-06-11 16:49:15 +02:00
|
|
|
Datum array = PG_GETARG_DATUM(0);
|
2008-12-19 17:25:19 +01:00
|
|
|
|
2011-02-20 06:17:18 +01:00
|
|
|
deflist_to_tuplestore((ReturnSetInfo *) fcinfo->resultinfo,
|
|
|
|
untransformRelOptions(array));
|
2008-12-19 17:25:19 +01:00
|
|
|
|
|
|
|
return (Datum) 0;
|
|
|
|
}
|
2009-02-24 11:06:36 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Describes the valid options for postgresql FDW, server, and user mapping.
|
|
|
|
*/
|
2009-06-11 16:49:15 +02:00
|
|
|
struct ConnectionOption
|
|
|
|
{
|
|
|
|
const char *optname;
|
|
|
|
Oid optcontext; /* Oid of catalog in which option may appear */
|
2009-02-24 11:06:36 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copied from fe-connect.c PQconninfoOptions.
|
|
|
|
*
|
|
|
|
* The list is small - don't bother with bsearch if it stays so.
|
|
|
|
*/
|
|
|
|
static struct ConnectionOption libpq_conninfo_options[] = {
|
2009-06-11 16:49:15 +02:00
|
|
|
{"authtype", ForeignServerRelationId},
|
|
|
|
{"service", ForeignServerRelationId},
|
|
|
|
{"user", UserMappingRelationId},
|
|
|
|
{"password", UserMappingRelationId},
|
|
|
|
{"connect_timeout", ForeignServerRelationId},
|
|
|
|
{"dbname", ForeignServerRelationId},
|
|
|
|
{"host", ForeignServerRelationId},
|
|
|
|
{"hostaddr", ForeignServerRelationId},
|
|
|
|
{"port", ForeignServerRelationId},
|
|
|
|
{"tty", ForeignServerRelationId},
|
|
|
|
{"options", ForeignServerRelationId},
|
|
|
|
{"requiressl", ForeignServerRelationId},
|
|
|
|
{"sslmode", ForeignServerRelationId},
|
|
|
|
{"gsslib", ForeignServerRelationId},
|
|
|
|
{NULL, InvalidOid}
|
2009-02-24 11:06:36 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if the provided option is one of libpq conninfo options.
|
|
|
|
* context is the Oid of the catalog the option came from, or 0 if we
|
|
|
|
* don't care.
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
is_conninfo_option(const char *option, Oid context)
|
|
|
|
{
|
|
|
|
struct ConnectionOption *opt;
|
|
|
|
|
|
|
|
for (opt = libpq_conninfo_options; opt->optname; opt++)
|
2009-12-23 13:23:59 +01:00
|
|
|
if (context == opt->optcontext && strcmp(opt->optname, option) == 0)
|
2009-02-24 11:06:36 +01:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Validate the generic option given to SERVER or USER MAPPING.
|
2013-02-21 11:26:23 +01:00
|
|
|
* Raise an ERROR if the option or its value is considered invalid.
|
2009-02-24 11:06:36 +01:00
|
|
|
*
|
|
|
|
* Valid server options are all libpq conninfo options except
|
|
|
|
* user and password -- these may only appear in USER MAPPING options.
|
2013-02-21 11:26:23 +01:00
|
|
|
*
|
|
|
|
* Caution: this function is deprecated, and is now meant only for testing
|
|
|
|
* purposes, because the list of options it knows about doesn't necessarily
|
|
|
|
* square with those known to whichever libpq instance you might be using.
|
|
|
|
* Inquire of libpq itself, instead.
|
2009-02-24 11:06:36 +01:00
|
|
|
*/
|
|
|
|
Datum
|
|
|
|
postgresql_fdw_validator(PG_FUNCTION_ARGS)
|
|
|
|
{
|
2009-06-11 16:49:15 +02:00
|
|
|
List *options_list = untransformRelOptions(PG_GETARG_DATUM(0));
|
|
|
|
Oid catalog = PG_GETARG_OID(1);
|
2009-02-24 11:06:36 +01:00
|
|
|
|
2009-06-11 16:49:15 +02:00
|
|
|
ListCell *cell;
|
2009-02-24 11:06:36 +01:00
|
|
|
|
2009-06-11 16:49:15 +02:00
|
|
|
foreach(cell, options_list)
|
2009-02-24 11:06:36 +01:00
|
|
|
{
|
|
|
|
DefElem *def = lfirst(cell);
|
|
|
|
|
|
|
|
if (!is_conninfo_option(def->defname, catalog))
|
|
|
|
{
|
2009-06-11 16:49:15 +02:00
|
|
|
struct ConnectionOption *opt;
|
|
|
|
StringInfoData buf;
|
2009-02-24 11:06:36 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Unknown option specified, complain about it. Provide a hint
|
|
|
|
* with list of valid options for the object.
|
|
|
|
*/
|
|
|
|
initStringInfo(&buf);
|
|
|
|
for (opt = libpq_conninfo_options; opt->optname; opt++)
|
2009-12-23 13:23:59 +01:00
|
|
|
if (catalog == opt->optcontext)
|
2009-02-24 11:06:36 +01:00
|
|
|
appendStringInfo(&buf, "%s%s", (buf.len > 0) ? ", " : "",
|
|
|
|
opt->optname);
|
|
|
|
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_SYNTAX_ERROR),
|
|
|
|
errmsg("invalid option \"%s\"", def->defname),
|
2011-02-20 06:17:18 +01:00
|
|
|
errhint("Valid options in this context are: %s",
|
|
|
|
buf.data)));
|
2009-02-24 11:06:36 +01:00
|
|
|
|
|
|
|
PG_RETURN_BOOL(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PG_RETURN_BOOL(true);
|
|
|
|
}
|
2011-04-01 17:28:28 +02:00
|
|
|
|
2012-03-08 00:20:58 +01:00
|
|
|
|
2011-04-01 17:28:28 +02:00
|
|
|
/*
|
|
|
|
* get_foreign_data_wrapper_oid - given a FDW name, look up the OID
|
|
|
|
*
|
|
|
|
* If missing_ok is false, throw an error if name not found. If true, just
|
|
|
|
* return InvalidOid.
|
|
|
|
*/
|
|
|
|
Oid
|
|
|
|
get_foreign_data_wrapper_oid(const char *fdwname, bool missing_ok)
|
|
|
|
{
|
|
|
|
Oid oid;
|
|
|
|
|
|
|
|
oid = GetSysCacheOid1(FOREIGNDATAWRAPPERNAME, CStringGetDatum(fdwname));
|
|
|
|
if (!OidIsValid(oid) && !missing_ok)
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
|
|
errmsg("foreign-data wrapper \"%s\" does not exist",
|
|
|
|
fdwname)));
|
|
|
|
return oid;
|
|
|
|
}
|
|
|
|
|
2012-03-08 00:20:58 +01:00
|
|
|
|
2011-04-01 17:28:28 +02:00
|
|
|
/*
|
|
|
|
* get_foreign_server_oid - given a FDW name, look up the OID
|
|
|
|
*
|
|
|
|
* If missing_ok is false, throw an error if name not found. If true, just
|
|
|
|
* return InvalidOid.
|
|
|
|
*/
|
|
|
|
Oid
|
|
|
|
get_foreign_server_oid(const char *servername, bool missing_ok)
|
|
|
|
{
|
|
|
|
Oid oid;
|
|
|
|
|
|
|
|
oid = GetSysCacheOid1(FOREIGNSERVERNAME, CStringGetDatum(servername));
|
|
|
|
if (!OidIsValid(oid) && !missing_ok)
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
|
|
errmsg("server \"%s\" does not exist", servername)));
|
|
|
|
return oid;
|
|
|
|
}
|