1 |
6daefa8c
|
Petr Lukašík
|
<?php
|
2 |
|
|
|
3 |
|
|
/**
|
4 |
|
|
* PostgreSQL 8.0 support
|
5 |
|
|
*
|
6 |
|
|
* $Id: Postgres80.php,v 1.18 2005/09/07 08:09:21 chriskl Exp $
|
7 |
|
|
*/
|
8 |
|
|
|
9 |
|
|
include_once('./classes/database/Postgres74.php');
|
10 |
|
|
|
11 |
|
|
class Postgres80 extends Postgres74 {
|
12 |
|
|
|
13 |
|
|
var $major_version = 8.0;
|
14 |
|
|
|
15 |
|
|
// List of all legal privileges that can be applied to different types
|
16 |
|
|
// of objects.
|
17 |
|
|
var $privlist = array(
|
18 |
|
|
'table' => array('SELECT', 'INSERT', 'UPDATE', 'DELETE', 'RULE', 'REFERENCES', 'TRIGGER', 'ALL PRIVILEGES'),
|
19 |
|
|
'view' => array('SELECT', 'INSERT', 'UPDATE', 'DELETE', 'RULE', 'REFERENCES', 'TRIGGER', 'ALL PRIVILEGES'),
|
20 |
|
|
'sequence' => array('SELECT', 'UPDATE', 'ALL PRIVILEGES'),
|
21 |
|
|
'database' => array('CREATE', 'TEMPORARY', 'ALL PRIVILEGES'),
|
22 |
|
|
'function' => array('EXECUTE', 'ALL PRIVILEGES'),
|
23 |
|
|
'language' => array('USAGE', 'ALL PRIVILEGES'),
|
24 |
|
|
'schema' => array('CREATE', 'USAGE', 'ALL PRIVILEGES'),
|
25 |
|
|
'tablespace' => array('CREATE', 'ALL PRIVILEGES')
|
26 |
|
|
);
|
27 |
|
|
|
28 |
|
|
// Last oid assigned to a system object
|
29 |
|
|
var $_lastSystemOID = 17228;
|
30 |
|
|
|
31 |
|
|
/**
|
32 |
|
|
* Constructor
|
33 |
|
|
* @param $conn The database connection
|
34 |
|
|
*/
|
35 |
|
|
function Postgres80($conn) {
|
36 |
|
|
$this->Postgres74($conn);
|
37 |
|
|
}
|
38 |
|
|
|
39 |
|
|
// Help functions
|
40 |
|
|
|
41 |
|
|
function getHelpPages() {
|
42 |
|
|
include_once('./help/PostgresDoc80.php');
|
43 |
|
|
return $this->help_page;
|
44 |
|
|
}
|
45 |
|
|
|
46 |
|
|
// Database functions
|
47 |
|
|
|
48 |
|
|
/**
|
49 |
|
|
* Return all database available on the server
|
50 |
|
|
* @return A list of databases, sorted alphabetically
|
51 |
|
|
*/
|
52 |
|
|
function getDatabases($currentdatabase = NULL) {
|
53 |
|
|
global $conf, $misc;
|
54 |
|
|
|
55 |
|
|
$server_info = $misc->getServerInfo();
|
56 |
|
|
|
57 |
|
|
if (isset($conf['owned_only']) && $conf['owned_only'] && !$this->isSuperUser($server_info['username'])) {
|
58 |
|
|
$username = $server_info['username'];
|
59 |
|
|
$this->clean($username);
|
60 |
|
|
$clause = " AND pu.usename='{$username}'";
|
61 |
|
|
}
|
62 |
|
|
else $clause = '';
|
63 |
|
|
|
64 |
|
|
if ($currentdatabase != NULL)
|
65 |
|
|
$orderby = "ORDER BY pdb.datname = '{$currentdatabase}' DESC, pdb.datname";
|
66 |
|
|
else
|
67 |
|
|
$orderby = "ORDER BY pdb.datname";
|
68 |
|
|
|
69 |
|
|
if (!$conf['show_system'])
|
70 |
|
|
$where = ' AND NOT pdb.datistemplate';
|
71 |
|
|
else
|
72 |
|
|
$where = ' AND pdb.datallowconn';
|
73 |
|
|
|
74 |
|
|
$sql = "SELECT pdb.datname AS datname, pu.usename AS datowner, pg_encoding_to_char(encoding) AS datencoding,
|
75 |
|
|
(SELECT description FROM pg_description pd WHERE pdb.oid=pd.objoid) AS datcomment,
|
76 |
|
|
(SELECT spcname FROM pg_catalog.pg_tablespace pt WHERE pt.oid=pdb.dattablespace) AS tablespace
|
77 |
|
|
FROM pg_database pdb, pg_user pu
|
78 |
|
|
WHERE pdb.datdba = pu.usesysid
|
79 |
|
|
{$where}
|
80 |
|
|
{$clause}
|
81 |
|
|
{$orderby}";
|
82 |
|
|
|
83 |
|
|
return $this->selectSet($sql);
|
84 |
|
|
}
|
85 |
|
|
|
86 |
|
|
/**
|
87 |
|
|
* Alters a database
|
88 |
|
|
* the multiple return vals are for postgres 8+ which support more functionality in alter database
|
89 |
|
|
* @param $dbName The name of the database
|
90 |
|
|
* @param $newName new name for the database
|
91 |
|
|
* @param $newOwner The new owner for the database
|
92 |
|
|
* @return 0 success
|
93 |
|
|
* @return -1 transaction error
|
94 |
|
|
* @return -2 owner error
|
95 |
|
|
* @return -3 rename error
|
96 |
|
|
*/
|
97 |
|
|
function alterDatabase($dbName, $newName, $newOwner = '')
|
98 |
|
|
{
|
99 |
|
|
$this->clean($dbName);
|
100 |
|
|
$this->clean($newName);
|
101 |
|
|
$this->clean($newOwner);
|
102 |
|
|
|
103 |
|
|
$status = $this->beginTransaction();
|
104 |
|
|
if ($status != 0) {
|
105 |
|
|
$this->rollbackTransaction();
|
106 |
|
|
return -1;
|
107 |
|
|
}
|
108 |
|
|
|
109 |
|
|
if ($dbName != $newName) {
|
110 |
|
|
$status = $this->alterDatabaseRename($dbName, $newName);
|
111 |
|
|
if ($status != 0) {
|
112 |
|
|
$this->rollbackTransaction();
|
113 |
|
|
return -3;
|
114 |
|
|
}
|
115 |
|
|
}
|
116 |
|
|
|
117 |
|
|
$status = $this->alterDatabaseOwner($newName, $newOwner);
|
118 |
|
|
if ($status != 0) {
|
119 |
|
|
$this->rollbackTransaction();
|
120 |
|
|
return -2;
|
121 |
|
|
}
|
122 |
|
|
return $this->endTransaction();
|
123 |
|
|
}
|
124 |
|
|
/**
|
125 |
|
|
* Changes ownership of a database
|
126 |
|
|
* This can only be done by a superuser or the owner of the database
|
127 |
|
|
* @param string $dbName database to change ownership of
|
128 |
|
|
* @param string $newOwner user that will own the database
|
129 |
|
|
* @return int 0 on success
|
130 |
|
|
*/
|
131 |
|
|
function alterDatabaseOwner($dbName, $newOwner) {
|
132 |
|
|
$this->clean($dbName);
|
133 |
|
|
$this->clean($newOwner);
|
134 |
|
|
|
135 |
|
|
$sql = "ALTER DATABASE \"{$dbName}\" OWNER TO \"{$newOwner}\"";
|
136 |
|
|
return $this->execute($sql);
|
137 |
|
|
}
|
138 |
|
|
|
139 |
|
|
// Table functions
|
140 |
|
|
|
141 |
|
|
/**
|
142 |
|
|
* Return all tables in current database (and schema)
|
143 |
|
|
* @param $all True to fetch all tables, false for just in current schema
|
144 |
|
|
* @return All tables, sorted alphabetically
|
145 |
|
|
*/
|
146 |
|
|
function getTables($all = false) {
|
147 |
|
|
if ($all) {
|
148 |
|
|
// Exclude pg_catalog and information_schema tables
|
149 |
|
|
$sql = "SELECT schemaname AS nspname, tablename AS relname, tableowner AS relowner
|
150 |
|
|
FROM pg_catalog.pg_tables
|
151 |
|
|
WHERE schemaname NOT IN ('pg_catalog', 'information_schema', 'pg_toast')
|
152 |
|
|
ORDER BY schemaname, tablename";
|
153 |
|
|
} else {
|
154 |
|
|
$sql = "SELECT c.relname, pg_catalog.pg_get_userbyid(c.relowner) AS relowner,
|
155 |
|
|
pg_catalog.obj_description(c.oid, 'pg_class') AS relcomment,
|
156 |
|
|
reltuples::integer,
|
157 |
|
|
(SELECT spcname FROM pg_catalog.pg_tablespace pt WHERE pt.oid=c.reltablespace) AS tablespace
|
158 |
|
|
FROM pg_catalog.pg_class c
|
159 |
|
|
LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
|
160 |
|
|
WHERE c.relkind = 'r'
|
161 |
|
|
AND nspname='{$this->_schema}'
|
162 |
|
|
ORDER BY c.relname";
|
163 |
|
|
}
|
164 |
|
|
|
165 |
|
|
return $this->selectSet($sql);
|
166 |
|
|
}
|
167 |
|
|
|
168 |
|
|
/**
|
169 |
|
|
* Returns table information
|
170 |
|
|
* @param $table The name of the table
|
171 |
|
|
* @return A recordset
|
172 |
|
|
*/
|
173 |
|
|
function getTable($table) {
|
174 |
|
|
$this->clean($table);
|
175 |
|
|
|
176 |
|
|
$sql = "
|
177 |
|
|
SELECT
|
178 |
|
|
c.relname, u.usename AS relowner,
|
179 |
|
|
pg_catalog.obj_description(c.oid, 'pg_class') AS relcomment,
|
180 |
|
|
(SELECT spcname FROM pg_catalog.pg_tablespace pt WHERE pt.oid=c.reltablespace) AS tablespace
|
181 |
|
|
FROM pg_catalog.pg_class c
|
182 |
|
|
LEFT JOIN pg_catalog.pg_user u ON u.usesysid = c.relowner
|
183 |
|
|
LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
|
184 |
|
|
WHERE c.relkind = 'r'
|
185 |
|
|
AND n.nspname = '{$this->_schema}'
|
186 |
|
|
AND c.relname = '{$table}'";
|
187 |
|
|
|
188 |
|
|
return $this->selectSet($sql);
|
189 |
|
|
}
|
190 |
|
|
|
191 |
|
|
/**
|
192 |
|
|
* Alters a column in a table
|
193 |
|
|
* @param $table The table in which the column resides
|
194 |
|
|
* @param $column The column to alter
|
195 |
|
|
* @param $name The new name for the column
|
196 |
|
|
* @param $notnull (boolean) True if not null, false otherwise
|
197 |
|
|
* @param $oldnotnull (boolean) True if column is already not null, false otherwise
|
198 |
|
|
* @param $default The new default for the column
|
199 |
|
|
* @param $olddefault The old default for the column
|
200 |
|
|
* @param $type The new type for the column
|
201 |
|
|
* @param $array True if array type, false otherwise
|
202 |
|
|
* @param $length The optional size of the column (ie. 30 for varchar(30))
|
203 |
|
|
* @param $oldtype The old type for the column
|
204 |
|
|
* @param $comment Comment for the column
|
205 |
|
|
* @return 0 success
|
206 |
|
|
* @return -1 batch alteration failed
|
207 |
|
|
* @return -3 rename column error
|
208 |
|
|
* @return -4 comment error
|
209 |
|
|
* @return -6 transaction error
|
210 |
|
|
*/
|
211 |
|
|
function alterColumn($table, $column, $name, $notnull, $oldnotnull, $default, $olddefault,
|
212 |
|
|
$type, $length, $array, $oldtype, $comment) {
|
213 |
|
|
$this->fieldClean($table);
|
214 |
|
|
$this->fieldClean($column);
|
215 |
|
|
$this->clean($comment);
|
216 |
|
|
|
217 |
|
|
// Initialise an empty SQL string
|
218 |
|
|
$sql = '';
|
219 |
|
|
|
220 |
|
|
// Create the command for changing nullability
|
221 |
|
|
if ($notnull != $oldnotnull) {
|
222 |
|
|
$sql .= "ALTER TABLE \"{$table}\" ALTER COLUMN \"{$column}\" " . (($notnull) ? 'SET' : 'DROP') . " NOT NULL";
|
223 |
|
|
}
|
224 |
|
|
|
225 |
|
|
// Add default, if it has changed
|
226 |
|
|
if ($default != $olddefault) {
|
227 |
|
|
if ($default == '') {
|
228 |
|
|
if ($sql == '') $sql = "ALTER TABLE \"{$table}\" ";
|
229 |
|
|
else $sql .= ", ";
|
230 |
|
|
$sql .= "ALTER COLUMN \"{$column}\" DROP DEFAULT";
|
231 |
|
|
}
|
232 |
|
|
else {
|
233 |
|
|
if ($sql == '') $sql = "ALTER TABLE \"{$table}\" ";
|
234 |
|
|
else $sql .= ", ";
|
235 |
|
|
$sql .= "ALTER COLUMN \"{$column}\" SET DEFAULT {$default}";
|
236 |
|
|
}
|
237 |
|
|
}
|
238 |
|
|
|
239 |
|
|
// Add type, if it has changed
|
240 |
|
|
if ($length == '')
|
241 |
|
|
$ftype = $type;
|
242 |
|
|
else {
|
243 |
|
|
switch ($type) {
|
244 |
|
|
// Have to account for weird placing of length for with/without
|
245 |
|
|
// time zone types
|
246 |
|
|
case 'timestamp with time zone':
|
247 |
|
|
case 'timestamp without time zone':
|
248 |
|
|
$qual = substr($type, 9);
|
249 |
|
|
$ftype = "timestamp({$length}){$qual}";
|
250 |
|
|
break;
|
251 |
|
|
case 'time with time zone':
|
252 |
|
|
case 'time without time zone':
|
253 |
|
|
$qual = substr($type, 4);
|
254 |
|
|
$ftype = "time({$length}){$qual}";
|
255 |
|
|
break;
|
256 |
|
|
default:
|
257 |
|
|
$ftype = "{$type}({$length})";
|
258 |
|
|
}
|
259 |
|
|
}
|
260 |
|
|
|
261 |
|
|
// Add array qualifier, if requested
|
262 |
|
|
if ($array) $ftype .= '[]';
|
263 |
|
|
|
264 |
|
|
if ($ftype != $oldtype) {
|
265 |
|
|
if ($sql == '') $sql = "ALTER TABLE \"{$table}\" ";
|
266 |
|
|
else $sql .= ", ";
|
267 |
|
|
$sql .= "ALTER COLUMN \"{$column}\" TYPE {$ftype}";
|
268 |
|
|
}
|
269 |
|
|
|
270 |
|
|
// Begin transaction
|
271 |
|
|
$status = $this->beginTransaction();
|
272 |
|
|
if ($status != 0) {
|
273 |
|
|
$this->rollbackTransaction();
|
274 |
|
|
return -6;
|
275 |
|
|
}
|
276 |
|
|
|
277 |
|
|
// Attempt to process the batch alteration, if anything has been changed
|
278 |
|
|
if ($sql != '') {
|
279 |
|
|
$status = $this->execute($sql);
|
280 |
|
|
if ($status != 0) {
|
281 |
|
|
$this->rollbackTransaction();
|
282 |
|
|
return -1;
|
283 |
|
|
}
|
284 |
|
|
}
|
285 |
|
|
|
286 |
|
|
// Update the comment on the column
|
287 |
|
|
$status = $this->setComment('COLUMN', $column, $table, $comment);
|
288 |
|
|
if ($status != 0) {
|
289 |
|
|
$this->rollbackTransaction();
|
290 |
|
|
return -4;
|
291 |
|
|
}
|
292 |
|
|
|
293 |
|
|
// Rename the column, if it has been changed
|
294 |
|
|
if ($column != $name) {
|
295 |
|
|
$status = $this->renameColumn($table, $column, $name);
|
296 |
|
|
if ($status != 0) {
|
297 |
|
|
$this->rollbackTransaction();
|
298 |
|
|
return -3;
|
299 |
|
|
}
|
300 |
|
|
}
|
301 |
|
|
|
302 |
|
|
return $this->endTransaction();
|
303 |
|
|
}
|
304 |
|
|
|
305 |
|
|
// Sequence functions
|
306 |
|
|
|
307 |
|
|
/**
|
308 |
|
|
* Returns all sequences in the current database
|
309 |
|
|
* @return A recordset
|
310 |
|
|
*/
|
311 |
|
|
function getSequences($all = false) {
|
312 |
|
|
if ($all) {
|
313 |
|
|
// Exclude pg_catalog and information_schema tables
|
314 |
|
|
$sql = "SELECT n.nspname, c.relname AS seqname, u.usename AS seqowner
|
315 |
|
|
FROM pg_catalog.pg_class c, pg_catalog.pg_user u, pg_catalog.pg_namespace n
|
316 |
|
|
WHERE c.relowner=u.usesysid AND c.relnamespace=n.oid
|
317 |
|
|
AND c.relkind = 'S'
|
318 |
|
|
AND n.nspname NOT IN ('pg_catalog', 'information_schema', 'pg_toast')
|
319 |
|
|
ORDER BY nspname, seqname";
|
320 |
|
|
} else {
|
321 |
|
|
$sql = "SELECT c.relname AS seqname, u.usename AS seqowner, pg_catalog.obj_description(c.oid, 'pg_class') AS seqcomment,
|
322 |
|
|
(SELECT spcname FROM pg_catalog.pg_tablespace pt WHERE pt.oid=c.reltablespace) AS tablespace
|
323 |
|
|
FROM pg_catalog.pg_class c, pg_catalog.pg_user u, pg_catalog.pg_namespace n
|
324 |
|
|
WHERE c.relowner=u.usesysid AND c.relnamespace=n.oid
|
325 |
|
|
AND c.relkind = 'S' AND n.nspname='{$this->_schema}' ORDER BY seqname";
|
326 |
|
|
}
|
327 |
|
|
|
328 |
|
|
return $this->selectSet( $sql );
|
329 |
|
|
}
|
330 |
|
|
|
331 |
|
|
// Tablespace functions
|
332 |
|
|
|
333 |
|
|
/**
|
334 |
|
|
* Retrieves information for all tablespaces
|
335 |
|
|
* @param $all Include all tablespaces (necessary when moving objects back to the default space)
|
336 |
|
|
* @return A recordset
|
337 |
|
|
*/
|
338 |
|
|
function getTablespaces($all = false) {
|
339 |
|
|
global $conf;
|
340 |
|
|
|
341 |
|
|
$sql = "SELECT spcname, pg_catalog.pg_get_userbyid(spcowner) AS spcowner, spclocation
|
342 |
|
|
FROM pg_catalog.pg_tablespace";
|
343 |
|
|
|
344 |
|
|
if (!$conf['show_system'] && !$all) {
|
345 |
|
|
$sql .= " WHERE spcname NOT LIKE 'pg\\\\_%'";
|
346 |
|
|
}
|
347 |
|
|
|
348 |
|
|
$sql .= " ORDER BY spcname";
|
349 |
|
|
|
350 |
|
|
return $this->selectSet($sql);
|
351 |
|
|
}
|
352 |
|
|
|
353 |
|
|
/**
|
354 |
|
|
* Retrieves a tablespace's information
|
355 |
|
|
* @return A recordset
|
356 |
|
|
*/
|
357 |
|
|
function getTablespace($spcname) {
|
358 |
|
|
$this->clean($spcname);
|
359 |
|
|
|
360 |
|
|
$sql = "SELECT spcname, pg_catalog.pg_get_userbyid(spcowner) AS spcowner, spclocation
|
361 |
|
|
FROM pg_catalog.pg_tablespace WHERE spcname='{$spcname}'";
|
362 |
|
|
|
363 |
|
|
return $this->selectSet($sql);
|
364 |
|
|
}
|
365 |
|
|
|
366 |
|
|
/**
|
367 |
|
|
* Creates a tablespace
|
368 |
|
|
* @param $spcname The name of the tablespace to create
|
369 |
|
|
* @param $spcowner The owner of the tablespace. '' for current
|
370 |
|
|
* @param $spcloc The directory in which to create the tablespace
|
371 |
|
|
* @return 0 success
|
372 |
|
|
*/
|
373 |
|
|
function createTablespace($spcname, $spcowner, $spcloc) {
|
374 |
|
|
$this->fieldClean($spcname);
|
375 |
|
|
$this->clean($spcloc);
|
376 |
|
|
|
377 |
|
|
$sql = "CREATE TABLESPACE \"{$spcname}\"";
|
378 |
|
|
|
379 |
|
|
if ($spcowner != '') {
|
380 |
|
|
$this->fieldClean($spcowner);
|
381 |
|
|
$sql .= " OWNER \"{$spcowner}\"";
|
382 |
|
|
}
|
383 |
|
|
|
384 |
|
|
$sql .= " LOCATION '{$spcloc}'";
|
385 |
|
|
|
386 |
|
|
return $this->execute($sql);
|
387 |
|
|
}
|
388 |
|
|
|
389 |
|
|
/**
|
390 |
|
|
* Drops a tablespace
|
391 |
|
|
* @param $spcname The name of the domain to drop
|
392 |
|
|
* @return 0 success
|
393 |
|
|
*/
|
394 |
|
|
function dropTablespace($spcname) {
|
395 |
|
|
$this->fieldClean($spcname);
|
396 |
|
|
|
397 |
|
|
$sql = "DROP TABLESPACE \"{$spcname}\"";
|
398 |
|
|
|
399 |
|
|
return $this->execute($sql);
|
400 |
|
|
}
|
401 |
|
|
|
402 |
|
|
/**
|
403 |
|
|
* Alters a tablespace
|
404 |
|
|
* @param $spcname The name of the tablespace
|
405 |
|
|
* @param $name The new name for the tablespace
|
406 |
|
|
* @param $owner The new owner for the tablespace
|
407 |
|
|
* @return 0 success
|
408 |
|
|
* @return -1 transaction error
|
409 |
|
|
* @return -2 owner error
|
410 |
|
|
* @return -3 rename error
|
411 |
|
|
*/
|
412 |
|
|
function alterTablespace($spcname, $name, $owner) {
|
413 |
|
|
$this->fieldClean($spcname);
|
414 |
|
|
$this->fieldClean($name);
|
415 |
|
|
$this->fieldClean($owner);
|
416 |
|
|
|
417 |
|
|
// Begin transaction
|
418 |
|
|
$status = $this->beginTransaction();
|
419 |
|
|
if ($status != 0) return -1;
|
420 |
|
|
|
421 |
|
|
// Owner
|
422 |
|
|
$sql = "ALTER TABLESPACE \"{$spcname}\" OWNER TO \"{$owner}\"";
|
423 |
|
|
$status = $this->execute($sql);
|
424 |
|
|
if ($status != 0) {
|
425 |
|
|
$this->rollbackTransaction();
|
426 |
|
|
return -2;
|
427 |
|
|
}
|
428 |
|
|
|
429 |
|
|
// Rename (only if name has changed)
|
430 |
|
|
if ($name != $spcname) {
|
431 |
|
|
$sql = "ALTER TABLESPACE \"{$spcname}\" RENAME TO \"{$name}\"";
|
432 |
|
|
$status = $this->execute($sql);
|
433 |
|
|
if ($status != 0) {
|
434 |
|
|
$this->rollbackTransaction();
|
435 |
|
|
return -3;
|
436 |
|
|
}
|
437 |
|
|
}
|
438 |
|
|
|
439 |
|
|
return $this->endTransaction();
|
440 |
|
|
}
|
441 |
|
|
|
442 |
|
|
// Backend process signalling functions
|
443 |
|
|
|
444 |
|
|
/**
|
445 |
|
|
* Sends a cancel or kill command to a process
|
446 |
|
|
* @param $pid The ID of the backend process
|
447 |
|
|
* @param $signal 'CANCEL'
|
448 |
|
|
* @return 0 success
|
449 |
|
|
* @return -1 invalid signal type
|
450 |
|
|
*/
|
451 |
|
|
function sendSignal($pid, $signal) {
|
452 |
|
|
// Clean
|
453 |
|
|
$pid = (int)$pid;
|
454 |
|
|
|
455 |
|
|
if ($signal == 'CANCEL')
|
456 |
|
|
$sql = "SELECT pg_catalog.pg_cancel_backend({$pid}) AS val";
|
457 |
|
|
else
|
458 |
|
|
return -1;
|
459 |
|
|
|
460 |
|
|
// Execute the query
|
461 |
|
|
$val = $this->selectField($sql, 'val');
|
462 |
|
|
|
463 |
|
|
if ($val === -1) return -1;
|
464 |
|
|
elseif ($val == '1') return 0;
|
465 |
|
|
else return -1;
|
466 |
|
|
}
|
467 |
|
|
|
468 |
|
|
/**
|
469 |
|
|
* Returns all details for a particular function
|
470 |
|
|
* @param $func The name of the function to retrieve
|
471 |
|
|
* @return Function info
|
472 |
|
|
*/
|
473 |
|
|
function getFunction($function_oid) {
|
474 |
|
|
$this->clean($function_oid);
|
475 |
|
|
|
476 |
|
|
$sql = "SELECT
|
477 |
|
|
pc.oid AS prooid,
|
478 |
|
|
proname,
|
479 |
|
|
lanname as prolanguage,
|
480 |
|
|
pg_catalog.format_type(prorettype, NULL) as proresult,
|
481 |
|
|
prosrc,
|
482 |
|
|
probin,
|
483 |
|
|
proretset,
|
484 |
|
|
proisstrict,
|
485 |
|
|
provolatile,
|
486 |
|
|
prosecdef,
|
487 |
|
|
pg_catalog.oidvectortypes(pc.proargtypes) AS proarguments,
|
488 |
|
|
proargnames AS proargnames,
|
489 |
|
|
pg_catalog.obj_description(pc.oid, 'pg_proc') AS procomment
|
490 |
|
|
FROM
|
491 |
|
|
pg_catalog.pg_proc pc, pg_catalog.pg_language pl
|
492 |
|
|
WHERE
|
493 |
|
|
pc.oid = '{$function_oid}'::oid
|
494 |
|
|
AND pc.prolang = pl.oid
|
495 |
|
|
";
|
496 |
|
|
|
497 |
|
|
return $this->selectSet($sql);
|
498 |
|
|
}
|
499 |
|
|
|
500 |
|
|
// Capabilities
|
501 |
|
|
function hasAlterDatabaseOwner() { return true; }
|
502 |
|
|
function hasAlterColumnType() { return true; }
|
503 |
|
|
function hasTablespaces() { return true; }
|
504 |
|
|
function hasSignals() { return true; }
|
505 |
|
|
function hasNamedParams() { return true; }
|
506 |
|
|
|
507 |
|
|
}
|
508 |
|
|
|
509 |
|
|
?>
|