2010-07-08 4 views
5

Я получил некоторые функции SQLpsycopg2 callproc и SQL параметры

CREATE OR REPLACE FUNCTION tools.update_company(IN company_id integer, OUT value integer) 
    RETURNS integer AS 
$BODY$ 

BEGIN 
select * into value from function_making_int(company_id) 
END;$BODY$ 

и от psycopg2 (его внутри Джанго, если это имеет значение) я

c = connection.cursor() 
c.callproc('tools.update_company', [1, ]) 

но функция возвращает точно ту же последовательность ввода, как я , игнорируя результаты и параметр OUT. Перейдите в IN OUT, и передача какого-либо значения foo ничего не изменит. При вызове в функции SQL базы данных работает, как ожидалось

ответ

1

Ну я сделал небольшое исследование, и я проверил psycopg2 код - текущее выполнение этой функции просто сделать

select * from function_name(params) 
return params 

так что ничего никак не изменить.

1

взят из исходного pyscopg2 кода, реализация C в callproc с использованием С-апи дать следующий код:

/* callproc method - execute a stored procedure */ 

#define psyco_curs_callproc_doc \ 
"callproc(procname, parameters=None) -- Execute stored procedure." 

static PyObject * 
psyco_curs_callproc(cursorObject *self, PyObject *args) 
{ 
    const char *procname = NULL; 
    char *sql = NULL; 
    Py_ssize_t procname_len, i, nparameters = 0, sl = 0; 
    PyObject *parameters = Py_None; 
    PyObject *operation = NULL; 
    PyObject *res = NULL; 

    if (!PyArg_ParseTuple(args, "s#|O", 
      &procname, &procname_len, &parameters 
     )) 
    { goto exit; } 

    EXC_IF_CURS_CLOSED(self); 
    EXC_IF_ASYNC_IN_PROGRESS(self, callproc); 
    EXC_IF_TPC_PREPARED(self->conn, callproc); 

    if (self->name != NULL) { 
     psyco_set_error(ProgrammingError, self, 
         "can't call .callproc() on named cursors"); 
     goto exit; 
    } 

    if (parameters != Py_None) { 
     if (-1 == (nparameters = PyObject_Length(parameters))) { goto exit;  } 
    } 

    /* allocate some memory, build the SQL and create a PyString from it */ 
    sl = procname_len + 17 + nparameters*3 - (nparameters ? 1 : 0); 
    sql = (char*)PyMem_Malloc(sl); 
    if (sql == NULL) { 
     PyErr_NoMemory(); 
     goto exit; 
    } 

    sprintf(sql, "SELECT * FROM %s(", procname); 
    for(i=0; i<nparameters; i++) { 
     strcat(sql, "%s,"); 
    } 
    sql[sl-2] = ')'; 
    sql[sl-1] = '\0'; 

    if (!(operation = Bytes_FromString(sql))) { goto exit; } 

    if (0 <= _psyco_curs_execute(self, operation, parameters, 
      self->conn->async, 0)) { 
     Py_INCREF(parameters); 
     res = parameters; 
    } 

exit: 
    Py_XDECREF(operation); 
    PyMem_Free((void*)sql); 
    return res; 
} 

вы можете заметить, что не существует локальная модификация на переданных аргументов в «callproc» но результат возвращается:

c = connection.cursor() 
res = c.callproc('tools.update_company', [1, ]) 

и да, на самом деле код произвести «SELECT» SQL на землю, чтобы perfom действие.

 Смежные вопросы

  • Нет связанных вопросов^_^