2010-02-11 07:38:10 +00:00
< ? php
2013-04-03 17:19:26 +13:00
2010-02-11 07:38:10 +00:00
/**
* This is a helper class for the SS installer .
*
* It does all the specific checking for MSSQLDatabase
* to ensure that the configuration is setup correctly .
*
* @ package mssql
*/
class MSSQLDatabaseConfigurationHelper implements DatabaseConfigurationHelper {
2013-04-03 17:19:26 +13:00
protected function isAzure ( $databaseConfig ) {
return $databaseConfig [ 'type' ] === 'MSSQLAzureDatabase' ;
}
2010-02-11 07:38:10 +00:00
/**
2013-04-03 17:19:26 +13:00
* Create a connection of the appropriate type
2010-02-11 07:38:10 +00:00
*
2013-04-03 17:19:26 +13:00
* @ param array $databaseConfig
* @ param string $error Error message passed by value
* @ return mixed | null Either the connection object , or null if error
2010-02-11 07:38:10 +00:00
*/
2013-04-03 17:19:26 +13:00
protected function createConnection ( $databaseConfig , & $error ) {
$error = null ;
try {
switch ( $databaseConfig [ 'type' ]) {
case 'MSSQLDatabase' :
case 'MSSQLAzureDatabase' :
$parameters = array (
'UID' => $databaseConfig [ 'username' ],
'PWD' => $databaseConfig [ 'password' ]
);
// Azure has additional parameter requirements
if ( $this -> isAzure ( $databaseConfig )) {
$parameters [ 'database' ] = $databaseConfig [ 'database' ];
$parameters [ 'multipleactiveresultsets' ] = 0 ;
}
$conn = @ sqlsrv_connect ( $databaseConfig [ 'server' ], $parameters );
if ( $conn ) return $conn ;
// Get error
if ( $errors = sqlsrv_errors ()) {
$error = '' ;
foreach ( $errors as $detail ) {
$error .= " { $detail [ 'message' ] } \n " ;
}
} else {
$error = 'Unknown connection error' ;
}
return null ;
case 'MSSQLPDODatabase' :
// May throw a PDOException if fails
$conn = @ new PDO ( 'sqlsrv:Server=' . $databaseConfig [ 'server' ], $databaseConfig [ 'username' ], $databaseConfig [ 'password' ]);
if ( $conn ) {
return $conn ;
} else {
$error = 'Unknown connection error' ;
return null ;
}
default :
$error = 'Invalid connection type' ;
return null ;
}
} catch ( Exception $ex ) {
$error = $ex -> getMessage ();
return null ;
}
2010-02-11 07:38:10 +00:00
}
2013-04-03 17:19:26 +13:00
2010-02-11 07:38:10 +00:00
/**
2013-04-03 17:19:26 +13:00
* Helper function to quote a string value
*
* @ param mixed $conn Connection object / resource
* @ param string $value Value to quote
* @ return string Quoted strieng
2010-02-11 07:38:10 +00:00
*/
2013-04-03 17:19:26 +13:00
protected function quote ( $conn , $value ) {
if ( $conn instanceof PDO ) {
return $conn -> quote ( $value );
} elseif ( is_resource ( $conn )) {
$value = str_replace ( " ' " , " '' " , $value );
$value = str_replace ( " \0 " , " [NULL] " , $value );
return " N' $value ' " ;
2010-02-11 07:38:10 +00:00
} else {
2013-04-03 17:19:26 +13:00
user_error ( 'Invalid database connection' , E_USER_ERROR );
2010-02-11 07:38:10 +00:00
}
2013-04-03 17:19:26 +13:00
}
/**
* Helper function to execute a query
*
* @ param mixed $conn Connection object / resource
* @ param string $sql SQL string to execute
* @ return array List of first value from each resulting row
*/
protected function query ( $conn , $sql ) {
$items = array ();
if ( $conn instanceof PDO ) {
$result = $conn -> query ( $sql );
if ( $result ) foreach ( $result as $row ) {
$items [] = $row [ 0 ];
}
} elseif ( is_resource ( $conn )) {
$result = sqlsrv_query ( $conn , $sql );
if ( $result ) while ( $row = sqlsrv_fetch_array ( $result , SQLSRV_FETCH_NUMERIC )) {
$items [] = $row [ 0 ];
}
2010-02-11 07:38:10 +00:00
}
2013-04-03 17:19:26 +13:00
return $items ;
}
public function requireDatabaseFunctions ( $databaseConfig ) {
$data = DatabaseAdapterRegistry :: get_adapter ( $databaseConfig [ 'type' ]);
return ! empty ( $data [ 'supported' ]);
}
public function requireDatabaseServer ( $databaseConfig ) {
$conn = $this -> createConnection ( $databaseConfig , $error );
$success = ! empty ( $conn );
2010-02-11 07:38:10 +00:00
return array (
2010-02-11 09:59:52 +00:00
'success' => $success ,
2010-02-11 07:38:10 +00:00
'error' => $error
);
}
public function requireDatabaseConnection ( $databaseConfig ) {
2013-04-03 17:19:26 +13:00
$conn = $this -> createConnection ( $databaseConfig , $error );
$success = ! empty ( $conn );
2010-02-11 07:38:10 +00:00
return array (
2010-02-11 09:59:52 +00:00
'success' => $success ,
2010-02-11 11:52:12 +00:00
'connection' => $conn ,
2010-02-11 07:38:10 +00:00
'error' => $error
);
}
2010-05-26 05:11:14 +00:00
public function getDatabaseVersion ( $databaseConfig ) {
2013-04-03 17:19:26 +13:00
$conn = $this -> createConnection ( $databaseConfig , $error );
$result = $this -> query ( $conn , " SELECT CONVERT(char(15), SERVERPROPERTY('ProductVersion')) " );
return empty ( $result ) ? 0 : reset ( $result );
2010-05-26 05:11:14 +00:00
}
/**
* Ensure that the SQL Server version is at least 10.00 . 2531 ( SQL Server 2008 SP1 ) .
2013-04-03 17:19:26 +13:00
*
2010-05-26 05:11:14 +00:00
* @ see http :// www . sqlteam . com / article / sql - server - versions
* @ param array $databaseConfig Associative array of db configuration , e . g . " server " , " username " etc
* @ return array Result - e . g . array ( 'success' => true , 'error' => 'details of error' )
*/
public function requireDatabaseVersion ( $databaseConfig ) {
$success = false ;
$error = '' ;
$version = $this -> getDatabaseVersion ( $databaseConfig );
2010-05-13 12:25:54 +00:00
if ( $version ) {
2010-05-13 12:27:25 +00:00
$success = version_compare ( $version , '10.00.2531' , '>=' );
2010-05-13 12:25:54 +00:00
if ( ! $success ) {
$error = " Your SQL Server version is $version . It's recommended you use at least 10.00.2531 (SQL Server 2008 SP1). " ;
}
} else {
$error = " Your SQL Server version could not be determined. " ;
}
return array (
'success' => $success ,
'error' => $error
);
}
2010-02-11 07:38:10 +00:00
public function requireDatabaseOrCreatePermissions ( $databaseConfig ) {
2013-04-03 17:19:26 +13:00
$conn = $this -> createConnection ( $databaseConfig , $error );
if ( empty ( $conn )) {
$success = false ;
$alreadyExists = false ;
} elseif ( $databaseConfig [ 'type' ] == 'MSSQLAzureDatabase' ) {
// Don't bother with DB selection for azure, as it's not supported
2010-02-11 09:59:52 +00:00
$success = true ;
$alreadyExists = true ;
2010-02-11 07:38:10 +00:00
} else {
2013-04-03 17:19:26 +13:00
// does this database exist already?
$list = $this -> query ( $conn , 'SELECT NAME FROM sys.sysdatabases' );
if ( in_array ( $databaseConfig [ 'database' ], $list )) {
2010-02-11 09:59:52 +00:00
$success = true ;
2013-04-03 17:19:26 +13:00
$alreadyExists = true ;
} else {
$permissions = $this -> query ( $conn , " select COUNT(*) from sys.fn_my_permissions('','') where permission_name like 'CREATE ANY DATABASE' or permission_name like 'CREATE DATABASE'; " );
$success = $permissions [ 0 ] > 0 ;
2010-02-11 09:59:52 +00:00
$alreadyExists = false ;
2010-02-11 07:38:10 +00:00
}
}
2010-03-11 20:09:00 +00:00
2010-02-11 07:38:10 +00:00
return array (
2010-02-11 09:59:52 +00:00
'success' => $success ,
'alreadyExists' => $alreadyExists
2010-02-11 07:38:10 +00:00
);
}
2013-04-03 17:19:26 +13:00
2013-03-24 03:05:04 +13:00
public function requireDatabaseAlterPermissions ( $databaseConfig ) {
2013-04-03 17:19:26 +13:00
$success = false ;
$conn = $this -> createConnection ( $databaseConfig , $error );
if ( ! empty ( $conn )) {
if ( ! $this -> isAzure ( $databaseConfig )) {
// Make sure to select the current database when checking permission against this database
$this -> query ( $conn , " USE \" { $databaseConfig [ 'database' ] } \" " );
}
$permissions = $this -> query ( $conn , " select COUNT(*) from sys.fn_my_permissions(NULL,'DATABASE') WHERE permission_name like 'create table'; " );
$success = $permissions [ 0 ] > 0 ;
}
return array (
'success' => $success ,
'applies' => true
);
2013-03-24 03:05:04 +13:00
}
}