Projekt

Obecné

Profil

Stáhnout (14.2 KB) Statistiky
| Větev: | Tag: | Revize:
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
?>