Peter Childs <[EMAIL PROTECTED]> writes:
> Yes it is possible and I've done it. The reason I'm not using it is because I
> wrote it in Pl/Python and if you attach the same trigger to more than one 
> table in the same transaction pg/python (actually the entire server crashes 
> but thats not the point) crashes. Well it did when I last tested it in early 
> versions.

I've been expecting someone to submit a fix for this, but nobody did
:-(.  So I went ahead and repaired it in CVS tip.  The patch is attached
if you want to try patching your local copy (it looks like it will apply
to 7.3 branch with some fuzz, but I have not actually tested it there).

                        regards, tom lane

*** src/pl/plpython/plpython.c.orig     Mon Aug  4 14:40:50 2003
--- src/pl/plpython/plpython.c  Sun Sep 14 13:07:02 2003
***************
*** 224,236 ****
  
  static PyObject *PLy_procedure_call(PLyProcedure *, char *, PyObject *);
  
! /* returns a cached PLyProcedure, or creates, stores and returns
!  * a new PLyProcedure.
!  */
! static PLyProcedure *PLy_procedure_get(FunctionCallInfo fcinfo, bool);
  
  static PLyProcedure *PLy_procedure_create(FunctionCallInfo fcinfo,
!                                        bool is_trigger,
                                         HeapTuple procTup, char *key);
  
  static void PLy_procedure_compile(PLyProcedure *, const char *);
--- 224,234 ----
  
  static PyObject *PLy_procedure_call(PLyProcedure *, char *, PyObject *);
  
! static PLyProcedure *PLy_procedure_get(FunctionCallInfo fcinfo,
!                                                                          Oid 
tgreloid);
  
  static PLyProcedure *PLy_procedure_create(FunctionCallInfo fcinfo,
!                                        Oid tgreloid,
                                         HeapTuple procTup, char *key);
  
  static void PLy_procedure_compile(PLyProcedure *, const char *);
***************
*** 326,332 ****
  {
        DECLARE_EXC();
        Datum           retval;
-       volatile bool is_trigger;
        PLyProcedure *volatile proc = NULL;
  
        enter();
--- 324,329 ----
***************
*** 337,343 ****
                elog(ERROR, "could not connect to SPI manager");
  
        CALL_LEVEL_INC();
-       is_trigger = CALLED_AS_TRIGGER(fcinfo);
  
        SAVE_EXC();
        if (TRAP_EXC())
--- 334,339 ----
***************
*** 364,379 ****
         * PLy_restart_in_progress);
         */
  
!       proc = PLy_procedure_get(fcinfo, is_trigger);
! 
!       if (is_trigger)
        {
!               HeapTuple       trv = PLy_trigger_handler(fcinfo, proc);
  
                retval = PointerGetDatum(trv);
        }
        else
                retval = PLy_function_handler(fcinfo, proc);
  
        CALL_LEVEL_DEC();
        RESTORE_EXC();
--- 360,380 ----
         * PLy_restart_in_progress);
         */
  
!       if (CALLED_AS_TRIGGER(fcinfo))
        {
!               TriggerData *tdata = (TriggerData *) fcinfo->context;
!               HeapTuple       trv;
  
+               proc = PLy_procedure_get(fcinfo,
+                                                                
RelationGetRelid(tdata->tg_relation));
+               trv = PLy_trigger_handler(fcinfo, proc);
                retval = PointerGetDatum(trv);
        }
        else
+       {
+               proc = PLy_procedure_get(fcinfo, InvalidOid);
                retval = PLy_function_handler(fcinfo, proc);
+       }
  
        CALL_LEVEL_DEC();
        RESTORE_EXC();
***************
*** 962,971 ****
  }
  
  
! /* PLyProcedure functions
   */
  static PLyProcedure *
! PLy_procedure_get(FunctionCallInfo fcinfo, bool is_trigger)
  {
        Oid                     fn_oid;
        HeapTuple       procTup;
--- 963,979 ----
  }
  
  
