error = ""; foreach (sqlsrv_errors() as $error) { $this->errno = $error["code"]; $this->error .= "$error[message]\n"; } $this->error = rtrim($this->error); } function connect($server, $username, $password) { global $adminer; $db = $adminer->database(); $connection_info = array("UID" => $username, "PWD" => $password, "CharacterSet" => "UTF-8"); if ($db != "") { $connection_info["Database"] = $db; } $this->_link = @sqlsrv_connect(preg_replace('~:~', ',', $server), $connection_info); if ($this->_link) { $info = sqlsrv_server_info($this->_link); $this->server_info = $info['SQLServerVersion']; } else { $this->_get_error(); } return (bool) $this->_link; } function quote($string) { return "'" . str_replace("'", "''", $string) . "'"; } function select_db($database) { return $this->query("USE " . idf_escape($database)); } function query($query, $unbuffered = false) { $result = sqlsrv_query($this->_link, $query); //! , array(), ($unbuffered ? array() : array("Scrollable" => "keyset")) $this->error = ""; if (!$result) { $this->_get_error(); return false; } return $this->store_result($result); } function multi_query($query) { $this->_result = sqlsrv_query($this->_link, $query); $this->error = ""; if (!$this->_result) { $this->_get_error(); return false; } return true; } function store_result($result = null) { if (!$result) { $result = $this->_result; } if (!$result) { return false; } if (sqlsrv_field_metadata($result)) { return new Min_Result($result); } $this->affected_rows = sqlsrv_rows_affected($result); return true; } function next_result() { return $this->_result ? sqlsrv_next_result($this->_result) : null; } function result($query, $field = 0) { $result = $this->query($query); if (!is_object($result)) { return false; } $row = $result->fetch_row(); return $row[$field]; } } class Min_Result { var $_result, $_offset = 0, $_fields, $num_rows; function __construct($result) { $this->_result = $result; // $this->num_rows = sqlsrv_num_rows($result); // available only in scrollable results } function _convert($row) { foreach ((array) $row as $key => $val) { if (is_a($val, 'DateTime')) { $row[$key] = $val->format("Y-m-d H:i:s"); } //! stream } return $row; } function fetch_assoc() { return $this->_convert(sqlsrv_fetch_array($this->_result, SQLSRV_FETCH_ASSOC)); } function fetch_row() { return $this->_convert(sqlsrv_fetch_array($this->_result, SQLSRV_FETCH_NUMERIC)); } function fetch_field() { if (!$this->_fields) { $this->_fields = sqlsrv_field_metadata($this->_result); } $field = $this->_fields[$this->_offset++]; $return = new stdClass; $return->name = $field["Name"]; $return->orgname = $field["Name"]; $return->type = ($field["Type"] == 1 ? 254 : 0); return $return; } function seek($offset) { for ($i=0; $i < $offset; $i++) { sqlsrv_fetch($this->_result); // SQLSRV_SCROLL_ABSOLUTE added in sqlsrv 1.1 } } function __destruct() { sqlsrv_free_stmt($this->_result); } } } elseif (extension_loaded("mssql")) { class Min_DB { var $extension = "MSSQL", $_link, $_result, $server_info, $affected_rows, $error; function connect($server, $username, $password) { $this->_link = @mssql_connect($server, $username, $password); if ($this->_link) { $result = $this->query("SELECT SERVERPROPERTY('ProductLevel'), SERVERPROPERTY('Edition')"); if ($result) { $row = $result->fetch_row(); $this->server_info = $this->result("sp_server_info 2", 2) . " [$row[0]] $row[1]"; } } else { $this->error = mssql_get_last_message(); } return (bool) $this->_link; } function quote($string) { return "'" . str_replace("'", "''", $string) . "'"; } function select_db($database) { return mssql_select_db($database); } function query($query, $unbuffered = false) { $result = @mssql_query($query, $this->_link); //! $unbuffered $this->error = ""; if (!$result) { $this->error = mssql_get_last_message(); return false; } if ($result === true) { $this->affected_rows = mssql_rows_affected($this->_link); return true; } return new Min_Result($result); } function multi_query($query) { return $this->_result = $this->query($query); } function store_result() { return $this->_result; } function next_result() { return mssql_next_result($this->_result->_result); } function result($query, $field = 0) { $result = $this->query($query); if (!is_object($result)) { return false; } return mssql_result($result->_result, 0, $field); } } class Min_Result { var $_result, $_offset = 0, $_fields, $num_rows; function __construct($result) { $this->_result = $result; $this->num_rows = mssql_num_rows($result); } function fetch_assoc() { return mssql_fetch_assoc($this->_result); } function fetch_row() { return mssql_fetch_row($this->_result); } function num_rows() { return mssql_num_rows($this->_result); } function fetch_field() { $return = mssql_fetch_field($this->_result); $return->orgtable = $return->table; $return->orgname = $return->name; return $return; } function seek($offset) { mssql_data_seek($this->_result, $offset); } function __destruct() { mssql_free_result($this->_result); } } } elseif (extension_loaded("pdo_dblib")) { class Min_DB extends Min_PDO { var $extension = "PDO_DBLIB"; function connect($server, $username, $password) { $this->dsn("dblib:charset=utf8;host=" . str_replace(":", ";unix_socket=", preg_replace('~:(\d)~', ';port=\1', $server)), $username, $password); return true; } function select_db($database) { // database selection is separated from the connection so dbname in DSN can't be used return $this->query("USE " . idf_escape($database)); } } } class Min_Driver extends Min_SQL { function insertUpdate($table, $rows, $primary) { foreach ($rows as $set) { $update = array(); $where = array(); foreach ($set as $key => $val) { $update[] = "$key = $val"; if (isset($primary[idf_unescape($key)])) { $where[] = "$key = $val"; } } //! can use only one query for all rows if (!queries("MERGE " . table($table) . " USING (VALUES(" . implode(", ", $set) . ")) AS source (c" . implode(", c", range(1, count($set))) . ") ON " . implode(" AND ", $where) //! source, c1 - possible conflict . " WHEN MATCHED THEN UPDATE SET " . implode(", ", $update) . " WHEN NOT MATCHED THEN INSERT (" . implode(", ", array_keys($set)) . ") VALUES (" . implode(", ", $set) . ");" // ; is mandatory )) { return false; } } return true; } function begin() { return queries("BEGIN TRANSACTION"); } } function idf_escape($idf) { return "[" . str_replace("]", "]]", $idf) . "]"; } function table($idf) { return ($_GET["ns"] != "" ? idf_escape($_GET["ns"]) . "." : "") . idf_escape($idf); } function connect() { global $adminer; $connection = new Min_DB; $credentials = $adminer->credentials(); if ($connection->connect($credentials[0], $credentials[1], $credentials[2])) { return $connection; } return $connection->error; } function get_databases() { return get_vals("SELECT name FROM sys.databases WHERE name NOT IN ('master', 'tempdb', 'model', 'msdb')"); } function limit($query, $where, $limit, $offset = 0, $separator = " ") { return ($limit !== null ? " TOP (" . ($limit + $offset) . ")" : "") . " $query$where"; // seek later } function limit1($table, $query, $where, $separator = "\n") { return limit($query, $where, 1, 0, $separator); } function db_collation($db, $collations) { global $connection; return $connection->result("SELECT collation_name FROM sys.databases WHERE name = " . q($db)); } function engines() { return array(); } function logged_user() { global $connection; return $connection->result("SELECT SUSER_NAME()"); } function tables_list() { return get_key_vals("SELECT name, type_desc FROM sys.all_objects WHERE schema_id = SCHEMA_ID(" . q(get_schema()) . ") AND type IN ('S', 'U', 'V') ORDER BY name"); } function count_tables($databases) { global $connection; $return = array(); foreach ($databases as $db) { $connection->select_db($db); $return[$db] = $connection->result("SELECT COUNT(*) FROM INFORMATION_SCHEMA.TABLES"); } return $return; } function table_status($name = "") { $return = array(); foreach (get_rows("SELECT ao.name AS Name, ao.type_desc AS Engine, (SELECT value FROM fn_listextendedproperty(default, 'SCHEMA', schema_name(schema_id), 'TABLE', ao.name, null, null)) AS Comment FROM sys.all_objects AS ao WHERE schema_id = SCHEMA_ID(" . q(get_schema()) . ") AND type IN ('S', 'U', 'V') " . ($name != "" ? "AND name = " . q($name) : "ORDER BY name")) as $row) { if ($name != "") { return $row; } $return[$row["Name"]] = $row; } return $return; } function is_view($table_status) { return $table_status["Engine"] == "VIEW"; } function fk_support($table_status) { return true; } function fields($table) { $comments = get_key_vals("SELECT objname, cast(value as varchar(max)) FROM fn_listextendedproperty('MS_DESCRIPTION', 'schema', " . q(get_schema()) . ", 'table', " . q($table) . ", 'column', NULL)"); $return = array(); foreach (get_rows("SELECT c.max_length, c.precision, c.scale, c.name, c.is_nullable, c.is_identity, c.collation_name, t.name type, CAST(d.definition as text) [default] FROM sys.all_columns c JOIN sys.all_objects o ON c.object_id = o.object_id JOIN sys.types t ON c.user_type_id = t.user_type_id LEFT JOIN sys.default_constraints d ON c.default_object_id = d.parent_column_id WHERE o.schema_id = SCHEMA_ID(" . q(get_schema()) . ") AND o.type IN ('S', 'U', 'V') AND o.name = " . q($table) ) as $row) { $type = $row["type"]; $length = (preg_match("~char|binary~", $type) ? $row["max_length"] : ($type == "decimal" ? "$row[precision],$row[scale]" : "")); $return[$row["name"]] = array( "field" => $row["name"], "full_type" => $type . ($length ? "($length)" : ""), "type" => $type, "length" => $length, "default" => $row["default"], "null" => $row["is_nullable"], "auto_increment" => $row["is_identity"], "collation" => $row["collation_name"], "privileges" => array("insert" => 1, "select" => 1, "update" => 1), "primary" => $row["is_identity"], //! or indexes.is_primary_key "comment" => $comments[$row["name"]], ); } return $return; } function indexes($table, $connection2 = null) { $return = array(); // sp_statistics doesn't return information about primary key foreach (get_rows("SELECT i.name, key_ordinal, is_unique, is_primary_key, c.name AS column_name, is_descending_key FROM sys.indexes i INNER JOIN sys.index_columns ic ON i.object_id = ic.object_id AND i.index_id = ic.index_id INNER JOIN sys.columns c ON ic.object_id = c.object_id AND ic.column_id = c.column_id WHERE OBJECT_NAME(i.object_id) = " . q($table) , $connection2) as $row) { $name = $row["name"]; $return[$name]["type"] = ($row["is_primary_key"] ? "PRIMARY" : ($row["is_unique"] ? "UNIQUE" : "INDEX")); $return[$name]["lengths"] = array(); $return[$name]["columns"][$row["key_ordinal"]] = $row["column_name"]; $return[$name]["descs"][$row["key_ordinal"]] = ($row["is_descending_key"] ? '1' : null); } return $return; } function view($name) { global $connection; return array("select" => preg_replace('~^(?:[^[]|\[[^]]*])*\s+AS\s+~isU', '', $connection->result("SELECT VIEW_DEFINITION FROM INFORMATION_SCHEMA.VIEWS WHERE TABLE_SCHEMA = SCHEMA_NAME() AND TABLE_NAME = " . q($name)))); } function collations() { $return = array(); foreach (get_vals("SELECT name FROM fn_helpcollations()") as $collation) { $return[preg_replace('~_.*~', '', $collation)][] = $collation; } return $return; } function information_schema($db) { return false; } function error() { global $connection; return nl_br(h(preg_replace('~^(\[[^]]*])+~m', '', $connection->error))); } function create_database($db, $collation) { return queries("CREATE DATABASE " . idf_escape($db) . (preg_match('~^[a-z0-9_]+$~i', $collation) ? " COLLATE $collation" : "")); } function drop_databases($databases) { return queries("DROP DATABASE " . implode(", ", array_map('idf_escape', $databases))); } function rename_database($name, $collation) { if (preg_match('~^[a-z0-9_]+$~i', $collation)) { queries("ALTER DATABASE " . idf_escape(DB) . " COLLATE $collation"); } queries("ALTER DATABASE " . idf_escape(DB) . " MODIFY NAME = " . idf_escape($name)); return true; //! false negative "The database name 'test2' has been set." } function auto_increment() { return " IDENTITY" . ($_POST["Auto_increment"] != "" ? "(" . number($_POST["Auto_increment"]) . ",1)" : "") . " PRIMARY KEY"; } function alter_table($table, $name, $fields, $foreign, $comment, $engine, $collation, $auto_increment, $partitioning) { $alter = array(); $comments = array(); foreach ($fields as $field) { $column = idf_escape($field[0]); $val = $field[1]; if (!$val) { $alter["DROP"][] = " COLUMN $column"; } else { $val[1] = preg_replace("~( COLLATE )'(\\w+)'~", '\1\2', $val[1]); $comments[$field[0]] = $val[5]; unset($val[5]); if ($field[0] == "") { $alter["ADD"][] = "\n " . implode("", $val) . ($table == "" ? substr($foreign[$val[0]], 16 + strlen($val[0])) : ""); // 16 - strlen(" FOREIGN KEY ()") } else { unset($val[6]); //! identity can't be removed if ($column != $val[0]) { queries("EXEC sp_rename " . q(table($table) . ".$column") . ", " . q(idf_unescape($val[0])) . ", 'COLUMN'"); } $alter["ALTER COLUMN " . implode("", $val)][] = ""; } } } if ($table == "") { return queries("CREATE TABLE " . table($name) . " (" . implode(",", (array) $alter["ADD"]) . "\n)"); } if ($table != $name) { queries("EXEC sp_rename " . q(table($table)) . ", " . q($name)); } if ($foreign) { $alter[""] = $foreign; } foreach ($alter as $key => $val) { if (!queries("ALTER TABLE " . idf_escape($name) . " $key" . implode(",", $val))) { return false; } } foreach ($comments as $key => $val) { $comment = substr($val, 9); // 9 - strlen(" COMMENT ") queries("EXEC sp_dropextendedproperty @name = N'MS_Description', @level0type = N'Schema', @level0name = " . q(get_schema()) . ", @level1type = N'Table', @level1name = " . q($name) . ", @level2type = N'Column', @level2name = " . q($key)); queries("EXEC sp_addextendedproperty @name = N'MS_Description', @value = " . $comment . ", @level0type = N'Schema', @level0name = " . q(get_schema()) . ", @level1type = N'Table', @level1name = " . q($name) . ", @level2type = N'Column', @level2name = " . q($key)); } return true; } function alter_indexes($table, $alter) { $index = array(); $drop = array(); foreach ($alter as $val) { if ($val[2] == "DROP") { if ($val[0] == "PRIMARY") { //! sometimes used also for UNIQUE $drop[] = idf_escape($val[1]); } else { $index[] = idf_escape($val[1]) . " ON " . table($table); } } elseif (!queries(($val[0] != "PRIMARY" ? "CREATE $val[0] " . ($val[0] != "INDEX" ? "INDEX " : "") . idf_escape($val[1] != "" ? $val[1] : uniqid($table . "_")) . " ON " . table($table) : "ALTER TABLE " . table($table) . " ADD PRIMARY KEY" ) . " (" . implode(", ", $val[2]) . ")")) { return false; } } return (!$index || queries("DROP INDEX " . implode(", ", $index))) && (!$drop || queries("ALTER TABLE " . table($table) . " DROP " . implode(", ", $drop))) ; } function last_id() { global $connection; return $connection->result("SELECT SCOPE_IDENTITY()"); // @@IDENTITY can return trigger INSERT } function explain($connection, $query) { $connection->query("SET SHOWPLAN_ALL ON"); $return = $connection->query($query); $connection->query("SET SHOWPLAN_ALL OFF"); // connection is used also for indexes return $return; } function found_rows($table_status, $where) { } function foreign_keys($table) { $return = array(); foreach (get_rows("EXEC sp_fkeys @fktable_name = " . q($table)) as $row) { $foreign_key = &$return[$row["FK_NAME"]]; $foreign_key["db"] = $row["PKTABLE_QUALIFIER"]; $foreign_key["table"] = $row["PKTABLE_NAME"]; $foreign_key["source"][] = $row["FKCOLUMN_NAME"]; $foreign_key["target"][] = $row["PKCOLUMN_NAME"]; } return $return; } function truncate_tables($tables) { return apply_queries("TRUNCATE TABLE", $tables); } function drop_views($views) { return queries("DROP VIEW " . implode(", ", array_map('table', $views))); } function drop_tables($tables) { return queries("DROP TABLE " . implode(", ", array_map('table', $tables))); } function move_tables($tables, $views, $target) { return apply_queries("ALTER SCHEMA " . idf_escape($target) . " TRANSFER", array_merge($tables, $views)); } function trigger($name) { if ($name == "") { return array(); } $rows = get_rows("SELECT s.name [Trigger], CASE WHEN OBJECTPROPERTY(s.id, 'ExecIsInsertTrigger') = 1 THEN 'INSERT' WHEN OBJECTPROPERTY(s.id, 'ExecIsUpdateTrigger') = 1 THEN 'UPDATE' WHEN OBJECTPROPERTY(s.id, 'ExecIsDeleteTrigger') = 1 THEN 'DELETE' END [Event], CASE WHEN OBJECTPROPERTY(s.id, 'ExecIsInsteadOfTrigger') = 1 THEN 'INSTEAD OF' ELSE 'AFTER' END [Timing], c.text FROM sysobjects s JOIN syscomments c ON s.id = c.id WHERE s.xtype = 'TR' AND s.name = " . q($name) ); // triggers are not schema-scoped $return = reset($rows); if ($return) { $return["Statement"] = preg_replace('~^.+\s+AS\s+~isU', '', $return["text"]); //! identifiers, comments } return $return; } function triggers($table) { $return = array(); foreach (get_rows("SELECT sys1.name, CASE WHEN OBJECTPROPERTY(sys1.id, 'ExecIsInsertTrigger') = 1 THEN 'INSERT' WHEN OBJECTPROPERTY(sys1.id, 'ExecIsUpdateTrigger') = 1 THEN 'UPDATE' WHEN OBJECTPROPERTY(sys1.id, 'ExecIsDeleteTrigger') = 1 THEN 'DELETE' END [Event], CASE WHEN OBJECTPROPERTY(sys1.id, 'ExecIsInsteadOfTrigger') = 1 THEN 'INSTEAD OF' ELSE 'AFTER' END [Timing] FROM sysobjects sys1 JOIN sysobjects sys2 ON sys1.parent_obj = sys2.id WHERE sys1.xtype = 'TR' AND sys2.name = " . q($table) ) as $row) { // triggers are not schema-scoped $return[$row["name"]] = array($row["Timing"], $row["Event"]); } return $return; } function trigger_options() { return array( "Timing" => array("AFTER", "INSTEAD OF"), "Event" => array("INSERT", "UPDATE", "DELETE"), "Type" => array("AS"), ); } function schemas() { return get_vals("SELECT name FROM sys.schemas"); } function get_schema() { global $connection; if ($_GET["ns"] != "") { return $_GET["ns"]; } return $connection->result("SELECT SCHEMA_NAME()"); } function set_schema($schema) { return true; // ALTER USER is permanent } function use_sql($database) { return "USE " . idf_escape($database); } function show_variables() { return array(); } function show_status() { return array(); } function convert_field($field) { } function unconvert_field($field, $return) { return $return; } function support($feature) { return preg_match('~^(comment|columns|database|drop_col|indexes|descidx|scheme|sql|table|trigger|view|view_trigger)$~', $feature); //! routine| } function driver_config() { $types = array(); $structured_types = array(); foreach (array( //! use sys.types lang('Numbers') => array("tinyint" => 3, "smallint" => 5, "int" => 10, "bigint" => 20, "bit" => 1, "decimal" => 0, "real" => 12, "float" => 53, "smallmoney" => 10, "money" => 20), lang('Date and time') => array("date" => 10, "smalldatetime" => 19, "datetime" => 19, "datetime2" => 19, "time" => 8, "datetimeoffset" => 10), lang('Strings') => array("char" => 8000, "varchar" => 8000, "text" => 2147483647, "nchar" => 4000, "nvarchar" => 4000, "ntext" => 1073741823), lang('Binary') => array("binary" => 8000, "varbinary" => 8000, "image" => 2147483647), ) as $key => $val) { $types += $val; $structured_types[$key] = array_keys($val); } return array( 'possible_drivers' => array("SQLSRV", "MSSQL", "PDO_DBLIB"), 'jush' => "mssql", 'types' => $types, 'structured_types' => $structured_types, 'unsigned' => array(), 'operators' => array("=", "<", ">", "<=", ">=", "!=", "LIKE", "LIKE %%", "IN", "IS NULL", "NOT LIKE", "NOT IN", "IS NOT NULL"), 'functions' => array("distinct", "len", "lower", "round", "upper"), 'grouping' => array("avg", "count", "count distinct", "max", "min", "sum"), 'edit_functions' => array( array( "date|time" => "getdate", ), array( "int|decimal|real|float|money|datetime" => "+/-", "char|text" => "+", ) ), ); } }