673 lines
21 KiB
PHP
673 lines
21 KiB
PHP
<?php
|
|
/**
|
|
* @author Jakub Cernohuby
|
|
* @author Vladimir Stastka
|
|
* @author Jakub Vrana
|
|
*/
|
|
|
|
$drivers["mssql"] = "MS SQL (beta)";
|
|
|
|
if (isset($_GET["mssql"])) {
|
|
$possible_drivers = array("SQLSRV", "MSSQL", "PDO_DBLIB");
|
|
define("DRIVER", "mssql");
|
|
if (extension_loaded("sqlsrv")) {
|
|
class Min_DB {
|
|
var $extension = "sqlsrv", $_link, $_result, $server_info, $affected_rows, $errno, $error;
|
|
|
|
function _get_error() {
|
|
$this->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) 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|
|
|
}
|
|
|
|
$jush = "mssql";
|
|
$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);
|
|
}
|
|
$unsigned = array();
|
|
$operators = array("=", "<", ">", "<=", ">=", "!=", "LIKE", "LIKE %%", "IN", "IS NULL", "NOT LIKE", "NOT IN", "IS NOT NULL");
|
|
$functions = array("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" => "+",
|
|
)
|
|
);
|
|
}
|