34 self::T_TEXT => array(
'length',
'notnull',
'default',
'fixed' ),
35 self::T_INTEGER => array(
'length',
'notnull',
'default',
'unsigned' ),
36 self::T_FLOAT => array(
'notnull',
'default' ),
37 self::T_DATE => array(
'notnull',
'default' ),
38 self::T_TIME => array(
'notnull',
'default' ),
39 self::T_TIMESTAMP => array(
'notnull',
'default' ),
40 self::T_CLOB => array(
'notnull',
'default' ),
41 self::T_BLOB => array(
'notnull',
'default' ),
47 "text" => array(
"length",
"notnull",
"default",
"fixed" ),
48 "integer" => array(
"length",
"notnull",
"default",
"unsigned" ),
49 "float" => array(
"notnull",
"default" ),
50 "date" => array(
"notnull",
"default" ),
51 "time" => array(
"notnull",
"default" ),
52 "timestamp" => array(
"notnull",
"default" ),
53 "clob" => array(
"length",
"notnull",
"default" ),
54 "blob" => array(
"length",
"notnull",
"default" ),
64 self::T_INTEGER => array( 1, 2, 3, 4, 8 ),
164 "CONSTRAINT_CATALOG",
278 "GEOMETRYCOLLECTION",
367 "MASTER_AUTO_POSITION",
369 "MASTER_CONNECT_RETRY",
371 "MASTER_HEARTBEAT_PERIOD",
377 "MASTER_RETRY_COUNT",
385 "MASTER_SSL_CRLPATH",
387 "MASTER_SSL_VERIFY_SERVER_CERT",
388 "MASTER_TLS_VERSION",
392 "MAX_CONNECTIONS_PER_HOUR",
393 "MAX_QUERIES_PER_HOUR",
396 "MAX_STATEMENT_TIME",
397 "MAX_UPDATES_PER_HOUR",
398 "MAX_USER_CONNECTIONS",
410 "MINUTE_MICROSECOND",
439 "NO_WRITE_TO_BINLOG",
522 "REPLICATE_DO_TABLE",
523 "REPLICATE_IGNORE_DB",
524 "REPLICATE_IGNORE_TABLE",
525 "REPLICATE_REWRITE_DB",
526 "REPLICATE_WILD_DO_TABLE",
527 "REPLICATE_WILD_IGNORE_TABLE",
557 "SECOND_MICROSECOND",
589 "SQL_AFTER_MTS_GAPS",
594 "SQL_CALC_FOUND_ROWS",
613 "STATS_SAMPLE_PAGES",
823 $this->db_instance = $ilDBInterface;
832 if (!$this->query_utils) {
849 'timestamp' =>
'1970-01-01 00:00:00',
850 'time' =>
'00:00:00',
851 'date' =>
'1970-01-01',
864 if (!preg_match(self::DEFINITION_TABLE_NAME, $table_name)) {
865 throw new ilDatabaseException(
'Table name must only contain _a-z0-9 and must start with a-z.');
869 throw new ilDatabaseException(
"Invalid table name '" . $table_name .
"' (Reserved Word).");
872 if (strtolower(substr($table_name, 0, 4)) ==
"sys_") {
873 throw new ilDatabaseException(
"Invalid table name '" . $table_name .
"'. Name must not start with 'sys_'.");
876 if (strlen($table_name) > 22) {
877 throw new ilDatabaseException(
"Invalid table name '" . $table_name .
"'. Maximum table identifer length is 22 bytes.");
946 if (!preg_match(
"/^[a-z]+[_a-z0-9]*$/", $column_name)) {
948 .
"'. Column name must only contain _a-z0-9 and must start with a-z.");
952 throw new ilDatabaseException(
"Invalid column name '" . $column_name .
"' (Reserved Word).");
955 if (strtolower(substr($column_name, 0, 4)) ==
"sys_") {
956 throw new ilDatabaseException(
"Invalid column name '" . $column_name .
"'. Name must not start with 'sys_'.");
959 if (strlen($column_name) > 30) {
960 throw new ilDatabaseException(
"Invalid column name '" . $column_name .
"'. Maximum column identifer length is 30 bytes.");
974 if (!preg_match(
"/^[a-z]+[_a-z0-9]*$/", $a_name)) {
975 throw new ilDatabaseException(
"Invalid column name '" . $a_name .
"'. Column name must only contain _a-z0-9 and must start with a-z.");
982 if (strlen($a_name) > 3) {
983 throw new ilDatabaseException(
"Invalid index name '" . $a_name .
"'. Maximum index identifer length is 3 bytes.");
999 switch ($a_def[
"type"]) {
1001 throw new ilDatabaseException(
"Invalid column type '" . $a_def[
"type"] .
"'. Use integer(1) instead.");
1005 throw new ilDatabaseException(
"Invalid column type '" . $a_def[
"type"] .
"'. Use float or integer instead.");
1009 throw new ilDatabaseException(
"Invalid column type '" . $a_def[
"type"] .
"'. Allowed types are: " 1016 foreach ($a_def as $k => $v) {
1018 throw new ilDatabaseException(
"Attribute '" . $k .
"' is not allowed for column type '" . $a_def[
"type"] .
"'.");
1024 switch ($a_def[
"type"]) {
1026 if (!isset($a_def[
"length"]) || $a_def[
"length"] < 1 || $a_def[
"length"] >
$max_length[self::T_TEXT]) {
1027 if (isset($a_def[
"length"])) {
1028 throw new ilDatabaseException(
"Invalid length '" . $a_def[
"length"] .
"' for type text." .
" Length must be >=1 and <= " 1034 case self::T_INTEGER:
1035 if (isset($a_def[
"length"]) && !in_array($a_def[
"length"],
$max_length[self::T_INTEGER])) {
1036 throw new ilDatabaseException(
"Invalid length '" . $a_def[
"length"] .
"' for type integer." .
" Length must be " 1037 . implode(
', ',
$max_length[self::T_INTEGER]) .
" (bytes).");
1039 if ($a_def[
"unsigned"] ?? null) {
1056 return in_array($attribute, $this->allowed_attributes[
$type]);
1131 if (!empty($db->options[
'datatype_map'])) {
1132 foreach ($db->options[
'datatype_map'] as
$type => $mapped_type) {
1133 if (array_key_exists($mapped_type, $types)) {
1134 $types[
$type] = $types[$mapped_type];
1135 } elseif (!empty($db->options[
'datatype_map_callback'][
$type])) {
1136 $parameter = array(
'type' =>
$type,
'mapped_type' => $mapped_type );
1137 $default = call_user_func_array($db->options[
'datatype_map_callback'][
$type], array( &$db, __FUNCTION__, $parameter ));
1138 $types[
$type] = $default;
1154 $types = is_array($types) ? $types : array( $types );
1155 foreach ($types as $key =>
$type) {
1156 if (!isset($this->valid_default_values[
$type])) {
1158 if (empty($db->options[
'datatype_map'][$type])) {
1159 throw new ilDatabaseException($type .
' for ' . $key .
' is not a supported column type');
1181 $value = rtrim($value);
1186 return intval($value);
1188 return !empty($value);
1192 return doubleval($value);
1201 $this->lobs[] = array(
1204 'lob_index' => null,
1205 'endOfLOB' =>
false,
1206 'resource' => $value,
1211 $lob_index = key($this->lobs);
1212 $this->lobs[$lob_index][
'lob_index'] = $lob_index;
1214 return fopen(
'MDB2LOB://' . $lob_index .
'@' . $this->db_index,
'r+');
1231 if (is_null($value)) {
1236 if (!empty($db->options[
'datatype_map'][
$type])) {
1238 if (!empty($db->options[
'datatype_map_callback'][
$type])) {
1239 $parameter = array(
'type' =>
$type,
'value' => $value,
'rtrim' => $rtrim );
1241 return call_user_func_array($db->options[
'datatype_map_callback'][
$type], array( &$db, __FUNCTION__, $parameter ));
1259 foreach ($row as $key => $value) {
1260 if (empty($types[$key])) {
1263 $value = $this->
convertResult($row[$key], $types[$key], $rtrim);
1265 $row[$key] = $value;
1282 $n_types = count($types);
1283 if ($n_cols > $n_types) {
1284 for (
$i = $n_cols - $n_types;
$i >= 0;
$i--) {
1288 $sorted_types = array();
1290 $sorted_types[$col] = null;
1293 if (array_key_exists(
$name, $sorted_types)) {
1295 unset($types[
$name]);
1300 if (count($types)) {
1302 foreach (array_keys($sorted_types) as $k) {
1303 if (is_null($sorted_types[$k])) {
1304 $sorted_types[$k] = current($types);
1310 return $sorted_types;
1325 if (!empty($db->options[
'datatype_map'][
$type])) {
1327 if (!empty($db->options[
'datatype_map_callback'][
$type])) {
1328 $parameter = array(
'type' =>
$type,
'name' =>
$name,
'field' => $field );
1330 return call_user_func_array($db->options[
'datatype_map_callback'][
$type], array( &$db, __FUNCTION__, $parameter ));
1332 $field[
'type'] =
$type;
1335 if (!method_exists($this,
"get{$type}Declaration")) {
1339 return $this->{
"get{$type}Declaration"}(
$name, $field);
1351 switch ($field[
'type']) {
1353 $length = !empty($field[
'length']) ? $field[
'length'] : $db->options[
'default_text_field_length'];
1354 $fixed = !empty($field[
'fixed']) ? $field[
'fixed'] :
false;
1356 return $fixed ? ($length ?
'CHAR(' . $length .
')' :
'CHAR(' . $db->options[
'default_text_field_length']
1357 .
')') : ($length ?
'VARCHAR(' . $length .
')' :
'TEXT');
1367 return 'CHAR (' . strlen(
'YYYY-MM-DD') .
')';
1369 return 'CHAR (' . strlen(
'HH:MM:SS') .
')';
1371 return 'CHAR (' . strlen(
'YYYY-MM-DD HH:MM:SS') .
')';
1392 $declaration_options = $db->getFieldDefinition()->getDeclarationOptions($field);
1408 if (array_key_exists(
'default', $field)) {
1409 if ($field[
'default'] ===
'') {
1412 if (empty($field[
'notnull'])) {
1413 $field[
'default'] = null;
1418 if ($field[
'default'] ===
'' 1419 && isset($db->options[
"portability"])
1420 && ($db->options[
'portability'] & 32)
1422 $field[
'default'] =
' ';
1425 $default =
' DEFAULT ' . $this->
quote($field[
'default'], $field[
'type']);
1426 } elseif (empty($field[
'notnull'])) {
1427 $default =
' DEFAULT NULL';
1430 $notnull = empty($field[
'notnull']) ?
'' :
' NOT NULL';
1432 if (isset($field[
"notnull"]) && $field[
'notnull'] ===
false) {
1439 return $charset . $default . $notnull . $collation;
1471 if (!empty($field[
'unsigned'])) {
1474 $db->warnings[] =
"unsigned integer field \"$name\" is being declared as signed integer";
1502 $notnull = empty($field[
'notnull']) ?
'' :
' NOT NULL';
1518 $notnull = empty($field[
'notnull']) ?
'' :
' NOT NULL';
1599 $type = !empty($current[
'type']) ? $current[
'type'] : null;
1601 if (!method_exists($this,
"compare{$type}Definition")) {
1604 if (!empty($db->options[
'datatype_map_callback'][
$type])) {
1605 $parameter = array(
'current' => $current,
'previous' => $previous );
1606 $change = call_user_func_array($db->options[
'datatype_map_callback'][
$type], array( &$db, __FUNCTION__, $parameter ));
1614 if (empty($previous[
'type']) || $previous[
'type'] !=
$type) {
1618 $change = $this->{
"compare{$type}Definition"}($current, $previous);
1620 if ($previous[
'type'] !=
$type) {
1621 $change[
'type'] =
true;
1624 $previous_notnull = !empty($previous[
'notnull']) ? $previous[
'notnull'] :
false;
1625 $notnull = !empty($current[
'notnull']) ? $current[
'notnull'] :
false;
1626 if ($previous_notnull != $notnull) {
1627 $change[
'notnull'] =
true;
1630 $previous_default = array_key_exists(
'default', $previous) ? $previous[
'default'] : ($previous_notnull ?
'' : null);
1631 $default = array_key_exists(
'default', $current) ? $current[
'default'] : ($notnull ?
'' : null);
1632 if ($previous_default !== $default) {
1633 $change[
'default'] =
true;
1648 $previous_unsigned = !empty($previous[
'unsigned']) ? $previous[
'unsigned'] :
false;
1649 $unsigned = !empty($current[
'unsigned']) ? $current[
'unsigned'] :
false;
1650 if ($previous_unsigned != $unsigned) {
1651 $change[
'unsigned'] =
true;
1653 $previous_autoincrement = !empty($previous[
'autoincrement']) ? $previous[
'autoincrement'] :
false;
1654 $autoincrement = !empty($current[
'autoincrement']) ? $current[
'autoincrement'] :
false;
1655 if ($previous_autoincrement != $autoincrement) {
1656 $change[
'autoincrement'] =
true;
1671 $previous_length = !empty($previous[
'length']) ? $previous[
'length'] : 0;
1672 $length = !empty($current[
'length']) ? $current[
'length'] : 0;
1673 if ($previous_length != $length) {
1674 $change[
'length'] =
true;
1676 $previous_fixed = !empty($previous[
'fixed']) ? $previous[
'fixed'] : 0;
1677 $fixed = !empty($current[
'fixed']) ? $current[
'fixed'] : 0;
1678 if ($previous_fixed != $fixed) {
1679 $change[
'fixed'] =
true;
1782 public function quote($value,
$type = null, $quote =
true, $escape_wildcards =
false)
1786 return $db->quote($value,
$type);
1789 || ($value ===
'' && $db->options[
'portability'])
1798 if (is_null(
$type)) {
1799 switch (gettype($value)) {
1812 $value = serialize($value);
1818 if (preg_match(
'/^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/', $value)) {
1819 $type =
'timestamp';
1820 } elseif (preg_match(
'/^\d{2}:\d{2}$/', $value)) {
1822 } elseif (preg_match(
'/^\d{4}-\d{2}-\d{2}$/', $value)) {
1829 } elseif (!empty($db->options[
'datatype_map'][
$type])) {
1831 if (!empty($db->options[
'datatype_map_callback'][
$type])) {
1832 $parameter = array(
'type' =>
$type,
'value' => $value,
'quote' => $quote,
'escape_wildcards' => $escape_wildcards );
1834 return call_user_func_array($db->options[
'datatype_map_callback'][
$type], array( &$db, __FUNCTION__, $parameter ));
1838 if (!method_exists($this,
"quote{$type}")) {
1841 $value = $this->{
"quote{$type}"}($value, $quote, $escape_wildcards);
1842 if ($quote && $escape_wildcards && $db->string_quoting[
'escape_pattern']
1843 && $db->string_quoting[
'escape'] !== $db->string_quoting[
'escape_pattern']
1860 return (
int) $value;
1870 protected function quoteText($value, $quote, $escape_wildcards)
1878 $value = $db->escape($value, $escape_wildcards);
1880 return "'" . $value .
"'";
1891 if (preg_match(
'/^(\w+:\/\/)(.*)$/', $value, $match)) {
1893 if ($match[1] ==
'file://') {
1897 #$value = @fopen($value, 'r'); 1900 if (is_resource($value)) {
1905 while (!@feof($fp)) {
1906 $value .= @fread($fp, $db->options[
'lob_buffer_length']);
1923 protected function quoteLOB($value, $quote, $escape_wildcards)
1927 return $this->
quoteText($value, $quote, $escape_wildcards);
1937 protected function quoteCLOB($value, $quote, $escape_wildcards)
1939 return $this->
quoteLOB($value, $quote, $escape_wildcards);
1949 protected function quoteBLOB($value, $quote, $escape_wildcards)
1951 return $this->
quoteLOB($value, $quote, $escape_wildcards);
1963 return ($value ? 1 : 0);
1973 protected function quoteDate($value, $quote, $escape_wildcards)
1975 if ($value ===
'CURRENT_DATE') {
1978 return 'CURRENT_DATE';
1981 return $this->
quoteText($value, $quote, $escape_wildcards);
1994 if ($value ===
'CURRENT_TIMESTAMP') {
1997 if (isset($db->function) && is_a($db->function,
'MDB2_Driver_Function_Common')) {
1998 return $db->function->now(
'timestamp');
2001 return 'CURRENT_TIMESTAMP';
2004 return $this->
quoteText($value, $quote, $escape_wildcards);
2014 protected function quoteTime($value, $quote, $escape_wildcards)
2017 if ($value ===
'CURRENT_TIME') {
2020 if (isset($db->function) && is_a($db->function,
'MDB2_Driver_Function_Common')) {
2021 return $db->function->now(
'time');
2024 return 'CURRENT_TIME';
2027 return $this->
quoteText($value, $quote, $escape_wildcards);
2039 if (preg_match(
'/^(.*)e([-+])(\d+)$/i', $value, $matches)) {
2040 $decimal = $this->
quoteDecimal($matches[1], $quote, $escape_wildcards);
2041 $sign = $matches[2];
2042 $exponent = str_pad($matches[3], 2,
'0', STR_PAD_LEFT);
2043 $value = $decimal .
'E' . $sign . $exponent;
2045 $value = $this->
quoteDecimal($value, $quote, $escape_wildcards);
2060 $value = (string) $value;
2061 $value = preg_replace(
'/[^\d\.,\-+eE]/',
'', $value);
2062 if (preg_match(
'/[^.0-9]/', $value)) {
2063 if (strpos($value,
',')) {
2065 if (!strpos($value,
'.')) {
2067 $value = strrev(str_replace(
',',
'.', strrev($value)));
2069 } elseif (strpos($value,
'.') && strpos($value,
'.') < strpos($value,
',')) {
2070 $value = str_replace(
'.',
'', $value);
2072 $value = strrev(str_replace(
',',
'.', strrev($value)));
2075 $value = str_replace(
',',
'', $value);
2094 if (preg_match(
'/^(\w+:\/\/)(.*)$/', $file, $match)) {
2095 if ($match[1] ==
'file://') {
2100 $fp = @fopen($file,
'wb');
2101 while (!@feof($lob)) {
2102 $result = @fread($lob, $db->options[
'lob_buffer_length']);
2104 if (@fwrite($fp,
$result, $read) != $read) {
2122 if (is_null($lob[
'value'])) {
2123 $lob[
'value'] = $lob[
'resource'];
2125 $lob[
'loaded'] =
true;
2138 return substr($lob[
'value'], $lob[
'position'], $length);
2148 return $lob[
'endOfLOB'];
2158 $lob_data = stream_get_meta_data($lob);
2159 $lob_index = $lob_data[
'wrapper_data']->lob_index;
2161 if (isset($this->lobs[$lob_index])) {
2163 unset($this->lobs[$lob_index]);
2188 if (!is_array($array) || empty($array)) {
2192 foreach ($array as $value) {
2199 return implode(
', ', $return);
2215 if (!is_null($operator)) {
2216 $operator = strtoupper($operator);
2217 switch ($operator) {
2220 if (is_null($field)) {
2221 throw new ilDatabaseException(
'case insensitive LIKE matching requires passing the field name');
2223 $db->loadModule(
'Function', null,
true);
2224 $match = $db->function->lower($field) .
' LIKE ';
2228 $match = is_null($field) ?
'LIKE ' : $field .
' LIKE ';
2235 foreach ($pattern as $key => $value) {
2239 if ($operator ===
'ILIKE') {
2240 $value = strtolower($value);
2242 $escaped = $db->escape($value);
2243 $match .= $db->escapePattern($escaped);
2269 $db_type = strtok($field[
'type'],
'(), ');
2270 if (!empty($db->options[
'nativetype_map_callback'][$db_type])) {
2271 return call_user_func_array($db->options[
'nativetype_map_callback'][$db_type], array( $db, $field ));
2294 if (!empty($db->options[
'datatype_map'][
$type])) {
2296 if (!empty($db->options[
'datatype_map_callback'][
$type])) {
2297 $parameter = array(
'type' =>
$type );
2299 return call_user_func_array($db->options[
'datatype_map_callback'][
$type], array( &$db, __FUNCTION__, $parameter ));
compareCLOBDefinition($current, $previous)
compareTimestampDefinition($current, $previous)
setAllowedAttributes($allowed_attributes)
quoteDate($value, $quote, $escape_wildcards)
getDateDeclaration($name, $field)
quote($value, $type=null, $quote=true, $escape_wildcards=false)
isAllowedAttribute($attribute, $type)
baseConvertResult($value, $type, $rtrim=true)
quoteLOB($value, $quote, $escape_wildcards)
quoteInteger($value, $quote, $escape_wildcards)
implodeArray($array, $type=false)
quoteTimestamp($value, $quote, $escape_wildcards)
compareBLOBDefinition($current, $previous)
Class ilDBPdoFieldDefinition.
quoteDecimal($value, $quote, $escape_wildcards)
compareBooleanDefinition($current, $previous)
compareFloatDefinition($current, $previous)
const DEFINITION_COLUMN_NAME
getCLOBDeclaration($name, $field)
getFloatDeclaration($name, $field)
mapNativeDatatype($field)
getCharsetFieldDeclaration($charset)
getDecimalDeclaration($name, $field)
Class ilDatabaseException.
getDeclaration($type, $name, $field)
checkColumnDefinition($a_def)
compareDateDefinition($current, $previous)
writeLOBToFile($lob, $file)
setReservedMysql($reserved_mysql)
getTypeDeclaration($field)
compareTextDefinition($current, $previous)
quoteFloat($value, $quote, $escape_wildcards)
checkTableName($table_name)
getTimeDeclaration($name, $field)
matchPattern($pattern, $operator=null, $field=null)
mapPrepareDatatype($type)
const DEFAULT_DECIMAL_PLACES
setReservedPostgres($reserved_postgres)
const SEQUENCE_COLUMNS_NAME
quoteCLOB($value, $quote, $escape_wildcards)
quoteBoolean($value, $quote, $escape_wildcards)
quoteText($value, $quote, $escape_wildcards)
getBLOBDeclaration($name, $field)
getIntegerDeclaration($name, $field)
getDeclarationOptions($field)
destroyLOBInternal(&$lob)
compareDefinition($current, $previous)
getInternalDeclaration($name, $field)
getCollationFieldDeclaration($collation)
getTextDeclaration($name, $field)
quoteBLOB($value, $quote, $escape_wildcards)
compareIntegerDefinition($current, $previous)
quoteTime($value, $quote, $escape_wildcards)
getTimestampDeclaration($name, $field)
sortResultFieldTypes($columns, $types)
__construct(\ilDBInterface $ilDBInterface)
ilDBPdoFieldDefinition constructor.
convertResultRow($types, $row, $rtrim=true)
const DEFINITION_TABLE_NAME
getBooleanDeclaration($name, $field)
compareTimeDefinition($current, $previous)
setMaxLength($max_length)
checkColumnName($column_name)
mapNativeDatatypeInternal($field)
convertResult($value, $type, $rtrim=true)
compareDecimalDefinition($current, $previous)
const DEFAULT_TEXT_LENGTH
setAvailableTypes($available_types)