Server IP : 103.119.228.120 / Your IP : 3.144.8.68 Web Server : Apache System : Linux v8.techscape8.com 3.10.0-1160.119.1.el7.tuxcare.els2.x86_64 #1 SMP Mon Jul 15 12:09:18 UTC 2024 x86_64 User : nobody ( 99) PHP Version : 5.6.40 Disable Function : shell_exec,symlink,system,exec,proc_get_status,proc_nice,proc_terminate,define_syslog_variables,syslog,openlog,closelog,escapeshellcmd,passthru,ocinum cols,ini_alter,leak,listen,chgrp,apache_note,apache_setenv,debugger_on,debugger_off,ftp_exec,dl,dll,myshellexec,proc_open,socket_bind,proc_close,escapeshellarg,parse_ini_filepopen,fpassthru,exec,passthru,escapeshellarg,escapeshellcmd,proc_close,proc_open,ini_alter,popen,show_source,proc_nice,proc_terminate,proc_get_status,proc_close,pfsockopen,leak,apache_child_terminate,posix_kill,posix_mkfifo,posix_setpgid,posix_setsid,posix_setuid,dl,symlink,shell_exec,system,dl,passthru,escapeshellarg,escapeshellcmd,myshellexec,c99_buff_prepare,c99_sess_put,fpassthru,getdisfunc,fx29exec,fx29exec2,is_windows,disp_freespace,fx29sh_getupdate,fx29_buff_prepare,fx29_sess_put,fx29shexit,fx29fsearch,fx29ftpbrutecheck,fx29sh_tools,fx29sh_about,milw0rm,imagez,sh_name,myshellexec,checkproxyhost,dosyayicek,c99_buff_prepare,c99_sess_put,c99getsource,c99sh_getupdate,c99fsearch,c99shexit,view_perms,posix_getpwuid,posix_getgrgid,posix_kill,parse_perms,parsesort,view_perms_color,set_encoder_input,ls_setcheckboxall,ls_reverse_all,rsg_read,rsg_glob,selfURL,dispsecinfo,unix2DosTime,addFile,system,get_users,view_size,DirFiles,DirFilesWide,DirPrintHTMLHeaders,GetFilesTotal,GetTitles,GetTimeTotal,GetMatchesCount,GetFileMatchesCount,GetResultFiles,fs_copy_dir,fs_copy_obj,fs_move_dir,fs_move_obj,fs_rmdir,SearchText,getmicrotime MySQL : ON | cURL : ON | WGET : ON | Perl : ON | Python : ON | Sudo : ON | Pkexec : ON Directory : /usr/local/ssl/local/ssl/share/doc/postgresql-9.2.24/html/ |
Upload File : |
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <HTML ><HEAD ><TITLE >Foreign Data Wrapper Query Planning</TITLE ><META NAME="GENERATOR" CONTENT="Modular DocBook HTML Stylesheet Version 1.79"><LINK REV="MADE" HREF="mailto:pgsql-docs@postgresql.org"><LINK REL="HOME" TITLE="PostgreSQL 9.2.24 Documentation" HREF="index.html"><LINK REL="UP" TITLE="Writing A Foreign Data Wrapper" HREF="fdwhandler.html"><LINK REL="PREVIOUS" TITLE="Foreign Data Wrapper Helper Functions" HREF="fdw-helpers.html"><LINK REL="NEXT" TITLE="Genetic Query Optimizer" HREF="geqo.html"><LINK REL="STYLESHEET" TYPE="text/css" HREF="stylesheet.css"><META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=ISO-8859-1"><META NAME="creation" CONTENT="2017-11-06T22:43:11"></HEAD ><BODY CLASS="SECT1" ><DIV CLASS="NAVHEADER" ><TABLE SUMMARY="Header navigation table" WIDTH="100%" BORDER="0" CELLPADDING="0" CELLSPACING="0" ><TR ><TH COLSPAN="5" ALIGN="center" VALIGN="bottom" ><A HREF="index.html" >PostgreSQL 9.2.24 Documentation</A ></TH ></TR ><TR ><TD WIDTH="10%" ALIGN="left" VALIGN="top" ><A TITLE="Foreign Data Wrapper Helper Functions" HREF="fdw-helpers.html" ACCESSKEY="P" >Prev</A ></TD ><TD WIDTH="10%" ALIGN="left" VALIGN="top" ><A HREF="fdwhandler.html" ACCESSKEY="U" >Up</A ></TD ><TD WIDTH="60%" ALIGN="center" VALIGN="bottom" >Chapter 50. Writing A Foreign Data Wrapper</TD ><TD WIDTH="20%" ALIGN="right" VALIGN="top" ><A TITLE="Genetic Query Optimizer" HREF="geqo.html" ACCESSKEY="N" >Next</A ></TD ></TR ></TABLE ><HR ALIGN="LEFT" WIDTH="100%"></DIV ><DIV CLASS="SECT1" ><H1 CLASS="SECT1" ><A NAME="FDW-PLANNING" >50.4. Foreign Data Wrapper Query Planning</A ></H1 ><P > The FDW callback functions <CODE CLASS="FUNCTION" >GetForeignRelSize</CODE >, <CODE CLASS="FUNCTION" >GetForeignPaths</CODE >, and <CODE CLASS="FUNCTION" >GetForeignPlan</CODE > must fit into the workings of the <SPAN CLASS="PRODUCTNAME" >PostgreSQL</SPAN > planner. Here are some notes about what they must do. </P ><P > The information in <TT CLASS="LITERAL" >root</TT > and <TT CLASS="LITERAL" >baserel</TT > can be used to reduce the amount of information that has to be fetched from the foreign table (and therefore reduce the cost). <TT CLASS="LITERAL" >baserel->baserestrictinfo</TT > is particularly interesting, as it contains restriction quals (<TT CLASS="LITERAL" >WHERE</TT > clauses) that should be used to filter the rows to be fetched. (The FDW itself is not required to enforce these quals, as the core executor can check them instead.) <TT CLASS="LITERAL" >baserel->reltargetlist</TT > can be used to determine which columns need to be fetched; but note that it only lists columns that have to be emitted by the <TT CLASS="STRUCTNAME" >ForeignScan</TT > plan node, not columns that are used in qual evaluation but not output by the query. </P ><P > Various private fields are available for the FDW planning functions to keep information in. Generally, whatever you store in FDW private fields should be palloc'd, so that it will be reclaimed at the end of planning. </P ><P > <TT CLASS="LITERAL" >baserel->fdw_private</TT > is a <TT CLASS="TYPE" >void</TT > pointer that is available for FDW planning functions to store information relevant to the particular foreign table. The core planner does not touch it except to initialize it to NULL when the <TT CLASS="LITERAL" >baserel</TT > node is created. It is useful for passing information forward from <CODE CLASS="FUNCTION" >GetForeignRelSize</CODE > to <CODE CLASS="FUNCTION" >GetForeignPaths</CODE > and/or <CODE CLASS="FUNCTION" >GetForeignPaths</CODE > to <CODE CLASS="FUNCTION" >GetForeignPlan</CODE >, thereby avoiding recalculation. </P ><P > <CODE CLASS="FUNCTION" >GetForeignPaths</CODE > can identify the meaning of different access paths by storing private information in the <TT CLASS="STRUCTFIELD" >fdw_private</TT > field of <TT CLASS="STRUCTNAME" >ForeignPath</TT > nodes. <TT CLASS="STRUCTFIELD" >fdw_private</TT > is declared as a <TT CLASS="TYPE" >List</TT > pointer, but could actually contain anything since the core planner does not touch it. However, best practice is to use a representation that's dumpable by <CODE CLASS="FUNCTION" >nodeToString</CODE >, for use with debugging support available in the backend. </P ><P > <CODE CLASS="FUNCTION" >GetForeignPlan</CODE > can examine the <TT CLASS="STRUCTFIELD" >fdw_private</TT > field of the selected <TT CLASS="STRUCTNAME" >ForeignPath</TT > node, and can generate <TT CLASS="STRUCTFIELD" >fdw_exprs</TT > and <TT CLASS="STRUCTFIELD" >fdw_private</TT > lists to be placed in the <TT CLASS="STRUCTNAME" >ForeignScan</TT > plan node, where they will be available at execution time. Both of these lists must be represented in a form that <CODE CLASS="FUNCTION" >copyObject</CODE > knows how to copy. The <TT CLASS="STRUCTFIELD" >fdw_private</TT > list has no other restrictions and is not interpreted by the core backend in any way. The <TT CLASS="STRUCTFIELD" >fdw_exprs</TT > list, if not NIL, is expected to contain expression trees that are intended to be executed at run time. These trees will undergo post-processing by the planner to make them fully executable. </P ><P > In <CODE CLASS="FUNCTION" >GetForeignPlan</CODE >, generally the passed-in target list can be copied into the plan node as-is. The passed scan_clauses list contains the same clauses as <TT CLASS="LITERAL" >baserel->baserestrictinfo</TT >, but may be re-ordered for better execution efficiency. In simple cases the FDW can just strip <TT CLASS="STRUCTNAME" >RestrictInfo</TT > nodes from the scan_clauses list (using <CODE CLASS="FUNCTION" >extract_actual_clauses</CODE >) and put all the clauses into the plan node's qual list, which means that all the clauses will be checked by the executor at run time. More complex FDWs may be able to check some of the clauses internally, in which case those clauses can be removed from the plan node's qual list so that the executor doesn't waste time rechecking them. </P ><P > As an example, the FDW might identify some restriction clauses of the form <TT CLASS="REPLACEABLE" ><I >foreign_variable</I ></TT > <TT CLASS="LITERAL" >=</TT > <TT CLASS="REPLACEABLE" ><I >sub_expression</I ></TT >, which it determines can be executed on the remote server given the locally-evaluated value of the <TT CLASS="REPLACEABLE" ><I >sub_expression</I ></TT >. The actual identification of such a clause should happen during <CODE CLASS="FUNCTION" >GetForeignPaths</CODE >, since it would affect the cost estimate for the path. The path's <TT CLASS="STRUCTFIELD" >fdw_private</TT > field would probably include a pointer to the identified clause's <TT CLASS="STRUCTNAME" >RestrictInfo</TT > node. Then <CODE CLASS="FUNCTION" >GetForeignPlan</CODE > would remove that clause from scan_clauses, but add the <TT CLASS="REPLACEABLE" ><I >sub_expression</I ></TT > to <TT CLASS="STRUCTFIELD" >fdw_exprs</TT > to ensure that it gets massaged into executable form. It would probably also put control information into the plan node's <TT CLASS="STRUCTFIELD" >fdw_private</TT > field to tell the execution functions what to do at run time. The query transmitted to the remote server would involve something like <TT CLASS="LITERAL" >WHERE <TT CLASS="REPLACEABLE" ><I >foreign_variable</I ></TT > = $1</TT >, with the parameter value obtained at run time from evaluation of the <TT CLASS="STRUCTFIELD" >fdw_exprs</TT > expression tree. </P ><P > The FDW should always construct at least one path that depends only on the table's restriction clauses. In join queries, it might also choose to construct path(s) that depend on join clauses, for example <TT CLASS="REPLACEABLE" ><I >foreign_variable</I ></TT > <TT CLASS="LITERAL" >=</TT > <TT CLASS="REPLACEABLE" ><I >local_variable</I ></TT >. Such clauses will not be found in <TT CLASS="LITERAL" >baserel->baserestrictinfo</TT > but must be sought in the relation's join lists. A path using such a clause is called a <SPAN CLASS="QUOTE" >"parameterized path"</SPAN >. It must identify the other relations used in the selected join clause(s) with a suitable value of <TT CLASS="LITERAL" >param_info</TT >; use <CODE CLASS="FUNCTION" >get_baserel_parampathinfo</CODE > to compute that value. In <CODE CLASS="FUNCTION" >GetForeignPlan</CODE >, the <TT CLASS="REPLACEABLE" ><I >local_variable</I ></TT > portion of the join clause would be added to <TT CLASS="STRUCTFIELD" >fdw_exprs</TT >, and then at run time the case works the same as for an ordinary restriction clause. </P ></DIV ><DIV CLASS="NAVFOOTER" ><HR ALIGN="LEFT" WIDTH="100%"><TABLE SUMMARY="Footer navigation table" WIDTH="100%" BORDER="0" CELLPADDING="0" CELLSPACING="0" ><TR ><TD WIDTH="33%" ALIGN="left" VALIGN="top" ><A HREF="fdw-helpers.html" ACCESSKEY="P" >Prev</A ></TD ><TD WIDTH="34%" ALIGN="center" VALIGN="top" ><A HREF="index.html" ACCESSKEY="H" >Home</A ></TD ><TD WIDTH="33%" ALIGN="right" VALIGN="top" ><A HREF="geqo.html" ACCESSKEY="N" >Next</A ></TD ></TR ><TR ><TD WIDTH="33%" ALIGN="left" VALIGN="top" >Foreign Data Wrapper Helper Functions</TD ><TD WIDTH="34%" ALIGN="center" VALIGN="top" ><A HREF="fdwhandler.html" ACCESSKEY="U" >Up</A ></TD ><TD WIDTH="33%" ALIGN="right" VALIGN="top" >Genetic Query Optimizer</TD ></TR ></TABLE ></DIV ></BODY ></HTML >