33require_once DOL_DOCUMENT_ROOT.
'/core/db/DoliDB.class.php';
58 public $unescapeslashquot =
false;
62 public $standard_conforming_strings =
false;
88 if (!empty(
$conf->db->character_set)) {
89 $this->forcecharset =
$conf->db->character_set;
91 if (!empty(
$conf->db->dolibarr_main_db_collation)) {
92 $this->forcecollate =
$conf->db->dolibarr_main_db_collation;
95 $this->database_user = $user;
96 $this->database_host = $host;
97 $this->database_port = $port;
99 $this->transaction_opened = 0;
103 if (!function_exists(
"pg_connect")) {
104 $this->connected =
false;
106 $this->
error =
"Pgsql PHP functions are not available in this version of PHP";
107 dol_syslog(get_class($this).
"::DoliDBPgsql : Pgsql PHP functions are not available in this version of PHP", LOG_ERR);
112 $this->connected =
false;
114 $this->
error = $langs->trans(
"ErrorWrongHostParameter");
115 dol_syslog(get_class($this).
"::DoliDBPgsql : Erreur Connect, wrong host parameters", LOG_ERR);
121 $this->db = $this->
connect($host, $user, $pass, $name, $port);
124 $this->connected =
true;
128 $this->connected =
false;
130 $this->
error =
'Host, login or password incorrect';
131 dol_syslog(get_class($this).
"::DoliDBPgsql : Erreur Connect ".$this->
error.
'. Failed to connect to host='.$host.
' port='.$port.
' user='.$user, LOG_ERR);
135 if ($this->connected && $name) {
137 $this->database_selected =
true;
138 $this->database_name = $name;
141 $this->database_selected =
false;
142 $this->database_name =
'';
145 dol_syslog(get_class($this).
"::DoliDBPgsql : Erreur Select_db ".$this->
error, LOG_ERR);
149 $this->database_selected =
false;
167 if (preg_match(
'/^--\s\$Id/i', $line)) {
171 if (preg_match(
'/^#/i', $line) || preg_match(
'/^$/i', $line) || preg_match(
'/^--/i', $line)) {
177 $line = preg_replace(
'/GROUP_CONCAT/i',
'STRING_AGG', $line);
178 $line = preg_replace(
'/ SEPARATOR/i',
',', $line);
179 $line = preg_replace(
'/STRING_AGG\(([^,\)]+)\)/i',
'STRING_AGG(\\1, \',\')', $line);
180 $line = preg_replace(
'/STRING_AGG\(([^,]+),([^\)]+)\)/i',
'STRING_AGG(\\1::TEXT,\\2::TEXT)', $line);
183 if (
$type ==
'auto') {
184 if (preg_match(
'/ALTER TABLE/i', $line)) {
186 } elseif (preg_match(
'/CREATE TABLE/i', $line)) {
188 } elseif (preg_match(
'/DROP TABLE/i', $line)) {
193 $line = preg_replace(
'/ as signed\)/i',
' as integer)', $line);
195 if (
$type ==
'dml') {
198 $line = preg_replace(
'/\s/',
' ', $line);
201 if (preg_match(
'/(ISAM|innodb)/i', $line)) {
202 $line = preg_replace(
'/\)[\s\t]*type[\s\t]*=[\s\t]*(MyISAM|innodb).*;/i',
');', $line);
203 $line = preg_replace(
'/\)[\s\t]*engine[\s\t]*=[\s\t]*(MyISAM|innodb).*;/i',
');', $line);
204 $line = preg_replace(
'/,$/',
'', $line);
208 if (preg_match(
'/[\s\t\(]*(\w*)[\s\t]+int.*auto_increment/i', $line, $reg)) {
209 $newline = preg_replace(
'/([\s\t\(]*)([a-zA-Z_0-9]*)[\s\t]+int.*auto_increment[^,]*/i',
'\\1 \\2 SERIAL PRIMARY KEY', $line);
214 if (preg_match(
'/[\s\t\(]*(\w*)[\s\t]+bigint.*auto_increment/i', $line, $reg)) {
215 $newline = preg_replace(
'/([\s\t\(]*)([a-zA-Z_0-9]*)[\s\t]+bigint.*auto_increment[^,]*/i',
'\\1 \\2 BIGSERIAL PRIMARY KEY', $line);
221 $line = preg_replace(
'/tinyint\(?[0-9]*\)?/',
'smallint', $line);
222 $line = preg_replace(
'/tinyint/i',
'smallint', $line);
225 $line = preg_replace(
'/(int\w+|smallint|bigint)\s+unsigned/i',
'\\1', $line);
228 $line = preg_replace(
'/\w*blob/i',
'text', $line);
231 $line = preg_replace(
'/tinytext/i',
'text', $line);
232 $line = preg_replace(
'/mediumtext/i',
'text', $line);
233 $line = preg_replace(
'/longtext/i',
'text', $line);
235 $line = preg_replace(
'/text\([0-9]+\)/i',
'text', $line);
239 $line = preg_replace(
'/datetime not null/i',
'datetime', $line);
240 $line = preg_replace(
'/datetime/i',
'timestamp', $line);
243 $line = preg_replace(
'/^double/i',
'numeric', $line);
244 $line = preg_replace(
'/(\s*)double/i',
'\\1numeric', $line);
246 $line = preg_replace(
'/^float/i',
'numeric', $line);
247 $line = preg_replace(
'/(\s*)float/i',
'\\1numeric', $line);
251 $line = preg_replace(
'/(\s*)tms(\s*)timestamp/i',
'\\1tms timestamp without time zone DEFAULT now() NOT NULL', $line);
254 $line = preg_replace(
'/(\s*)DEFAULT(\s*)CURRENT_TIMESTAMP/i',
'\\1', $line);
257 $line = preg_replace(
'/(\s*)ON(\s*)UPDATE(\s*)CURRENT_TIMESTAMP/i',
'\\1', $line);
260 if (preg_match(
'/unique index\s*\((\w+\s*,\s*\w+)\)/i', $line)) {
261 $line = preg_replace(
'/unique index\s*\((\w+\s*,\s*\w+)\)/i',
'UNIQUE\(\\1\)', $line);
265 $line = preg_replace(
'/\sAFTER [a-z0-9_]+/i',
'', $line);
268 $line = preg_replace(
'/ALTER TABLE [a-z0-9_]+\s+DROP INDEX/i',
'DROP INDEX', $line);
271 if (preg_match(
'/ALTER TABLE ([a-z0-9_]+)\s+CHANGE(?: COLUMN)? ([a-z0-9_]+) ([a-z0-9_]+)(.*)$/i', $line, $reg)) {
272 $line =
"-- ".$line.
" replaced by --\n";
273 $line .=
"ALTER TABLE ".$reg[1].
" RENAME COLUMN ".$reg[2].
" TO ".$reg[3];
277 if (preg_match(
'/ALTER TABLE ([a-z0-9_]+)\s+MODIFY(?: COLUMN)? ([a-z0-9_]+) (.*)$/i', $line, $reg)) {
278 $line =
"-- ".$line.
" replaced by --\n";
280 $newreg3 = preg_replace(
'/ DEFAULT NULL/i',
'', $newreg3);
281 $newreg3 = preg_replace(
'/ NOT NULL/i',
'', $newreg3);
282 $newreg3 = preg_replace(
'/ NULL/i',
'', $newreg3);
283 $newreg3 = preg_replace(
'/ DEFAULT 0/i',
'', $newreg3);
284 $newreg3 = preg_replace(
'/ DEFAULT \'?[0-9a-zA-Z_@]*\'?/i',
'', $newreg3);
285 $line .=
"ALTER TABLE ".$reg[1].
" ALTER COLUMN ".$reg[2].
" TYPE ".$newreg3;
291 if (preg_match(
'/ALTER\s+TABLE\s*(.*)\s*ADD\s+PRIMARY\s+KEY\s*(.*)\s*\((.*)$/i', $line, $reg)) {
292 $line =
"-- ".$line.
" replaced by --\n";
293 $line .=
"ALTER TABLE ".$reg[1].
" ADD PRIMARY KEY (".$reg[3];
298 if (preg_match(
'/ALTER\s+TABLE\s*(.*)\s*DROP\s+PRIMARY\s+KEY\s*([^;]+)$/i', $line, $reg)) {
299 $line =
"-- ".$line.
" replaced by --\n";
300 $line .=
"ALTER TABLE ".$reg[1].
" DROP CONSTRAINT ".$reg[2];
305 if (preg_match(
'/ALTER\s+TABLE\s*(.*)\s*DROP\s+FOREIGN\s+KEY\s*(.*)$/i', $line, $reg)) {
306 $line =
"-- ".$line.
" replaced by --\n";
307 $line .=
"ALTER TABLE ".$reg[1].
" DROP CONSTRAINT ".$reg[2];
312 if (preg_match(
'/ALTER\s+TABLE\s+(.*)\s*ADD CONSTRAINT\s+(.*)\s*FOREIGN\s+KEY\s*(.*)$/i', $line, $reg)) {
313 $line = preg_replace(
'/;$/',
'', $line);
314 $line .=
" DEFERRABLE INITIALLY IMMEDIATE;";
319 if (preg_match(
'/ALTER\s+TABLE\s*(.*)\s*ADD\s+(UNIQUE INDEX|INDEX|UNIQUE)\s+(.*)\s*\(([\w,\s]+)\)/i', $line, $reg)) {
320 $fieldlist = $reg[4];
322 $tablename = $reg[1];
323 $line =
"-- ".$line.
" replaced by --\n";
324 $line .=
"CREATE ".(preg_match(
'/UNIQUE/', $reg[2]) ?
'UNIQUE ' :
'').
"INDEX ".$idxname.
" ON ".$tablename.
" (".$fieldlist.
")";
330 $line = str_replace(
" LIKE '",
" ILIKE '", $line, $count_like);
333 $line = preg_replace(
'/\s+(\(+\s*)([a-zA-Z0-9\-\_\.]+) ILIKE /',
' \1unaccent(\2) ILIKE ', $line);
336 $line = str_replace(
" LIKE BINARY '",
" LIKE '", $line);
339 $line = preg_replace(
'/^INSERT IGNORE/',
'INSERT', $line);
343 if (preg_match(
'/DELETE FROM ([a-z_]+) USING ([a-z_]+), ([a-z_]+)/i', $line, $reg)) {
344 if ($reg[1] == $reg[2]) {
345 $line = preg_replace(
'/DELETE FROM ([a-z_]+) USING ([a-z_]+), ([a-z_]+)/i',
'DELETE FROM \\1 USING \\3', $line);
350 $line = preg_replace(
'/FROM\s*\((([a-z_]+)\s+as\s+([a-z_]+)\s*)\)/i',
'FROM \\1', $line);
354 $line = preg_replace(
'/FROM\s*\(([a-z_]+\s+as\s+[a-z_]+)\s*,\s*([a-z_]+\s+as\s+[a-z_]+\s*)\)/i',
'FROM \\1, \\2', $line);
358 $line = preg_replace(
'/FROM\s*\(([a-z_]+\s+as\s+[a-z_]+)\s*,\s*([a-z_]+\s+as\s+[a-z_]+\s*),\s*([a-z_]+\s+as\s+[a-z_]+\s*)\)/i',
'FROM \\1, \\2, \\3', $line);
362 $line = preg_replace(
'/FROM\s*\(([a-z_]+\s+as\s+[a-z_]+)\s*,\s*([a-z_]+\s+as\s+[a-z_]+\s*),\s*([a-z_]+\s+as\s+[a-z_]+\s*),\s*([a-z_]+\s+as\s+[a-z_]+\s*)\)/i',
'FROM \\1, \\2, \\3, \\4', $line);
366 $line = preg_replace(
'/FROM\s*\(([a-z_]+\s+as\s+[a-z_]+)\s*,\s*([a-z_]+\s+as\s+[a-z_]+\s*),\s*([a-z_]+\s+as\s+[a-z_]+\s*),\s*([a-z_]+\s+as\s+[a-z_]+\s*),\s*([a-z_]+\s+as\s+[a-z_]+\s*)\)/i',
'FROM \\1, \\2, \\3, \\4, \\5', $line);
373 if ($unescapeslashquot) {
374 $line = preg_replace(
"/\\\'/",
"''", $line);
395 if ($database == $this->database_name) {
413 public function connect($host, $login, $passwd, $name, $port = 0)
420 $host = str_replace(array(
"\\",
"'"), array(
"\\\\",
"\\'"), $host);
421 $login = str_replace(array(
"\\",
"'"), array(
"\\\\",
"\\'"), $login);
422 $passwd = str_replace(array(
"\\",
"'"), array(
"\\\\",
"\\'"), $passwd);
423 $name = str_replace(array(
"\\",
"'"), array(
"\\\\",
"\\'"), $name);
424 $port = str_replace(array(
"\\",
"'"), array(
"\\\\",
"\\'"), (
string) $port);
431 if ((!empty($host) && $host ==
"socket") && !defined(
'NOLOCALSOCKETPGCONNECT')) {
432 $con_string =
"dbname='".$name.
"' user='".$login.
"' password='".$passwd.
"'";
434 $this->db = @pg_connect($con_string);
441 if (empty($this->db)) {
449 $con_string =
"host='".$host.
"' port='".$port.
"' dbname='".$name.
"' user='".$login.
"' password='".$passwd.
"'";
451 $this->db = @pg_connect($con_string);
453 print $e->getMessage();
459 $this->database_name = $name;
460 pg_set_error_verbosity($this->db, PGSQL_ERRORS_VERBOSE);
461 pg_query($this->db,
"set datestyle = 'ISO, YMD';");
474 $resql = $this->
query(
'SHOW server_version');
477 return $liste[
'server_version'];
489 return 'pgsql php driver';
501 if ($this->transaction_opened > 0) {
502 dol_syslog(get_class($this).
"::close Closing a connection with an opened transaction depth=".$this->transaction_opened, LOG_ERR);
504 $this->connected =
false;
505 return pg_close($this->db);
519 public function query($query, $usesavepoint = 0,
$type =
'auto', $result_mode = 0)
521 global $dolibarr_main_db_readonly;
523 $query = trim($query);
526 $query = $this->
convertSQLFromMysql($query,
$type, ($this->unescapeslashquot && $this->standard_conforming_strings));
533 if (preg_match(
'/([^\'])([0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9] [0-9][0-9]:[0-9][0-9]:[0-9][0-9])/', $query)) {
534 $query = preg_replace(
'/([^\'])([0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9] [0-9][0-9]:[0-9][0-9]:[0-9][0-9])/',
'\\1\'\\2\'', $query);
535 dol_syslog(
"Warning: Bad formed request converted into ".$query, LOG_WARNING);
542 if ($usesavepoint && $this->transaction_opened) {
543 @pg_query($this->db,
'SAVEPOINT mysavepoint');
546 if (!in_array($query, array(
'BEGIN',
'COMMIT',
'ROLLBACK'))) {
547 $SYSLOG_SQL_LIMIT = 10000;
548 dol_syslog(
'sql='.substr($query, 0, $SYSLOG_SQL_LIMIT), LOG_DEBUG);
554 if (!empty($dolibarr_main_db_readonly)) {
555 if (preg_match(
'/^(INSERT|UPDATE|REPLACE|DELETE|CREATE|ALTER|TRUNCATE|DROP)/i', $query)) {
556 $this->
lasterror =
'Application in read-only mode';
563 $ret = @pg_query($this->db, $query);
566 if (!preg_match(
"/^COMMIT/i", $query) && !preg_match(
"/^ROLLBACK/i", $query)) {
568 if ($this->
errno() !=
'DB_ERROR_25P02') {
574 dol_syslog(get_class($this).
"::query SQL Error query: ".$query, LOG_ERR);
576 dol_syslog(get_class($this).
"::query SQL Error message: ".$this->
lasterror.
" (".$this->lasterrno.
")", LOG_ERR);
577 dol_syslog(get_class($this).
"::query SQL Error usesavepoint = ".$usesavepoint, LOG_ERR);
580 if ($usesavepoint && $this->transaction_opened) {
581 @pg_query($this->db,
'ROLLBACK TO SAVEPOINT mysavepoint');
585 $this->_results = $ret;
602 if (!is_resource($resultset) && !is_object($resultset)) {
603 $resultset = $this->_results;
605 return pg_fetch_object($resultset);
619 if (!is_resource($resultset) && !is_object($resultset)) {
620 $resultset = $this->_results;
622 return pg_fetch_array($resultset);
637 if (!is_resource($resultset) && !is_object($resultset)) {
638 $resultset = $this->_results;
640 if (is_bool($resultset)) {
643 return pg_fetch_row($resultset);
658 if (!is_resource($resultset) && !is_object($resultset)) {
659 $resultset = $this->_results;
663 return pg_num_rows($resultset);
681 if (!is_resource($resultset) && !is_object($resultset)) {
682 $resultset = $this->_results;
686 return pg_affected_rows($resultset);
696 public function free($resultset =
null)
699 if (!is_resource($resultset) && !is_object($resultset)) {
700 $resultset = $this->_results;
703 if (is_resource($resultset) || is_object($resultset)) {
704 pg_free_result($resultset);
716 public function plimit($limit = 0, $offset = 0)
723 $limit =
$conf->liste_limit;
726 return " LIMIT ".$limit.
" OFFSET ".$offset.
" ";
728 return " LIMIT $limit ";
741 return pg_escape_string($this->db, $stringtoencode);
752 return str_replace(array(
'\\',
'_',
'%'), array(
'\\\\',
'\_',
'\%'), (
string) $stringtoencode);
763 public function ifsql($test, $resok, $resko)
765 return '(CASE WHEN '.$test.
' THEN '.$resok.
' ELSE '.$resko.
' END)';
776 public function regexpsql($subject, $pattern, $sqlstring = 0)
779 return "(". $subject .
" ~ '" . $this->
escape($pattern) .
"')";
782 return "('". $this->
escape($subject) .
"' ~ '" . $this->
escape($pattern) .
"')";
793 if (!$this->connected) {
795 return 'DB_ERROR_FAILED_TO_CONNECT';
798 $errorcode_map = array(
799 1004 =>
'DB_ERROR_CANNOT_CREATE',
800 1005 =>
'DB_ERROR_CANNOT_CREATE',
801 1006 =>
'DB_ERROR_CANNOT_CREATE',
802 1007 =>
'DB_ERROR_ALREADY_EXISTS',
803 1008 =>
'DB_ERROR_CANNOT_DROP',
804 1025 =>
'DB_ERROR_NO_FOREIGN_KEY_TO_DROP',
805 1044 =>
'DB_ERROR_ACCESSDENIED',
806 1046 =>
'DB_ERROR_NODBSELECTED',
807 1048 =>
'DB_ERROR_CONSTRAINT',
808 '42P07' =>
'DB_ERROR_TABLE_OR_KEY_ALREADY_EXISTS',
809 '42703' =>
'DB_ERROR_NOSUCHFIELD',
810 1060 =>
'DB_ERROR_COLUMN_ALREADY_EXISTS',
811 42701 =>
'DB_ERROR_COLUMN_ALREADY_EXISTS',
812 '42710' =>
'DB_ERROR_KEY_NAME_ALREADY_EXISTS',
813 '23505' =>
'DB_ERROR_RECORD_ALREADY_EXISTS',
814 '42704' =>
'DB_ERROR_NO_INDEX_TO_DROP',
815 '42601' =>
'DB_ERROR_SYNTAX',
816 '42P16' =>
'DB_ERROR_PRIMARY_KEY_ALREADY_EXISTS',
817 1075 =>
'DB_ERROR_CANT_DROP_PRIMARY_KEY',
818 1091 =>
'DB_ERROR_NOSUCHFIELD',
819 1100 =>
'DB_ERROR_NOT_LOCKED',
820 1136 =>
'DB_ERROR_VALUE_COUNT_ON_ROW',
821 '42P01' =>
'DB_ERROR_NOSUCHTABLE',
822 '23503' =>
'DB_ERROR_NO_PARENT',
823 1217 =>
'DB_ERROR_CHILD_EXISTS',
824 1451 =>
'DB_ERROR_CHILD_EXISTS',
825 '42P04' =>
'DB_DATABASE_ALREADY_EXISTS'
828 $errorlabel = pg_last_error($this->db);
831 if (preg_match(
'/: *([0-9P]+):/', $errorlabel, $reg)) {
832 $errorcode = $reg[1];
833 if (isset($errorcode_map[$errorcode])) {
834 return $errorcode_map[$errorcode];
837 $errno = $errorcode ? $errorcode : $errorlabel;
838 return ($errno ?
'DB_ERROR_'.$errno :
'0');
857 return pg_last_error($this->db);
871 $sequencename = $table.
"_".$fieldid.
"_seq";
874 $result = pg_query($this->db,
"SELECT currval('".$sequencename.
"')");
876 print pg_last_error($this->db);
880 $row = pg_fetch_result($result, 0, 0);
892 public function encrypt($fieldorvalue, $withQuotes = 1)
902 $return = $fieldorvalue;
903 return ($withQuotes ?
"'" :
"").$this->
escape($return).($withQuotes ?
"'" :
"");
954 public function DDLCreateDb($database, $charset =
'', $collation =
'', $owner =
'')
957 if (empty($charset)) {
960 if (empty($collation)) {
968 $sql =
"CREATE DATABASE ".$this->escape($database).
" OWNER '".$this->
escape($owner).
"' ENCODING '".$this->
escape((
string) $charset).
"'";
971 $ret = $this->
query($sql);
987 $listtables = array();
991 $tmptable = preg_replace(
'/[^a-z0-9\.\-\_%]/i',
'', $table);
993 $escapedlike =
" AND table_name LIKE '".$this->escape($tmptable).
"'";
995 $result = pg_query($this->db,
"SELECT table_name FROM information_schema.tables WHERE table_schema = 'public'".$escapedlike.
" ORDER BY table_name");
997 while ($row = $this->
fetch_row($result)) {
998 $listtables[] = $row[0];
1015 $listtables = array();
1019 $tmptable = preg_replace(
'/[^a-z0-9\.\-\_%]/i',
'', $table);
1021 $escapedlike =
" AND table_name LIKE '".$this->escape($tmptable).
"'";
1023 $result = pg_query($this->db,
"SELECT table_name, table_type FROM information_schema.tables WHERE table_schema = 'public'".$escapedlike.
" ORDER BY table_name");
1025 while ($row = $this->
fetch_row($result)) {
1026 $listtables[] = $row;
1042 $infotables = array();
1045 $sql .=
" infcol.column_name as \"Column\",";
1046 $sql .=
" CASE WHEN infcol.character_maximum_length IS NOT NULL THEN infcol.udt_name || '('||infcol.character_maximum_length||')'";
1047 $sql .=
" ELSE infcol.udt_name";
1048 $sql .=
" END as \"Type\",";
1049 $sql .=
" infcol.collation_name as \"Collation\",";
1050 $sql .=
" infcol.is_nullable as \"Null\",";
1051 $sql .=
" '' as \"Key\",";
1052 $sql .=
" infcol.column_default as \"Default\",";
1053 $sql .=
" '' as \"Extra\",";
1054 $sql .=
" '' as \"Privileges\"";
1055 $sql .=
" FROM information_schema.columns infcol";
1056 $sql .=
" WHERE table_schema = 'public' ";
1057 $sql .=
" AND table_name = '".$this->escape($table).
"'";
1058 $sql .=
" ORDER BY ordinal_position;";
1060 $result = $this->
query($sql);
1062 while ($row = $this->
fetch_row($result)) {
1063 $infotables[] = $row;
1083 public function DDLCreateTable($table, $fields, $primary_key,
$type, $unique_keys =
null, $fulltext_keys =
null, $keys =
null)
1098 $sql =
"CREATE TABLE ".$this->sanitize($table).
"(";
1100 $sqlfields = array();
1101 foreach ($fields as $field_name => $field_desc) {
1102 $sqlfields[$i] = $this->
sanitize($field_name).
" ";
1103 $sqlfields[$i] .= $this->
sanitize($field_desc[
'type']);
1104 if (isset($field_desc[
'value']) && $field_desc[
'value'] !==
'') {
1105 $sqlfields[$i] .=
"(".$this->sanitize($field_desc[
'value']).
")";
1107 if (isset($field_desc[
'attribute']) && $field_desc[
'attribute'] !==
'') {
1108 $sqlfields[$i] .=
" ".$this->sanitize($field_desc[
'attribute'], 0, 0, 1);
1110 if (isset($field_desc[
'default']) && $field_desc[
'default'] !==
'') {
1111 if (in_array($field_desc[
'type'], array(
'tinyint',
'smallint',
'int',
'double'))) {
1112 $sqlfields[$i] .=
" DEFAULT ".((float) $field_desc[
'default']);
1113 } elseif ($field_desc[
'default'] ==
'null' || $field_desc[
'default'] ==
'CURRENT_TIMESTAMP') {
1114 $sqlfields[$i] .=
" DEFAULT ".$this->sanitize($field_desc[
'default']);
1116 $sqlfields[$i] .=
" DEFAULT '".$this->escape($field_desc[
'default']).
"'";
1119 if (isset($field_desc[
'null']) && $field_desc[
'null'] !==
'') {
1120 $sqlfields[$i] .=
" ".$this->sanitize($field_desc[
'null'], 0, 0, 1);
1122 if (isset($field_desc[
'extra']) && $field_desc[
'extra'] !==
'') {
1123 $sqlfields[$i] .=
" ".$this->sanitize($field_desc[
'extra'], 0, 0, 1);
1125 if (!empty($primary_key) && $primary_key == $field_name) {
1126 $sqlfields[$i] .=
" AUTO_INCREMENT PRIMARY KEY";
1131 if (is_array($unique_keys)) {
1133 foreach ($unique_keys as $key => $value) {
1134 $sqluq[$i] =
"UNIQUE KEY '".$this->sanitize($key).
"' ('".$this->
escape($value).
"')";
1138 if (is_array($keys)) {
1140 foreach ($keys as $key => $value) {
1141 $sqlk[$i] =
"KEY ".$this->sanitize($key).
" (".$value.
")";
1145 $sql .= implode(
', ', $sqlfields);
1146 if (!is_array($unique_keys) && $unique_keys !=
"") {
1147 $sql .=
",".implode(
',', $sqluq);
1149 if (is_array($keys)) {
1150 $sql .=
",".implode(
',', $sqlk);
1155 if (!$this->
query($sql, 1)) {
1172 $tmptable = preg_replace(
'/[^a-z0-9\.\-\_]/i',
'', $table);
1174 $sql =
"DROP TABLE ".$this->sanitize($tmptable);
1176 if (!$this->
query($sql, 1)) {
1194 $sql =
"SELECT attname FROM pg_attribute, pg_type WHERE typname = '".$this->escape($table).
"' AND attrelid = typrelid";
1195 $sql .=
" AND attname NOT IN ('cmin', 'cmax', 'ctid', 'oid', 'tableoid', 'xmin', 'xmax')";
1197 $sql .=
" AND attname = '".$this->escape($field).
"'";
1201 $this->_results = $this->
query($sql);
1202 return $this->_results;
1215 public function DDLAddField($table, $field_name, $field_desc, $field_position =
"")
1220 $sql =
"ALTER TABLE ".$this->sanitize($table).
" ADD ".$this->
sanitize($field_name).
" ";
1222 if ($field_desc[
'type'] !==
'datetimegmt') {
1223 $sql .= $this->
sanitize($field_desc[
'type']);
1228 if (in_array($field_desc[
'type'], array(
'varchar')) && array_key_exists(
'value', $field_desc) && !empty($field_desc[
'value'])) {
1229 $sql .=
"(".$this->sanitize($field_desc[
'value']).
")";
1231 if (isset($field_desc[
'attribute']) && preg_match(
"/^[^\s]/i", $field_desc[
'attribute'])) {
1232 $sql .=
" ".$this->sanitize($field_desc[
'attribute']);
1234 if (isset($field_desc[
'null']) && preg_match(
"/^[^\s]/i", $field_desc[
'null'])) {
1235 if ($field_desc[
'null'] ==
'NOT NULL') {
1236 $sql .=
" ".$this->sanitize($field_desc[
'null'], 0, 0, 1);
1238 $sql .=
" ".$this->sanitize($field_desc[
'null']);
1241 if (isset($field_desc[
'default']) && preg_match(
"/^[^\s]/i", $field_desc[
'default'])) {
1242 if (in_array($field_desc[
'type'], array(
'tinyint',
'smallint',
'int',
'double'))) {
1243 $sql .=
" DEFAULT ".((float) $field_desc[
'default']);
1244 } elseif ($field_desc[
'default'] ==
'null' || $field_desc[
'default'] ==
'CURRENT_TIMESTAMP') {
1245 $sql .=
" DEFAULT ".$this->sanitize($field_desc[
'default']);
1247 $sql .=
" DEFAULT '".$this->escape($field_desc[
'default']).
"'";
1250 if (isset($field_desc[
'extra']) && preg_match(
"/^[^\s]/i", $field_desc[
'extra'])) {
1251 $sql .=
" ".$this->sanitize($field_desc[
'extra'], 0, 0, 1);
1253 $sql .=
" ".$this->sanitize($field_position, 0, 0, 1);
1255 dol_syslog(get_class($this).
"::DDLAddField ".$sql, LOG_DEBUG);
1256 if ($this->
query($sql)) {
1274 $sql =
"ALTER TABLE ".$this->sanitize($table);
1275 $sql .=
" ALTER COLUMN ".$this->sanitize($field_name).
" TYPE ";
1277 if ($field_desc[
'type'] !==
'datetimegmt') {
1278 $sql .= $this->
sanitize($field_desc[
'type']);
1283 if (in_array($field_desc[
'type'], array(
'varchar')) && array_key_exists(
'value', $field_desc) && !empty($field_desc[
'value'])) {
1284 $sql .=
"(".$this->sanitize($field_desc[
'value']).
")";
1287 if (isset($field_desc[
'null']) && ($field_desc[
'null'] ==
'not null' || $field_desc[
'null'] ==
'NOT NULL')) {
1289 if ($field_desc[
'type'] ==
'varchar' || $field_desc[
'type'] ==
'text') {
1290 $sqlbis =
"UPDATE ".$this->sanitize($table).
" SET ".$this->
escape($field_name).
" = '".$this->
escape(isset($field_desc[
'default']) ? $field_desc[
'default'] :
'').
"' WHERE ".$this->
escape($field_name).
" IS NULL";
1291 $this->
query($sqlbis);
1292 } elseif (in_array($field_desc[
'type'], array(
'tinyint',
'smallint',
'int',
'double'))) {
1293 $sqlbis =
"UPDATE ".$this->sanitize($table).
" SET ".$this->
escape($field_name).
" = ".((float) $this->
escape(isset($field_desc[
'default']) ? $field_desc[
'default'] : 0)).
" WHERE ".$this->
escape($field_name).
" IS NULL";
1294 $this->
query($sqlbis);
1298 if (isset($field_desc[
'default']) && $field_desc[
'default'] !=
'') {
1299 if (in_array($field_desc[
'type'], array(
'tinyint',
'smallint',
'int',
'double'))) {
1300 $sql .=
", ALTER COLUMN ".$this->sanitize($field_name).
" SET DEFAULT ".((float) $field_desc[
'default']);
1301 } elseif ($field_desc[
'type'] !=
'text') {
1302 $sql .=
", ALTER COLUMN ".$this->sanitize($field_name).
" SET DEFAULT '".$this->
escape($field_desc[
'default']).
"'";
1307 if (!$this->
query($sql)) {
1324 $tmp_field_name = preg_replace(
'/[^a-z0-9\.\-\_]/i',
'', $field_name);
1326 $sql =
"ALTER TABLE ".$this->sanitize($table).
" DROP COLUMN ".$this->
sanitize($tmp_field_name);
1327 if (!$this->
query($sql)) {
1344 public function DDLCreateUser($dolibarr_main_db_host, $dolibarr_main_db_user, $dolibarr_main_db_pass, $dolibarr_main_db_name)
1348 $sql =
"CREATE USER ".$this->sanitize($dolibarr_main_db_user).
" with password '".$this->
escape($dolibarr_main_db_pass).
"'";
1350 dol_syslog(get_class($this).
"::DDLCreateUser", LOG_DEBUG);
1351 $resql = $this->
query($sql);
1366 $resql = $this->
query(
'SHOW SERVER_ENCODING');
1369 return $liste[
'server_encoding'];
1382 $resql = $this->
query(
'SHOW SERVER_ENCODING');
1387 $liste[$i][
'charset'] = $obj->server_encoding;
1388 $liste[$i][
'description'] =
'Default database charset';
1391 $this->
free($resql);
1405 $resql = $this->
query(
'SHOW LC_COLLATE');
1408 return $liste[
'lc_collate'];
1421 $resql = $this->
query(
'SHOW LC_COLLATE');
1426 $liste[$i][
'collation'] = $obj->lc_collate;
1429 $this->
free($resql);
1443 $fullpathofdump =
'/pathtopgdump/pg_dump';
1445 if (file_exists(
'/usr/bin/pg_dump')) {
1446 $fullpathofdump =
'/usr/bin/pg_dump';
1449 $resql = $this->
query(
'SHOW data_directory');
1452 $basedir = $liste[
'data_directory'];
1453 $fullpathofdump = preg_replace(
'/data$/',
'bin', $basedir).
'/pg_dump';
1457 return $fullpathofdump;
1470 $fullpathofdump =
'/pathtopgrestore/'.$tool;
1472 if (file_exists(
'/usr/bin/'.$tool)) {
1473 $fullpathofdump =
'/usr/bin/'.$tool;
1476 $resql = $this->
query(
'SHOW data_directory');
1479 $basedir = $liste[
'data_directory'];
1480 $fullpathofdump = preg_replace(
'/data$/',
'bin', $basedir).
'/'.$tool;
1484 return $fullpathofdump;
1497 $resql =
'select name,setting from pg_settings';
1499 $resql .=
" WHERE name = '".$this->escape($filter).
"'";
1501 $resql = $this->
query($resql);
1504 $result[$obj->name] = $obj->setting;
1553 $stmtname =
'dolipgstmt_' . bin2hex(random_bytes(8));
1555 $result = pg_prepare($this->db, $stmtname, $sql);
1557 $this->
lasterror = pg_last_error($this->db);
Class to manage Dolibarr database access.
lastqueryerror()
Return last query in error.
lasterror()
Return last error label.
lasterrno()
Return last error code.
lastquery()
Return last request executed with query()
sanitize($stringtosanitize, $allowsimplequote=0, $allowsequals=0, $allowsspace=0, $allowschars=1)
Sanitize a string for SQL forging.
Class to drive a PostgreSQL database for Dolibarr.
errno()
Renvoie le code erreur generique de l'operation precedente.
DDLListTablesFull($database, $table='')
List tables into a database.
DDLGetConnectId()
Return connection ID.
num_rows($resultset)
Return number of lines for result of a SELECT.
const VERSIONMIN
Version min database.
__construct($type, $host, $user, $pass, $name='', $port=0)
Constructor.
DDLCreateUser($dolibarr_main_db_host, $dolibarr_main_db_user, $dolibarr_main_db_pass, $dolibarr_main_db_name)
Create a user to connect to database.
DDLCreateTable($table, $fields, $primary_key, $type, $unique_keys=null, $fulltext_keys=null, $keys=null)
Create a table into database.
getNextAutoIncrementId($table)
Get the last ID of an auto-increment field of a table.
DDLDropTable($table)
Drop a table into database.
DDLUpdateField($table, $field_name, $field_desc)
Update format of a field into a table.
getPathOfDump()
Return full path of dump program.
select_db($database)
Select a database PostgreSQL does not have an equivalent for mysql_select_db Only compare if the chos...
getServerStatusValues($filter='')
Return value of server status.
plimit($limit=0, $offset=0)
Define limits and offset of request.
decrypt($value)
Decrypt sensitive data in database.
error()
Renvoie le texte de l'erreur pgsql de l'operation precedente.
escape($stringtoencode)
Escape a string to insert data.
query($query, $usesavepoint=0, $type='auto', $result_mode=0)
Convert request to PostgreSQL syntax, execute it and return the resultset.
fetch_object($resultset)
Returns the current line (as an object) for the resultset cursor.
close()
Close database connection.
encrypt($fieldorvalue, $withQuotes=1)
Encrypt sensitive data in database Warning: This function includes the escape and add the SQL simple ...
getListOfCharacterSet()
Return list of available charset that can be used to store data in database.
fetch_array($resultset)
Return datas as an array.
getPathOfRestore()
Return full path of restore program.
DDLAddField($table, $field_name, $field_desc, $field_position="")
Create a new field into table.
DDLInfoTable($table)
List information of columns in a table.
connect($host, $login, $passwd, $name, $port=0)
Connection to server.
getVersion()
Return version of database server.
escapeforlike($stringtoencode)
Escape a string to insert data into a like.
last_insert_id($table, $fieldid='rowid')
Get last ID after an insert INSERT.
affected_rows($resultset)
Return the number of lines in the result of a request INSERT, DELETE or UPDATE.
DDLDropField($table, $field_name)
Drop a field from table.
regexpsql($subject, $pattern, $sqlstring=0)
Format a SQL REGEXP.
DDLCreateDb($database, $charset='', $collation='', $owner='')
Create a new database Do not use function xxx_create_db (xxx=mysql, ...) as they are deprecated We fo...
const LABEL
Database label.
getListOfCollation()
Return list of available collation that can be used for database.
getDriverInfo()
Return version of database client driver.
free($resultset=null)
Free the last pointer resultset used by this connection.
DDLDescTable($table, $field="")
Return a pointer of line with description of a table or field.
ifsql($test, $resok, $resko)
Format a SQL IF.
getDefaultCollationDatabase()
Return collation used in database.
convertSQLFromMysql($line, $type='auto', $unescapeslashquot=false)
Convert a SQL request in Mysql syntax to native syntax.
getDefaultCharacterSetDatabase()
Return charset used to store data in database.
$forcecollate
Collate used to force collate when creating database.
fetch_row($resultset)
Return datas as an array.
getServerParametersValues($filter='')
Return value of server parameters.
prepare($sql)
Prepare a SQL statement for execution (PostgreSQL prepared statement)
DDLListTables($database, $table='')
List tables into a database.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.