mirror of
https://github.com/php/php-src.git
synced 2025-01-23 12:13:37 +08:00
1504 lines
40 KiB
C
1504 lines
40 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| PHP version 4.0 |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 1997, 1998, 1999 The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 2.0 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available at through the world-wide-web at |
|
|
| http://www.php.net/license/2_0.txt. |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Authors: Zeev Suraski <zeev@zend.com> |
|
|
| Jouni Ahto <jah@cultnet.fi> (large object interface) |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
#include <stdlib.h>
|
|
|
|
#if COMPILE_DL
|
|
#include "dl/phpdl.h"
|
|
#endif
|
|
|
|
#include "php.h"
|
|
#include "php_pgsql.h"
|
|
#include "ext/standard/php_standard.h"
|
|
#include "php_globals.h"
|
|
|
|
#if HAVE_PGSQL
|
|
|
|
|
|
#define PGSQL_ASSOC 1<<0
|
|
#define PGSQL_NUM 1<<1
|
|
#define PGSQL_BOTH (PGSQL_ASSOC|PGSQL_NUM)
|
|
|
|
|
|
function_entry pgsql_functions[] = {
|
|
PHP_FE(pg_connect, NULL)
|
|
PHP_FE(pg_pconnect, NULL)
|
|
PHP_FE(pg_close, NULL)
|
|
PHP_FE(pg_cmdtuples, NULL)
|
|
PHP_FE(pg_dbname, NULL)
|
|
PHP_FE(pg_errormessage, NULL)
|
|
PHP_FE(pg_options, NULL)
|
|
PHP_FE(pg_port, NULL)
|
|
PHP_FE(pg_tty, NULL)
|
|
PHP_FE(pg_host, NULL)
|
|
PHP_FE(pg_exec, NULL)
|
|
PHP_FE(pg_numrows, NULL)
|
|
PHP_FE(pg_numfields, NULL)
|
|
PHP_FE(pg_fieldname, NULL)
|
|
PHP_FE(pg_fieldsize, NULL)
|
|
PHP_FE(pg_fieldtype, NULL)
|
|
PHP_FE(pg_fieldnum, NULL)
|
|
PHP_FE(pg_result, NULL)
|
|
PHP_FE(pg_fetch_row, NULL)
|
|
PHP_FE(pg_fetch_array, NULL)
|
|
PHP_FE(pg_fetch_object, NULL)
|
|
PHP_FE(pg_fieldprtlen, NULL)
|
|
PHP_FE(pg_fieldisnull, NULL)
|
|
PHP_FE(pg_freeresult, NULL)
|
|
PHP_FE(pg_getlastoid, NULL)
|
|
PHP_FE(pg_locreate, NULL)
|
|
PHP_FE(pg_lounlink, NULL)
|
|
PHP_FE(pg_loopen, NULL)
|
|
PHP_FE(pg_loclose, NULL)
|
|
PHP_FE(pg_loread, NULL)
|
|
PHP_FE(pg_write, NULL)
|
|
PHP_FE(pg_loreadall, NULL)
|
|
{NULL, NULL, NULL}
|
|
};
|
|
|
|
zend_module_entry pgsql_module_entry = {
|
|
"PostgreSQL", pgsql_functions, PHP_MINIT(pgsql), NULL, PHP_RINIT(pgsql), NULL, NULL, STANDARD_MODULE_PROPERTIES
|
|
};
|
|
|
|
#if COMPILE_DL
|
|
PHP_PGSQL_API zend_module_entry *get_module() { return &pgsql_module_entry; }
|
|
#endif
|
|
|
|
static int le_link,le_plink,le_result,le_lofp,le_string;
|
|
|
|
#ifdef ZTS
|
|
int pgsql_globals_id;
|
|
#else
|
|
PHP_PGSQL_API php_pgsql_globals pgsql_globals;
|
|
#endif
|
|
|
|
static void _close_pgsql_link(PGconn *link)
|
|
{
|
|
PGLS_FETCH();
|
|
|
|
PQfinish(link);
|
|
PGG(num_links)--;
|
|
}
|
|
|
|
|
|
static void _close_pgsql_plink(PGconn *link)
|
|
{
|
|
PGLS_FETCH();
|
|
|
|
PQfinish(link);
|
|
PGG(num_persistent)--;
|
|
PGG(num_links)--;
|
|
}
|
|
|
|
|
|
static void _free_ptr(pgLofp *lofp)
|
|
{
|
|
efree(lofp);
|
|
}
|
|
|
|
|
|
static void _free_result(pgsql_result_handle *pg_result)
|
|
{
|
|
PQclear(pg_result->result);
|
|
efree(pg_result);
|
|
}
|
|
|
|
static void php_pgsql_init_globals(PGLS_D)
|
|
{
|
|
PGG(num_persistent) = 0;
|
|
if (cfg_get_long("pgsql.allow_persistent",&PGG(allow_persistent))==FAILURE) {
|
|
PGG(allow_persistent)=1;
|
|
}
|
|
if (cfg_get_long("pgsql.max_persistent",&PGG(max_persistent))==FAILURE) {
|
|
PGG(max_persistent)=-1;
|
|
}
|
|
if (cfg_get_long("pgsql.max_links",&PGG(max_links))==FAILURE) {
|
|
PGG(max_links)=-1;
|
|
}
|
|
}
|
|
|
|
PHP_MINIT_FUNCTION(pgsql)
|
|
{
|
|
ELS_FETCH();
|
|
|
|
#ifdef ZTS
|
|
pgsql_globals_id = ts_allocate_id(sizeof(php_pgsql_globals), (ts_allocate_ctor) php_pgsql_init_globals, NULL);
|
|
#else
|
|
php_pgsql_init_globals(PGLS_C);
|
|
#endif
|
|
|
|
le_link = register_list_destructors(_close_pgsql_link,NULL);
|
|
le_plink = register_list_destructors(NULL,_close_pgsql_plink);
|
|
/* PGG(le_result = register_list_destructors(PQclear,NULL); */
|
|
le_result = register_list_destructors(_free_result,NULL);
|
|
le_lofp = register_list_destructors(_free_ptr,NULL);
|
|
le_string = register_list_destructors(_free_ptr,NULL);
|
|
|
|
REGISTER_LONG_CONSTANT("PGSQL_ASSOC", PGSQL_ASSOC, CONST_CS | CONST_PERSISTENT);
|
|
REGISTER_LONG_CONSTANT("PGSQL_NUM", PGSQL_NUM, CONST_CS | CONST_PERSISTENT);
|
|
REGISTER_LONG_CONSTANT("PGSQL_BOTH", PGSQL_BOTH, CONST_CS | CONST_PERSISTENT);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
|
|
PHP_RINIT_FUNCTION(pgsql)
|
|
{
|
|
PGLS_FETCH();
|
|
|
|
PGG(default_link)=-1;
|
|
PGG(num_links) = PGG(num_persistent);
|
|
return SUCCESS;
|
|
}
|
|
|
|
|
|
void php_pgsql_do_connect(INTERNAL_FUNCTION_PARAMETERS,int persistent)
|
|
{
|
|
char *host=NULL,*port=NULL,*options=NULL,*tty=NULL,*dbname=NULL,*connstring=NULL;
|
|
char *hashed_details;
|
|
int hashed_details_length;
|
|
PGconn *pgsql;
|
|
PGLS_FETCH();
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 1: { /* new style, using connection string */
|
|
pval *yyconnstring;
|
|
if (getParameters(ht, 1, &yyconnstring) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_string(yyconnstring);
|
|
connstring = yyconnstring->value.str.val;
|
|
hashed_details_length = yyconnstring->value.str.len+5+1;
|
|
hashed_details = (char *) emalloc(hashed_details_length+1);
|
|
sprintf(hashed_details,"pgsql_%s",connstring); /* SAFE */
|
|
}
|
|
break;
|
|
case 3: { /* host, port, dbname */
|
|
pval *yyhost, *yyport, *yydbname;
|
|
|
|
if (getParameters(ht, 3, &yyhost, &yyport, &yydbname) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_string(yyhost);
|
|
convert_to_string(yyport);
|
|
convert_to_string(yydbname);
|
|
host = yyhost->value.str.val;
|
|
port = yyport->value.str.val;
|
|
dbname = yydbname->value.str.val;
|
|
options=tty=NULL;
|
|
hashed_details_length = yyhost->value.str.len+yyport->value.str.len+yydbname->value.str.len+5+5;
|
|
hashed_details = (char *) emalloc(hashed_details_length+1);
|
|
sprintf(hashed_details,"pgsql_%s_%s___%s",host,port,dbname); /* SAFE */
|
|
}
|
|
break;
|
|
case 4: { /* host, port, options, dbname */
|
|
pval *yyhost, *yyport, *yyoptions, *yydbname;
|
|
|
|
if (getParameters(ht, 4, &yyhost, &yyport, &yyoptions, &yydbname) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_string(yyhost);
|
|
convert_to_string(yyport);
|
|
convert_to_string(yyoptions);
|
|
convert_to_string(yydbname);
|
|
host = yyhost->value.str.val;
|
|
port = yyport->value.str.val;
|
|
options = yyoptions->value.str.val;
|
|
dbname = yydbname->value.str.val;
|
|
tty=NULL;
|
|
hashed_details_length = yyhost->value.str.len+yyport->value.str.len+yyoptions->value.str.len+yydbname->value.str.len+5+5;
|
|
hashed_details = (char *) emalloc(hashed_details_length+1);
|
|
sprintf(hashed_details,"pgsql_%s_%s_%s__%s",host,port,options,dbname); /* SAFE */
|
|
}
|
|
break;
|
|
case 5: { /* host, port, options, tty, dbname */
|
|
pval *yyhost, *yyport, *yyoptions, *yytty, *yydbname;
|
|
|
|
if (getParameters(ht, 5, &yyhost, &yyport, &yyoptions, &yytty, &yydbname) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_string(yyhost);
|
|
convert_to_string(yyport);
|
|
convert_to_string(yyoptions);
|
|
convert_to_string(yytty);
|
|
convert_to_string(yydbname);
|
|
host = yyhost->value.str.val;
|
|
port = yyport->value.str.val;
|
|
options = yyoptions->value.str.val;
|
|
tty = yytty->value.str.val;
|
|
dbname = yydbname->value.str.val;
|
|
hashed_details_length = yyhost->value.str.len+yyport->value.str.len+yyoptions->value.str.len+yytty->value.str.len+yydbname->value.str.len+5+5;
|
|
hashed_details = (char *) emalloc(hashed_details_length+1);
|
|
sprintf(hashed_details,"pgsql_%s_%s_%s_%s_%s",host,port,options,tty,dbname); /* SAFE */
|
|
}
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
if (persistent) {
|
|
list_entry *le;
|
|
|
|
/* try to find if we already have this link in our persistent list */
|
|
if (zend_hash_find(plist, hashed_details, hashed_details_length+1, (void **) &le)==FAILURE) { /* we don't */
|
|
list_entry new_le;
|
|
|
|
if (PGG(max_links)!=-1 && PGG(num_links)>=PGG(max_links)) {
|
|
php_error(E_WARNING,"PostgresSQL: Too many open links (%d)",PGG(num_links));
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
if (PGG(max_persistent)!=-1 && PGG(num_persistent)>=PGG(max_persistent)) {
|
|
php_error(E_WARNING,"PostgresSQL: Too many open persistent links (%d)",PGG(num_persistent));
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
/* create the link */
|
|
if (connstring) {
|
|
pgsql=PQconnectdb(connstring);
|
|
} else {
|
|
pgsql=PQsetdb(host,port,options,tty,dbname);
|
|
}
|
|
if (pgsql==NULL || PQstatus(pgsql)==CONNECTION_BAD) {
|
|
php_error(E_WARNING,"Unable to connect to PostgresSQL server: %s",PQerrorMessage(pgsql));
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
/* hash it up */
|
|
new_le.type = le_plink;
|
|
new_le.ptr = pgsql;
|
|
if (zend_hash_update(plist, hashed_details, hashed_details_length+1, (void *) &new_le, sizeof(list_entry), NULL)==FAILURE) {
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
PGG(num_links)++;
|
|
PGG(num_persistent)++;
|
|
} else { /* we do */
|
|
if (le->type != le_plink) {
|
|
RETURN_FALSE;
|
|
}
|
|
/* ensure that the link did not die */
|
|
if (PQstatus(le->ptr)==CONNECTION_BAD) { /* the link died */
|
|
if (connstring) {
|
|
le->ptr=PQconnectdb(connstring);
|
|
} else {
|
|
le->ptr=PQsetdb(host,port,options,tty,dbname);
|
|
}
|
|
if (le->ptr==NULL || PQstatus(le->ptr)==CONNECTION_BAD) {
|
|
php_error(E_WARNING,"PostgresSQL link lost, unable to reconnect");
|
|
zend_hash_del(plist,hashed_details,hashed_details_length+1);
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
pgsql = (PGconn *) le->ptr;
|
|
}
|
|
return_value->value.lval = zend_list_insert(pgsql,le_plink);
|
|
return_value->type = IS_LONG;
|
|
} else {
|
|
list_entry *index_ptr,new_index_ptr;
|
|
|
|
/* first we check the hash for the hashed_details key. if it exists,
|
|
* it should point us to the right offset where the actual pgsql link sits.
|
|
* if it doesn't, open a new pgsql link, add it to the resource list,
|
|
* and add a pointer to it with hashed_details as the key.
|
|
*/
|
|
if (zend_hash_find(list,hashed_details,hashed_details_length+1,(void **) &index_ptr)==SUCCESS) {
|
|
int type,link;
|
|
void *ptr;
|
|
|
|
if (index_ptr->type != le_index_ptr) {
|
|
RETURN_FALSE;
|
|
}
|
|
link = (int) (long) index_ptr->ptr;
|
|
ptr = zend_list_find(link,&type); /* check if the link is still there */
|
|
if (ptr && (type==le_link || type==le_plink)) {
|
|
return_value->value.lval = PGG(default_link) = link;
|
|
return_value->type = IS_LONG;
|
|
efree(hashed_details);
|
|
return;
|
|
} else {
|
|
zend_hash_del(list,hashed_details,hashed_details_length+1);
|
|
}
|
|
}
|
|
if (PGG(max_links)!=-1 && PGG(num_links)>=PGG(max_links)) {
|
|
php_error(E_WARNING,"PostgresSQL: Too many open links (%d)",PGG(num_links));
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
if (connstring) {
|
|
pgsql=PQconnectdb(connstring);
|
|
} else {
|
|
pgsql=PQsetdb(host,port,options,tty,dbname);
|
|
}
|
|
if (pgsql==NULL || PQstatus(pgsql)==CONNECTION_BAD) {
|
|
php_error(E_WARNING,"Unable to connect to PostgresSQL server: %s",PQerrorMessage(pgsql));
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
/* add it to the list */
|
|
return_value->value.lval = zend_list_insert(pgsql,le_link);
|
|
return_value->type = IS_LONG;
|
|
|
|
/* add it to the hash */
|
|
new_index_ptr.ptr = (void *) return_value->value.lval;
|
|
new_index_ptr.type = le_index_ptr;
|
|
if (zend_hash_update(list,hashed_details,hashed_details_length+1,(void *) &new_index_ptr, sizeof(list_entry), NULL)==FAILURE) {
|
|
efree(hashed_details);
|
|
RETURN_FALSE;
|
|
}
|
|
PGG(num_links)++;
|
|
}
|
|
efree(hashed_details);
|
|
PGG(default_link)=return_value->value.lval;
|
|
}
|
|
|
|
|
|
int php_pgsql_get_default_link(INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
PGLS_FETCH();
|
|
|
|
if (PGG(default_link)==-1) { /* no link opened yet, implicitly open one */
|
|
ht = 0;
|
|
php_pgsql_do_connect(INTERNAL_FUNCTION_PARAM_PASSTHRU,0);
|
|
}
|
|
return PGG(default_link);
|
|
}
|
|
|
|
|
|
/* {{{ proto int pg_connect([string connection_string] | [string host, string port, [string options, [string tty,]] string database)
|
|
Open a PostgreSQL connection */
|
|
PHP_FUNCTION(pg_connect)
|
|
{
|
|
php_pgsql_do_connect(INTERNAL_FUNCTION_PARAM_PASSTHRU,0);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_pconnect([string connection_string] | [string host, string port, [string options, [string tty,]] string database)
|
|
Open a persistent PostgreSQL connection */
|
|
PHP_FUNCTION(pg_pconnect)
|
|
{
|
|
php_pgsql_do_connect(INTERNAL_FUNCTION_PARAM_PASSTHRU,1);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool pg_close([int connection])
|
|
Close a PostgreSQL connection */
|
|
PHP_FUNCTION(pg_close)
|
|
{
|
|
pval *pgsql_link;
|
|
int id,type;
|
|
PGconn *pgsql;
|
|
PGLS_FETCH();
|
|
|
|
switch (ARG_COUNT(ht)) {
|
|
case 0:
|
|
id = PGG(default_link);
|
|
break;
|
|
case 1:
|
|
if (getParameters(ht, 1, &pgsql_link)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_link);
|
|
id = pgsql_link->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (PGconn *) zend_list_find(id,&type);
|
|
if (type!=le_link && type!=le_plink) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL link index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
zend_list_delete(pgsql_link->value.lval);
|
|
RETURN_TRUE;
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
#define PHP_PG_DBNAME 1
|
|
#define PHP_PG_ERROR_MESSAGE 2
|
|
#define PHP_PG_OPTIONS 3
|
|
#define PHP_PG_PORT 4
|
|
#define PHP_PG_TTY 5
|
|
#define PHP_PG_HOST 6
|
|
|
|
void php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAMETERS, int entry_type)
|
|
{
|
|
pval *pgsql_link;
|
|
int id,type;
|
|
PGconn *pgsql;
|
|
PGLS_FETCH();
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 0:
|
|
id = PGG(default_link);
|
|
break;
|
|
case 1:
|
|
if (getParameters(ht, 1, &pgsql_link)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_link);
|
|
id = pgsql_link->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (PGconn *) zend_list_find(id,&type);
|
|
if (type!=le_link && type!=le_plink) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL link index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
switch(entry_type) {
|
|
case PHP_PG_DBNAME:
|
|
return_value->value.str.val = PQdb(pgsql);
|
|
break;
|
|
case PHP_PG_ERROR_MESSAGE:
|
|
return_value->value.str.val = PQerrorMessage(pgsql);
|
|
break;
|
|
case PHP_PG_OPTIONS:
|
|
return_value->value.str.val = PQoptions(pgsql);
|
|
break;
|
|
case PHP_PG_PORT:
|
|
return_value->value.str.val = PQport(pgsql);
|
|
break;
|
|
case PHP_PG_TTY:
|
|
return_value->value.str.val = PQtty(pgsql);
|
|
break;
|
|
case PHP_PG_HOST:
|
|
return_value->value.str.val = PQhost(pgsql);
|
|
break;
|
|
default:
|
|
RETURN_FALSE;
|
|
}
|
|
return_value->value.str.len = strlen(return_value->value.str.val);
|
|
return_value->value.str.val = (char *) estrdup(return_value->value.str.val);
|
|
return_value->type = IS_STRING;
|
|
}
|
|
|
|
/* {{{ proto string pg_dbname([int connection])
|
|
Get the database name */
|
|
PHP_FUNCTION(pg_dbname)
|
|
{
|
|
php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_DBNAME);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string pg_errormessage([int connection])
|
|
Get the error message string */
|
|
PHP_FUNCTION(pg_error_message)
|
|
{
|
|
php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_ERROR_MESSAGE);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string pg_options([int connection])
|
|
Get the options associated with the connection */
|
|
PHP_FUNCTION(pg_options)
|
|
{
|
|
php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_OPTIONS);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_port([int connection])
|
|
Return the port number associated with the connection */
|
|
PHP_FUNCTION(pg_port)
|
|
{
|
|
php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_PORT);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string pg_tty([int connection])
|
|
Return the tty name associated with the connection */
|
|
PHP_FUNCTION(pg_tty)
|
|
{
|
|
php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_TTY);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string pg_host([int connection])
|
|
Returns the host name associated with the connection */
|
|
PHP_FUNCTION(pg_host)
|
|
{
|
|
php_pgsql_get_link_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_HOST);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_exec([int connection,] string query)
|
|
Execute a query */
|
|
PHP_FUNCTION(pg_exec)
|
|
{
|
|
pval *query,*pgsql_link;
|
|
int id,type;
|
|
PGconn *pgsql;
|
|
PGresult *pgsql_result;
|
|
ExecStatusType status;
|
|
pgsql_result_handle *pg_result;
|
|
PGLS_FETCH();
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 1:
|
|
if (getParameters(ht, 1, &query)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
id = PGG(default_link);
|
|
break;
|
|
case 2:
|
|
if (getParameters(ht, 2, &pgsql_link, &query)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_link);
|
|
id = pgsql_link->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (PGconn *) zend_list_find(id,&type);
|
|
if (type!=le_link && type!=le_plink) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL link index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
convert_to_string(query);
|
|
pgsql_result=PQexec(pgsql,query->value.str.val);
|
|
|
|
if (pgsql_result) {
|
|
status = PQresultStatus(pgsql_result);
|
|
} else {
|
|
status = (ExecStatusType) PQstatus(pgsql);
|
|
}
|
|
|
|
|
|
switch (status) {
|
|
case PGRES_EMPTY_QUERY:
|
|
case PGRES_BAD_RESPONSE:
|
|
case PGRES_NONFATAL_ERROR:
|
|
case PGRES_FATAL_ERROR:
|
|
php_error(E_WARNING,"PostgresSQL query failed: %s",PQerrorMessage(pgsql));
|
|
RETURN_FALSE;
|
|
break;
|
|
case PGRES_COMMAND_OK: /* successful command that did not return rows */
|
|
default:
|
|
if (pgsql_result) {
|
|
pg_result = (pgsql_result_handle *) emalloc(sizeof(pgsql_result_handle));
|
|
pg_result->conn = pgsql;
|
|
pg_result->result = pgsql_result;
|
|
return_value->value.lval = zend_list_insert(pg_result,le_result);
|
|
return_value->type = IS_LONG;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
#define PHP_PG_NUM_ROWS 1
|
|
#define PHP_PG_NUM_FIELDS 2
|
|
#define PHP_PG_CMD_TUPLES 3
|
|
|
|
void php_pgsql_get_result_info(INTERNAL_FUNCTION_PARAMETERS, int entry_type)
|
|
{
|
|
pval *result;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type;
|
|
|
|
if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &result)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
pgsql_result = pg_result->result;
|
|
|
|
switch (entry_type) {
|
|
case PHP_PG_NUM_ROWS:
|
|
return_value->value.lval = PQntuples(pgsql_result);
|
|
break;
|
|
case PHP_PG_NUM_FIELDS:
|
|
return_value->value.lval = PQnfields(pgsql_result);
|
|
break;
|
|
case PHP_PG_CMD_TUPLES:
|
|
#if HAVE_PQCMDTUPLES
|
|
return_value->value.lval = atoi(PQcmdTuples(pgsql_result));
|
|
#else
|
|
php_error(E_WARNING,"This compilation does not support pg_cmdtuples()");
|
|
return_value->value.lval = 0;
|
|
#endif
|
|
break;
|
|
default:
|
|
RETURN_FALSE;
|
|
}
|
|
return_value->type = IS_LONG;
|
|
}
|
|
|
|
/* {{{ proto int pg_numrows(int result)
|
|
Return the number of rows in the result */
|
|
PHP_FUNCTION(pg_num_rows)
|
|
{
|
|
php_pgsql_get_result_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_NUM_ROWS);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_numfields(int result)
|
|
Return the number of fields in the result */
|
|
PHP_FUNCTION(pg_num_fields)
|
|
{
|
|
php_pgsql_get_result_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_NUM_FIELDS);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_cmdtuples(int result)
|
|
Returns the number of affected tuples */
|
|
PHP_FUNCTION(pg_cmdtuples)
|
|
{
|
|
php_pgsql_get_result_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_CMD_TUPLES);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
char *get_field_name(PGconn *pgsql, Oid oid, HashTable *list)
|
|
{
|
|
PGresult *result;
|
|
char hashed_oid_key[32];
|
|
list_entry *field_type;
|
|
char *ret=NULL;
|
|
|
|
/* try to lookup the type in the resource list */
|
|
snprintf(hashed_oid_key,31,"pgsql_oid_%d",(int) oid);
|
|
hashed_oid_key[31]=0;
|
|
|
|
if (zend_hash_find(list,hashed_oid_key,strlen(hashed_oid_key)+1,(void **) &field_type)==SUCCESS) {
|
|
ret = estrdup((char *)field_type->ptr);
|
|
} else { /* hash all oid's */
|
|
int i,num_rows;
|
|
int oid_offset,name_offset;
|
|
char *tmp_oid, *tmp_name;
|
|
list_entry new_oid_entry;
|
|
|
|
if ((result=PQexec(pgsql,"select oid,typname from pg_type"))==NULL) {
|
|
return empty_string;
|
|
}
|
|
num_rows=PQntuples(result);
|
|
oid_offset = PQfnumber(result,"oid");
|
|
name_offset = PQfnumber(result,"typname");
|
|
|
|
for (i=0; i<num_rows; i++) {
|
|
if ((tmp_oid=PQgetvalue(result,i,oid_offset))==NULL) {
|
|
continue;
|
|
}
|
|
snprintf(hashed_oid_key,31,"pgsql_oid_%s",tmp_oid);
|
|
if ((tmp_name=PQgetvalue(result,i,name_offset))==NULL) {
|
|
continue;
|
|
}
|
|
new_oid_entry.type = le_string;
|
|
new_oid_entry.ptr = estrdup(tmp_name);
|
|
zend_hash_update(list,hashed_oid_key,strlen(hashed_oid_key)+1,(void *) &new_oid_entry, sizeof(list_entry), NULL);
|
|
if (!ret && atoi(tmp_oid)==oid) {
|
|
ret = estrdup(tmp_name);
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
|
|
#define PHP_PG_FIELD_NAME 1
|
|
#define PHP_PG_FIELD_SIZE 2
|
|
#define PHP_PG_FIELD_TYPE 3
|
|
|
|
void php_pgsql_get_field_info(INTERNAL_FUNCTION_PARAMETERS, int entry_type)
|
|
{
|
|
pval *result,*field;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type;
|
|
|
|
if (ARG_COUNT(ht)!=2 || getParameters(ht, 2, &result, &field)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
pgsql_result = pg_result->result;
|
|
convert_to_long(field);
|
|
|
|
if (field->value.lval<0 || field->value.lval>=PQnfields(pgsql_result)) {
|
|
php_error(E_WARNING,"Bad field offset specified");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
switch (entry_type) {
|
|
case PHP_PG_FIELD_NAME:
|
|
return_value->value.str.val = PQfname(pgsql_result,field->value.lval);
|
|
return_value->value.str.len = strlen(return_value->value.str.val);
|
|
return_value->value.str.val = estrndup(return_value->value.str.val,return_value->value.str.len);
|
|
return_value->type = IS_STRING;
|
|
break;
|
|
case PHP_PG_FIELD_SIZE:
|
|
return_value->value.lval = PQfsize(pgsql_result,field->value.lval);
|
|
return_value->type = IS_LONG;
|
|
break;
|
|
case PHP_PG_FIELD_TYPE:
|
|
return_value->value.str.val = get_field_name(pg_result->conn,PQftype(pgsql_result,field->value.lval),list);
|
|
return_value->value.str.len = strlen(return_value->value.str.val);
|
|
return_value->type = IS_STRING;
|
|
break;
|
|
default:
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
/* {{{ proto string pg_fieldname(int result, int field_number)
|
|
Returns the name of the field */
|
|
PHP_FUNCTION(pg_field_name)
|
|
{
|
|
php_pgsql_get_field_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_FIELD_NAME);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_fieldsize(int result, int field_number)
|
|
Returns the internal size of the field */
|
|
PHP_FUNCTION(pg_field_size)
|
|
{
|
|
php_pgsql_get_field_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_FIELD_SIZE);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string pg_fieldtype(int result, int field_number)
|
|
Returns the type name for the given field */
|
|
PHP_FUNCTION(pg_field_type)
|
|
{
|
|
php_pgsql_get_field_info(INTERNAL_FUNCTION_PARAM_PASSTHRU,PHP_PG_FIELD_TYPE);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_fieldnum(int result, string field_name)
|
|
Returns the field number of the named field */
|
|
PHP_FUNCTION(pg_field_number)
|
|
{
|
|
pval *result,*field;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type;
|
|
|
|
if (ARG_COUNT(ht)!=2 || getParameters(ht, 2, &result, &field)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
pgsql_result = pg_result->result;
|
|
|
|
convert_to_string(field);
|
|
return_value->value.lval = PQfnumber(pgsql_result,field->value.str.val);
|
|
return_value->type = IS_LONG;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto mixed pg_result(int result, int row_number, mixed field_name)
|
|
Returns values from a result identifier */
|
|
PHP_FUNCTION(pg_result)
|
|
{
|
|
pval *result, *row, *field=NULL;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type,field_offset;
|
|
|
|
if (ARG_COUNT(ht)!=3 || getParameters(ht, 3, &result, &row, &field)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
pgsql_result = pg_result->result;
|
|
|
|
convert_to_long(row);
|
|
if (row->value.lval<0 || row->value.lval>=PQntuples(pgsql_result)) {
|
|
php_error(E_WARNING,"Unable to jump to row %d on PostgresSQL result index %d",row->value.lval,result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
switch(field->type) {
|
|
case IS_STRING:
|
|
field_offset = PQfnumber(pgsql_result,field->value.str.val);
|
|
break;
|
|
default:
|
|
convert_to_long(field);
|
|
field_offset = field->value.lval;
|
|
break;
|
|
}
|
|
if (field_offset<0 || field_offset>=PQnfields(pgsql_result)) {
|
|
php_error(E_WARNING,"Bad column offset specified");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
return_value->value.str.val = PQgetvalue(pgsql_result,row->value.lval,field_offset);
|
|
return_value->value.str.len = (return_value->value.str.val ? strlen(return_value->value.str.val) : 0);
|
|
return_value->value.str.val = safe_estrndup(return_value->value.str.val,return_value->value.str.len);
|
|
return_value->type = IS_STRING;
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
static void php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAMETERS, int result_type)
|
|
{
|
|
pval *result, *row, *arg3;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type;
|
|
int i,num_fields;
|
|
char *element,*field_name;
|
|
uint element_len;
|
|
PLS_FETCH();
|
|
|
|
switch (ARG_COUNT(ht)) {
|
|
case 2:
|
|
if (getParameters(ht, 2, &result, &row)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
if (!result_type) {
|
|
result_type = PGSQL_BOTH;
|
|
}
|
|
break;
|
|
case 3:
|
|
if (getParameters(ht, 3, &result, &row, &arg3)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(arg3);
|
|
result_type = arg3->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
pgsql_result = pg_result->result;
|
|
|
|
convert_to_long(row);
|
|
if (row->value.lval<0 || row->value.lval>=PQntuples(pgsql_result)) {
|
|
php_error(E_WARNING,"Unable to jump to row %d on PostgresSQL result index %d",row->value.lval,result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
array_init(return_value);
|
|
for (i=0,num_fields=PQnfields(pgsql_result); i<num_fields; i++) {
|
|
element = PQgetvalue(pgsql_result,row->value.lval,i);
|
|
element_len = (element ? strlen(element) : 0);
|
|
if (element) {
|
|
char *data;
|
|
int data_len;
|
|
int should_copy=0;
|
|
|
|
if (PG(magic_quotes_runtime)) {
|
|
data = php_addslashes(element,element_len,&data_len,0);
|
|
} else {
|
|
data = safe_estrndup(element,element_len);
|
|
data_len = element_len;
|
|
}
|
|
|
|
if (result_type & PGSQL_NUM) {
|
|
add_index_stringl(return_value, i, data, data_len, should_copy);
|
|
should_copy=1;
|
|
}
|
|
|
|
if (result_type & PGSQL_ASSOC) {
|
|
field_name = PQfname(pgsql_result,i);
|
|
add_assoc_stringl(return_value, field_name, data, data_len, should_copy);
|
|
}
|
|
} else {
|
|
/* NULL field, don't set it */
|
|
/* add_get_index_stringl(return_value, i, empty_string, 0, (void **) &pval_ptr); */
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* {{{ proto array pg_fetchrow(int result, int row)
|
|
Get a row as an enumerated array */
|
|
PHP_FUNCTION(pg_fetch_row)
|
|
{
|
|
php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAM_PASSTHRU, PGSQL_NUM);
|
|
}
|
|
/* }}} */
|
|
|
|
/* ?? This is a rather odd function - why not just point pg_fetcharray() directly at fetch_hash ? -RL */
|
|
/* {{{ proto array pg_fetch_array(int result, int row [, int result_type])
|
|
Fetch a row as an array */
|
|
PHP_FUNCTION(pg_fetch_array)
|
|
{
|
|
php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto object pg_fetch_object(int result, int row [, int result_type])
|
|
Fetch a row as an object */
|
|
PHP_FUNCTION(pg_fetch_object)
|
|
{
|
|
php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
|
|
if (return_value->type==IS_ARRAY) {
|
|
return_value->type=IS_OBJECT;
|
|
return_value->value.obj.properties = return_value->value.ht;
|
|
return_value->value.obj.ce = &zend_standard_class_def;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
#define PHP_PG_DATA_LENGTH 1
|
|
#define PHP_PG_DATA_ISNULL 2
|
|
|
|
void php_pgsql_data_info(INTERNAL_FUNCTION_PARAMETERS, int entry_type)
|
|
{
|
|
pval *result,*row,*field;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type,field_offset;
|
|
|
|
if (ARG_COUNT(ht)!=3 || getParameters(ht, 3, &result, &row, &field)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
pgsql_result = pg_result->result;
|
|
|
|
convert_to_long(row);
|
|
if (row->value.lval<0 || row->value.lval>=PQntuples(pgsql_result)) {
|
|
php_error(E_WARNING,"Unable to jump to row %d on PostgresSQL result index %d",row->value.lval,result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
switch(field->type) {
|
|
case IS_STRING:
|
|
field_offset = PQfnumber(pgsql_result,field->value.str.val);
|
|
break;
|
|
default:
|
|
convert_to_long(field);
|
|
field_offset = field->value.lval;
|
|
break;
|
|
}
|
|
if (field_offset<0 || field_offset>=PQnfields(pgsql_result)) {
|
|
php_error(E_WARNING,"Bad column offset specified");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
switch (entry_type) {
|
|
case PHP_PG_DATA_LENGTH:
|
|
return_value->value.lval = PQgetlength(pgsql_result,row->value.lval,field_offset);
|
|
break;
|
|
case PHP_PG_DATA_ISNULL:
|
|
return_value->value.lval = PQgetisnull(pgsql_result,row->value.lval,field_offset);
|
|
break;
|
|
}
|
|
return_value->type = IS_LONG;
|
|
}
|
|
|
|
/* {{{ proto int pg_fieldprtlen(int result, int row, mixed field_name_or_number)
|
|
Returns the printed length */
|
|
PHP_FUNCTION(pg_data_length)
|
|
{
|
|
php_pgsql_data_info(INTERNAL_FUNCTION_PARAM_PASSTHRU, PHP_PG_DATA_LENGTH);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_fieldisnull(int result, int row, mixed field_name_or_number)
|
|
Test if a field is NULL */
|
|
PHP_FUNCTION(pg_data_isnull)
|
|
{
|
|
php_pgsql_data_info(INTERNAL_FUNCTION_PARAM_PASSTHRU, PHP_PG_DATA_ISNULL);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_freeresult(int result)
|
|
Free result memory */
|
|
PHP_FUNCTION(pg_free_result)
|
|
{
|
|
pval *result;
|
|
pgsql_result_handle *pg_result;
|
|
int type;
|
|
|
|
if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &result)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
if (result->value.lval==0) {
|
|
RETURN_FALSE;
|
|
}
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
zend_list_delete(result->value.lval);
|
|
RETURN_TRUE;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_getlastoid(int result)
|
|
Returns the last object identifier */
|
|
PHP_FUNCTION(pg_last_oid)
|
|
{
|
|
pval *result;
|
|
PGresult *pgsql_result;
|
|
pgsql_result_handle *pg_result;
|
|
int type;
|
|
|
|
if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &result)==FAILURE) {
|
|
WRONG_PARAM_COUNT;
|
|
}
|
|
|
|
convert_to_long(result);
|
|
pg_result = (pgsql_result_handle *) zend_list_find(result->value.lval,&type);
|
|
|
|
if (type!=le_result) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL result index",result->value.lval);
|
|
RETURN_FALSE;
|
|
}
|
|
pgsql_result = pg_result->result;
|
|
return_value->value.str.val = (char *) PQoidStatus(pgsql_result);
|
|
if (return_value->value.str.val) {
|
|
return_value->value.str.len = strlen(return_value->value.str.val);
|
|
return_value->value.str.val = estrndup(return_value->value.str.val, return_value->value.str.len);
|
|
return_value->type = IS_STRING;
|
|
} else {
|
|
return_value->value.str.val = empty_string;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_locreate(int connection)
|
|
Create a large object */
|
|
PHP_FUNCTION(pg_lo_create)
|
|
{
|
|
pval *pgsql_link;
|
|
PGconn *pgsql;
|
|
Oid pgsql_oid;
|
|
int id, type;
|
|
PGLS_FETCH();
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 0:
|
|
id = PGG(default_link);
|
|
break;
|
|
case 1:
|
|
if (getParameters(ht, 1, &pgsql_link)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_link);
|
|
id = pgsql_link->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (PGconn *) zend_list_find(id,&type);
|
|
if (type!=le_link && type!=le_plink) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL link index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
/* XXX: Archive modes not supported until I get some more data. Don't think anybody's
|
|
using it anyway. I believe it's also somehow related to the 'time travel' feature of
|
|
PostgreSQL, that's on the list of features to be removed... Create modes not supported.
|
|
What's the use of an object that can be only written to, but not read from, and vice
|
|
versa? Beats me... And the access type (r/w) must be specified again when opening
|
|
the object, probably (?) overrides this. (Jouni)
|
|
*/
|
|
|
|
if ((pgsql_oid=lo_creat(pgsql, INV_READ|INV_WRITE))==0) {
|
|
php_error(E_WARNING,"Unable to create PostgresSQL large object");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
return_value->value.lval = pgsql_oid;
|
|
return_value->type = IS_LONG;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void pg_lounlink([int connection, ] int large_obj_id)
|
|
Delete a large object */
|
|
PHP_FUNCTION(pg_lo_unlink)
|
|
{
|
|
pval *pgsql_link, *oid;
|
|
PGconn *pgsql;
|
|
Oid pgsql_oid;
|
|
int id, type;
|
|
PGLS_FETCH();
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 1:
|
|
if (getParameters(ht, 1, &oid)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(oid);
|
|
pgsql_oid = oid->value.lval;
|
|
id = PGG(default_link);
|
|
break;
|
|
case 2:
|
|
if (getParameters(ht, 2, &pgsql_link, &oid)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_link);
|
|
id = pgsql_link->value.lval;
|
|
convert_to_long(oid);
|
|
pgsql_oid = oid->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (PGconn *) zend_list_find(id,&type);
|
|
if (type!=le_link && type!=le_plink) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL link index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (lo_unlink(pgsql, pgsql_oid)==-1) {
|
|
php_error(E_WARNING,"Unable to delete PostgresSQL large object %d", (int) pgsql_oid);
|
|
RETURN_FALSE;
|
|
}
|
|
RETURN_TRUE;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_loopen([int connection,] int objoid, string mode)
|
|
Open a large object and return fd */
|
|
PHP_FUNCTION(pg_lo_open)
|
|
{
|
|
pval *pgsql_link, *oid, *mode;
|
|
PGconn *pgsql;
|
|
Oid pgsql_oid;
|
|
int id, type, pgsql_mode=0, pgsql_lofd;
|
|
int create=0;
|
|
char *mode_string=NULL;
|
|
pgLofp *pgsql_lofp;
|
|
PGLS_FETCH();
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 2:
|
|
if (getParameters(ht, 2, &oid, &mode)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(oid);
|
|
pgsql_oid = oid->value.lval;
|
|
convert_to_string(mode);
|
|
mode_string = mode->value.str.val;
|
|
id = PGG(default_link);
|
|
break;
|
|
case 3:
|
|
if (getParameters(ht, 3, &pgsql_link, &oid, &mode)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_link);
|
|
id = pgsql_link->value.lval;
|
|
convert_to_long(oid);
|
|
pgsql_oid = oid->value.lval;
|
|
convert_to_string(mode);
|
|
mode_string = mode->value.str.val;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (PGconn *) zend_list_find(id,&type);
|
|
if (type!=le_link && type!=le_plink) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL link index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
/* r/w/+ is little bit more PHP-like than INV_READ/INV_WRITE and a lot of
|
|
faster to type. Unfortunately, doesn't behave the same way as fopen()...
|
|
(Jouni)
|
|
*/
|
|
|
|
if (strchr(mode_string, 'r')==mode_string) {
|
|
pgsql_mode |= INV_READ;
|
|
if (strchr(mode_string, '+')==mode_string+1) {
|
|
pgsql_mode |= INV_WRITE;
|
|
}
|
|
}
|
|
if (strchr(mode_string, 'w')==mode_string) {
|
|
pgsql_mode |= INV_WRITE;
|
|
create = 1;
|
|
if (strchr(mode_string, '+')==mode_string+1) {
|
|
pgsql_mode |= INV_READ;
|
|
}
|
|
}
|
|
|
|
|
|
pgsql_lofp = (pgLofp *) emalloc(sizeof(pgLofp));
|
|
|
|
if ((pgsql_lofd=lo_open(pgsql, pgsql_oid, pgsql_mode))==-1) {
|
|
if (create) {
|
|
if ((pgsql_oid=lo_creat(pgsql, INV_READ|INV_WRITE))==0) {
|
|
efree(pgsql_lofp);
|
|
php_error(E_WARNING,"Unable to create PostgresSQL large object");
|
|
RETURN_FALSE;
|
|
} else {
|
|
if ((pgsql_lofd=lo_open(pgsql, pgsql_oid, pgsql_mode))==-1) {
|
|
if (lo_unlink(pgsql, pgsql_oid)==-1) {
|
|
efree(pgsql_lofp);
|
|
php_error(E_WARNING,"Something's really messed up!!! Your database is badly corrupted in a way NOT related to PHP.");
|
|
RETURN_FALSE;
|
|
}
|
|
efree(pgsql_lofp);
|
|
php_error(E_WARNING,"Unable to open PostgresSQL large object");
|
|
RETURN_FALSE;
|
|
} else {
|
|
pgsql_lofp->conn = pgsql;
|
|
pgsql_lofp->lofd = pgsql_lofd;
|
|
return_value->value.lval = zend_list_insert(pgsql_lofp, le_lofp);
|
|
return_value->type = IS_LONG;
|
|
}
|
|
}
|
|
} else {
|
|
efree(pgsql_lofp);
|
|
php_error(E_WARNING,"Unable to open PostgresSQL large object");
|
|
RETURN_FALSE;
|
|
}
|
|
} else {
|
|
pgsql_lofp->conn = pgsql;
|
|
pgsql_lofp->lofd = pgsql_lofd;
|
|
return_value->value.lval = zend_list_insert(pgsql_lofp, le_lofp);
|
|
return_value->type = IS_LONG;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void pg_loclose(int fd)
|
|
Close a large object */
|
|
PHP_FUNCTION(pg_lo_close)
|
|
{
|
|
pval *pgsql_lofp;
|
|
int id, type;
|
|
pgLofp *pgsql;
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 1:
|
|
if (getParameters(ht, 1, &pgsql_lofp)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_lofp);
|
|
id = pgsql_lofp->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (pgLofp *) zend_list_find(id,&type);
|
|
if (type!=le_lofp) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL large object index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (lo_close((PGconn *)pgsql->conn, pgsql->lofd)<0) {
|
|
php_error(E_WARNING,"Unable to close PostgresSQL large object descriptor %d", pgsql->lofd);
|
|
RETVAL_FALSE;
|
|
} else {
|
|
RETVAL_TRUE;
|
|
}
|
|
zend_list_delete(id);
|
|
return;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string pg_loread(int fd, int len)
|
|
Read a large object */
|
|
PHP_FUNCTION(pg_lo_read)
|
|
{
|
|
pval *pgsql_id, *len;
|
|
int id, buf_len, type, nbytes;
|
|
char *buf;
|
|
pgLofp *pgsql;
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 2:
|
|
if (getParameters(ht, 2, &pgsql_id, &len)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_id);
|
|
id = pgsql_id->value.lval;
|
|
convert_to_long(len);
|
|
buf_len = len->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (pgLofp *) zend_list_find(id,&type);
|
|
if (type!=le_lofp) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL large object index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
buf = (char *) emalloc(sizeof(char)*(buf_len+1));
|
|
if ((nbytes = lo_read((PGconn *)pgsql->conn, pgsql->lofd, buf, buf_len))<0) {
|
|
efree(buf);
|
|
RETURN_FALSE;
|
|
}
|
|
return_value->value.str.val = buf;
|
|
return_value->value.str.len = nbytes;
|
|
return_value->value.str.val[nbytes] = 0;
|
|
return_value->type = IS_STRING;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int pg_lowrite(int fd, string buf)
|
|
Write a large object */
|
|
PHP_FUNCTION(pg_lo_write)
|
|
{
|
|
pval *pgsql_id, *str;
|
|
int id, buf_len, nbytes, type;
|
|
char *buf;
|
|
pgLofp *pgsql;
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 2:
|
|
if (getParameters(ht, 2, &pgsql_id, &str)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_id);
|
|
id = pgsql_id->value.lval;
|
|
convert_to_string(str);
|
|
buf = str->value.str.val;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (pgLofp *) zend_list_find(id,&type);
|
|
if (type!=le_lofp) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL large object index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
buf_len = str->value.str.len;
|
|
if ((nbytes = lo_write((PGconn *)pgsql->conn, pgsql->lofd, buf, buf_len))==-1) {
|
|
RETURN_FALSE;
|
|
}
|
|
return_value->value.lval = nbytes;
|
|
return_value->type = IS_LONG;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void pg_loreadall(int fd)
|
|
Read a large object and send straight to browser */
|
|
PHP_FUNCTION(pg_lo_readall)
|
|
{
|
|
pval *pgsql_id;
|
|
int i, id, tbytes, type;
|
|
volatile int nbytes;
|
|
char buf[8192];
|
|
pgLofp *pgsql;
|
|
int output=1;
|
|
|
|
switch(ARG_COUNT(ht)) {
|
|
case 1:
|
|
if (getParameters(ht, 1, &pgsql_id)==FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
convert_to_long(pgsql_id);
|
|
id = pgsql_id->value.lval;
|
|
break;
|
|
default:
|
|
WRONG_PARAM_COUNT;
|
|
break;
|
|
}
|
|
|
|
pgsql = (pgLofp *) zend_list_find(id,&type);
|
|
if (type!=le_lofp) {
|
|
php_error(E_WARNING,"%d is not a PostgresSQL large object index",id);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
tbytes = 0;
|
|
while ((nbytes = lo_read((PGconn *)pgsql->conn, pgsql->lofd, buf, 8192))>0) {
|
|
for(i=0; i<nbytes; i++) {
|
|
if (output) { (void) PUTC(buf[i]); }
|
|
}
|
|
tbytes += i;
|
|
}
|
|
return_value->value.lval = tbytes;
|
|
return_value->type = IS_LONG;
|
|
}
|
|
/* }}} */
|
|
|
|
#endif
|
|
|
|
/*
|
|
* Local variables:
|
|
* tab-width: 4
|
|
* c-basic-offset: 4
|
|
* End:
|
|
*/
|