! /*
!  * PLyProcedure functions
!  */
! 
! /* PLy_procedure_get: returns a cached PLyProcedure, or creates, stores and
!  * returns a new PLyProcedure.  fcinfo is the call info, tgreloid is the
!  * relation OID when calling a trigger, or InvalidOid (zero) for ordinary
!  * function calls.
   */
  static PLyProcedure *
! PLy_procedure_get(FunctionCallInfo fcinfo, Oid tgreloid)
  {
        Oid                     fn_oid;
        HeapTuple       procTup;
***************
*** 983,991 ****
        if (!HeapTupleIsValid(procTup))
                elog(ERROR, "cache lookup failed for function %u", fn_oid);
  
!       rv = snprintf(key, sizeof(key), "%u%s",
!                                 fn_oid,
!                                 is_trigger ? "_trigger" : "");
        if ((rv >= sizeof(key)) || (rv < 0))
                elog(ERROR, "key too long");
  
--- 991,997 ----
        if (!HeapTupleIsValid(procTup))
                elog(ERROR, "cache lookup failed for function %u", fn_oid);
  
!       rv = snprintf(key, sizeof(key), "%u_%u", fn_oid, tgreloid);
        if ((rv >= sizeof(key)) || (rv < 0))
                elog(ERROR, "key too long");
  
***************
*** 1012,1018 ****
        }
  
        if (proc == NULL)
!               proc = PLy_procedure_create(fcinfo, is_trigger, procTup, key);
  
        ReleaseSysCache(procTup);
  
--- 1018,1024 ----
        }
  
        if (proc == NULL)
!               proc = PLy_procedure_create(fcinfo, tgreloid, procTup, key);
  
        ReleaseSysCache(procTup);
  
***************
*** 1020,1026 ****
  }
  
  static PLyProcedure *
! PLy_procedure_create(FunctionCallInfo fcinfo, bool is_trigger,
                                         HeapTuple procTup, char *key)
  {
        char            procName[NAMEDATALEN + 256];
--- 1026,1032 ----
  }
  
  static PLyProcedure *
! PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid,
                                         HeapTuple procTup, char *key)
  {
        char            procName[NAMEDATALEN + 256];
***************
*** 1037,1047 ****
  
        procStruct = (Form_pg_proc) GETSTRUCT(procTup);
  
!       rv = snprintf(procName, sizeof(procName),
!                                 "__plpython_procedure_%s_%u%s",
!                                 NameStr(procStruct->proname),
!                                 fcinfo->flinfo->fn_oid,
!                                 is_trigger ? "_trigger" : "");
        if ((rv >= sizeof(procName)) || (rv < 0))
                elog(ERROR, "procedure name would overrun buffer");
  
--- 1043,1059 ----
  
        procStruct = (Form_pg_proc) GETSTRUCT(procTup);
  
!       if (OidIsValid(tgreloid))
!               rv = snprintf(procName, sizeof(procName),
!                                         "__plpython_procedure_%s_%u_trigger_%u",
!                                         NameStr(procStruct->proname),
!                                         fcinfo->flinfo->fn_oid,
!                                         tgreloid);
!       else
!               rv = snprintf(procName, sizeof(procName),
!                                         "__plpython_procedure_%s_%u",
!                                         NameStr(procStruct->proname),
!                                         fcinfo->flinfo->fn_oid);
        if ((rv >= sizeof(procName)) || (rv < 0))
                elog(ERROR, "procedure name would overrun buffer");
  
***************
*** 1073,1079 ****
         * get information required for output conversion of the return value,
         * but only if this isn't a trigger.
         */
!       if (!is_trigger)
        {
                HeapTuple       rvTypeTup;
                Form_pg_type rvTypeStruct;
--- 1085,1091 ----
         * get information required for output conversion of the return value,
         * but only if this isn't a trigger.
         */
!       if (!CALLED_AS_TRIGGER(fcinfo))
        {
                HeapTuple       rvTypeTup;
                Form_pg_type rvTypeStruct;

---------------------------(end of broadcast)---------------------------
TIP 8: explain analyze is your friend

Reply via email to