commit 41c2cea1734b66d92ba9843e8d03c6fc5114e07e Author: Sam Minnee Date: Wed Feb 25 05:44:52 2009 +0000 Initial import of MS SQL module diff --git a/_config.php b/_config.php new file mode 100644 index 0000000..e69de29 diff --git a/code/MSSQLDatabase.php b/code/MSSQLDatabase.php new file mode 100644 index 0000000..491db7d --- /dev/null +++ b/code/MSSQLDatabase.php @@ -0,0 +1,856 @@ +dbConn = mssql_connect($parameters['server'], $parameters['username'], $parameters['password']); + + if(!$this->dbConn) { + $this->databaseError("Couldn't connect to MS SQL database"); + } else { + $this->active=true; + $this->database = $parameters['database']; + mssql_select_db($parameters['database'], $this->dbConn); + } + + parent::__construct(); + + // Configure the connection + $this->query('SET QUOTED_IDENTIFIER ON'); + } + + /** + * Not implemented, needed for PDO + */ + public function getConnect($parameters) { + return null; + } + + /** + * Returns true if this database supports collations + * @return boolean + */ + public function supportsCollations() { + return true; + } + + /** + * The version of MSSQL + * @var float + */ + private $mssqlVersion; + + /** + * Get the version of MySQL. + * @return float + */ + public function getVersion() { + user_error("getVersion not implemented", E_USER_WARNING); + return 2008; + /* + if(!$this->pgsqlVersion) { + //returns something like this: PostgreSQL 8.3.3 on i386-apple-darwin9.3.0, compiled by GCC i686-apple-darwin9-gcc-4.0.1 (GCC) 4.0.1 (Apple Inc. build 5465) + $postgres=strlen('PostgreSQL '); + $db_version=$this->query("SELECT VERSION()")->value(); + + $this->pgsqlVersion = (float)trim(substr($db_version, $postgres, strpos($db_version, ' on '))); + } + return $this->pgsqlVersion; + */ + } + + /** + * Get the database server, namely mysql. + * @return string + */ + public function getDatabaseServer() { + return "mssql"; + } + + public function query($sql, $errorLevel = E_USER_ERROR) { + if(isset($_REQUEST['previewwrite']) && in_array(strtolower(substr($sql,0,strpos($sql,' '))), array('insert','update','delete','replace'))) { + Debug::message("Will execute: $sql"); + return; + } + + if(isset($_REQUEST['showqueries'])) { + $starttime = microtime(true); + } + + echo 'sql: ' . $sql . '
'; + + $handle = mssql_query($sql, $this->dbConn); + + if(isset($_REQUEST['showqueries'])) { + $endtime = round(microtime(true) - $starttime,4); + Debug::message("\n$sql\n{$endtime}ms\n", false); + } + + DB::$lastQuery=$handle; + + if(!$handle && $errorLevel) $this->databaseError("Couldn't run query: $sql", $errorLevel); + return new MSSQLQuery($this, $handle); + } + + public function getGeneratedID($table) { + $result=DB::query("SELECT last_value FROM \"{$table}_ID_seq\";"); + $row=$result->first(); + return $row['last_value']; + } + + /** + * OBSOLETE: Get the ID for the next new record for the table. + * + * @var string $table The name od the table. + * @return int + */ + public function getNextID($table) { + user_error('getNextID is OBSOLETE (and will no longer work properly)', E_USER_WARNING); + $result = $this->query("SELECT MAX(ID)+1 FROM \"$table\"")->value(); + return $result ? $result : 1; + } + + public function isActive() { + return $this->active ? true : false; + } + + public function createDatabase() { + $this->query("CREATE DATABASE $this->database"); + } + + /** + * Drop the database that this object is currently connected to. + * Use with caution. + */ + public function dropDatabase() { + $this->query("DROP DATABASE $this->database"); + } + + /** + * Returns the name of the currently selected database + */ + public function currentDatabase() { + return $this->database; + } + + /** + * Switches to the given database. + * If the database doesn't exist, you should call createDatabase() after calling selectDatabase() + */ + public function selectDatabase($dbname) { + $this->database = $dbname; + if($this->databaseExists($this->database)) mysql_select_db($this->database, $this->dbConn); + $this->tableList = $this->fieldList = $this->indexList = null; + } + + /** + * Returns true if the named database exists. + */ + public function databaseExists($name) { + $SQL_name = Convert::raw2sql($name); + return $this->query("SHOW DATABASES LIKE '$SQL_name'")->value() ? true : false; + } + + public function createTable($tableName, $fields = null, $indexes = null) { + $fieldSchemas = $indexSchemas = ""; + if($fields) foreach($fields as $k => $v) $fieldSchemas .= "\"$k\" $v,\n"; + + //if($indexes) foreach($indexes as $k => $v) $indexSchemas .= $this->getIndexSqlDefinition($k, $v) . ",\n"; + //we need to generate indexes like this: CREATE INDEX IX_vault_to_export ON vault (to_export); + + //If we have a fulltext search request, then we need to create a special column + //for GiST searches + $fulltexts=''; + + /* + foreach($indexes as $name=>$this_index){ + if($this_index['type']=='fulltext'){ + //For full text search, we need to create a column for the index + $fulltexts .= "\"ts_$name\" tsvector, "; + + } + } + */ + + //if($indexes) foreach($indexes as $k => $v) $indexSchemas .= $this->getIndexSqlDefinition($tableName, $k, $v) . "\n"; + + $this->query("CREATE TABLE \"$tableName\" ( + $fieldSchemas + $fulltexts + primary key (\"ID\") + ); $indexSchemas"); + } + + /** + * Alter a table's schema. + * @param $table The name of the table to alter + * @param $newFields New fields, a map of field name => field schema + * @param $newIndexes New indexes, a map of index name => index type + * @param $alteredFields Updated fields, a map of field name => field schema + * @param $alteredIndexes Updated indexes, a map of index name => index type + */ + public function alterTable($tableName, $newFields = null, $newIndexes = null, $alteredFields = null, $alteredIndexes = null) { + $fieldSchemas = $indexSchemas = ""; + + $alterList = array(); + if($newFields) foreach($newFields as $k => $v) $alterList[] .= "ADD \"$k\" $v"; + //if($newIndexes) foreach($newIndexes as $k => $v) $alterList[] .= "ADD " . $this->getIndexSqlDefinition($tableName, $k, $v); + + if($alteredFields) { + foreach($alteredFields as $k => $v) { + + $val=$this->alterTableAlterColumn($tableName, $k, $v); + if($val!='') + $alterList[] .= $val; + } + } + + //DB ABSTRACTION: we need to change the constraints to be a separate 'add' command, + //see http://www.postgresql.org/docs/8.1/static/sql-altertable.html + + if($alteredIndexes) foreach($alteredIndexes as $k => $v) { + $alterList[] .= "DROP INDEX \"$k\""; + $alterList[] .= "ADD ". $this->getIndexSqlDefinition($tableName, $k, $v); + } + + if($alterList) { + $alterations = implode(",\n", $alterList); + $this->query("ALTER TABLE \"$tableName\" " . $alterations); + } + } + + /* + * Creates an ALTER expression for a column in MS SQL + * + * @param $tableName Name of the table to be altered + * @param $colName Name of the column to be altered + * @param $colSpec String which contains conditions for a column + * @return string + */ + private function alterTableAlterColumn($tableName, $colName, $colSpec){ + // Alterations not implemented + return; + + // First, we split the column specifications into parts + // TODO: this returns an empty array for the following string: int(11) not null auto_increment + // on second thoughts, why is an auto_increment field being passed through? + + $pattern = '/^([\w()]+)\s?((?:not\s)?null)?\s?(default\s[\w\']+)?\s?(check\s[\w()\'",\s]+)?$/i'; + preg_match($pattern, $colSpec, $matches); + + /*if (isset($matches)) { + echo "sql:$colSpec
";
+			print_r($matches);
+			echo '
'; + }*/ + + //if($matches[1]=='serial8') + // return ''; + + if(isset($matches[1])) { + $alterCol = "ALTER COLUMN \"$colName\" TYPE $matches[1]\n"; + + // SET null / not null + if(!empty($matches[2])) $alterCol .= ",\nALTER COLUMN \"$colName\" SET $matches[2]"; + + // SET default (we drop it first, for reasons of precaution) + if(!empty($matches[3])) { + $alterCol .= ",\nALTER COLUMN \"$colName\" DROP DEFAULT"; + $alterCol .= ",\nALTER COLUMN \"$colName\" SET $matches[3]"; + } + + // SET check constraint (The constraint HAS to be dropped) + if(!empty($matches[4])) { + $alterCol .= ",\nDROP CONSTRAINT \"{$tableName}_{$colName}_check\""; + $alterCol .= ",\nADD CONSTRAINT \"{$tableName}_{$colName}_check\" $matches[4]"; + } + } + + return isset($alterCol) ? $alterCol : ''; + } + + public function renameTable($oldTableName, $newTableName) { + $this->query("ALTER TABLE \"$oldTableName\" RENAME \"$newTableName\""); + } + + /** + * Checks a table's integrity and repairs it if necessary. + * @var string $tableName The name of the table. + * @return boolean Return true if the table has integrity after the method is complete. + */ + public function checkAndRepairTable($tableName) { + /* + $this->runTableCheckCommand("VACUUM FULL \"$tableName\""); + */ + return true; + } + + /** + * Helper function used by checkAndRepairTable. + * @param string $sql Query to run. + * @return boolean Returns if the query returns a successful result. + */ + protected function runTableCheckCommand($sql) { + $testResults = $this->query($sql); + foreach($testResults as $testRecord) { + if(strtolower($testRecord['Msg_text']) != 'ok') { + return false; + } + } + return true; + } + + public function createField($tableName, $fieldName, $fieldSpec) { + $this->query("ALTER TABLE \"$tableName\" ADD \"$fieldName\" $fieldSpec"); + } + + /** + * Change the database type of the given field. + * @param string $tableName The name of the tbale the field is in. + * @param string $fieldName The name of the field to change. + * @param string $fieldSpec The new field specification + */ + public function alterField($tableName, $fieldName, $fieldSpec) { + $this->query("ALTER TABLE \"$tableName\" CHANGE \"$fieldName\" \"$fieldName\" $fieldSpec"); + } + + /** + * Change the database column name of the given field. + * + * @param string $tableName The name of the tbale the field is in. + * @param string $oldName The name of the field to change. + * @param string $newName The new name of the field + */ + public function renameField($tableName, $oldName, $newName) { + $fieldList = $this->fieldList($tableName); + if(array_key_exists($oldName, $fieldList)) { + $this->query("ALTER TABLE \"$tableName\" RENAME COLUMN \"$oldName\" TO \"$newName\""); + } + } + + public function fieldList($table) { + //Query from http://www.alberton.info/postgresql_meta_info.html + //This gets us more information than we need, but I've included it all for the moment.... + $fields = $this->query("SELECT ordinal_position, column_name, data_type, column_default, is_nullable, character_maximum_length, numeric_precision FROM information_schema.columns WHERE table_name = '$table' ORDER BY ordinal_position;"); + + $output = array(); + if($fields) foreach($fields as $field) { + switch($field){ + case 'bigint': + //We will assume that this is the ID column: + $output['column_name']=$this->IdColumn(); + break; + default: + $output[$field['column_name']] = $field; + } + + } + + return $output; + } + + /** + * Create an index on a table. + * @param string $tableName The name of the table. + * @param string $indexName The name of the index. + * @param string $indexSpec The specification of the index, see Database::requireIndex() for more details. + */ + public function createIndex($tableName, $indexName, $indexSpec) { + $this->query($this->getIndexSqlDefinition($tableName, $indexName, $indexSpec)); + } + + /* + * This takes the index spec which has been provided by a class (ie static $indexes = blah blah) + * and turns it into a proper string. + * Some indexes may be arrays, such as fulltext and unique indexes, and this allows database-specific + * arrays to be created. + */ + public function convertIndexSpec($indexSpec){ + if(is_array($indexSpec)){ + //Here we create a db-specific version of whatever index we need to create. + switch($indexSpec['type']){ + case 'fulltext': + $indexSpec='fulltext (' . str_replace(' ', '', $indexSpec['value']) . ')'; + break; + case 'unique': + $indexSpec='unique (' . $indexSpec['value'] . ')'; + break; + } + } + + return $indexSpec; + //return ''; + } + + protected function getIndexSqlDefinition($tableName, $indexName, $indexSpec) { + + if(!is_array($indexSpec)){ + $indexSpec=trim($indexSpec, '()'); + $bits=explode(',', $indexSpec); + $indexes="\"" . implode("\",\"", $bits) . "\""; + + return 'create index ix_' . $tableName . '_' . $indexName . " ON \"" . $tableName . "\" (" . $indexes . ");"; + } else { + //create a type-specific index + if($indexSpec['type']=='fulltext') + return 'create index ix_' . $tableName . '_' . $indexName . " ON \"" . $tableName . "\" USING gist(\"ts_" . $indexName . "\");"; + + if($indexSpec['type']=='unique') + return 'create unique index ix_' . $tableName . '_' . $indexName . " ON \"" . $tableName . "\" (\"" . $indexSpec['value'] . "\");"; + } + + } + + /** + * Alter an index on a table. + * @param string $tableName The name of the table. + * @param string $indexName The name of the index. + * @param string $indexSpec The specification of the index, see Database::requireIndex() for more details. + */ + public function alterIndex($tableName, $indexName, $indexSpec) { + $indexSpec = trim($indexSpec); + if($indexSpec[0] != '(') { + list($indexType, $indexFields) = explode(' ',$indexSpec,2); + } else { + $indexFields = $indexSpec; + } + + if(!$indexType) { + $indexType = "index"; + } + + $this->query("DROP INDEX $indexName"); + $this->query("ALTER TABLE \"$tableName\" ADD $indexType \"$indexName\" $indexFields"); + } + + /** + * Return the list of indexes in a table. + * @param string $table The table name. + * @return array + */ + public function indexList($table) { + //user_error("indexList not implemented", E_USER_WARNING); + return array(); + /* + + //Retrieve a list of indexes for the specified table + $indexes = DB::query("SELECT i.relname AS \"indexname\" + FROM pg_index x + JOIN pg_class c ON c.oid = x.indrelid + JOIN pg_class i ON i.oid = x.indexrelid + LEFT JOIN pg_namespace n ON n.oid = c.relnamespace + LEFT JOIN pg_tablespace t ON t.oid = i.reltablespace + WHERE c.relkind = 'r'::\"char\" AND i.relkind = 'i'::\"char\" + AND c.relname = '$table';"); + + $indexList[$index['indexname']]=$index['indexname']; + + return isset($indexList) ? $indexList : null; + */ + + } + + /** + * Returns a list of all the tables in the database. + * Table names will all be in lowercase. + * @return array + */ + public function tableList() { + foreach($this->query("EXEC sp_tables") as $record) { + $table = strtolower($record['TABLE_NAME']); + $tables[$table] = $table; + } + return isset($tables) ? $tables : null; + } + + /** + * Return the number of rows affected by the previous operation. + * @return int + */ + public function affectedRows() { + return mssql_rows_affected($this->dbConn); + } + + /** + * A function to return the field names and datatypes for the particular table + */ + public function tableDetails($tableName){ + user_error("tableDetails not implemented", E_USER_WARNING); + return array(); + /* + $query="SELECT a.attname as \"Column\", pg_catalog.format_type(a.atttypid, a.atttypmod) as \"Datatype\" FROM pg_catalog.pg_attribute a WHERE a.attnum > 0 AND NOT a.attisdropped AND a.attrelid = ( SELECT c.oid FROM pg_catalog.pg_class c LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace WHERE c.relname ~ '^($tableName)$' AND pg_catalog.pg_table_is_visible(c.oid));"; + $result=DB::query($query); + + $table=Array(); + while($row=pg_fetch_assoc($result)){ + $table[]=Array('Column'=>$row['Column'], 'DataType'=>$row['DataType']); + } + + return $table; + */ + } + + /** + * Return a boolean type-formatted string + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function boolean($values, $asDbValue=false){ + //Annoyingly, we need to do a good ol' fashioned switch here: + ($values['default']) ? $default='1' : $default='0'; + return 'bit not null default ' . $default; + } + + /** + * Return a date type-formatted string + * For MySQL, we simply return the word 'date', no other parameters are necessary + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function date($values){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'date'); + //DB::requireField($this->tableName, $this->name, "date"); + + return 'date'; + } + + /** + * Return a decimal type-formatted string + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function decimal($values, $asDbValue=false){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'decimal', 'precision'=>"$this->wholeSize,$this->decimalSize"); + //DB::requireField($this->tableName, $this->name, "decimal($this->wholeSize,$this->decimalSize)"); + + // Avoid empty strings being put in the db + if($values['precision'] == '') { + $precision = 1; + } else { + $precision = $values['precision']; + } + + if($asDbValue) + return Array('data_type'=>'numeric', 'numeric_precision'=>'9'); + else return 'decimal(' . $precision . ') not null'; + } + + /** + * Return a enum type-formatted string + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function enum($values){ + //Enums are a bit different. We'll be creating a varchar(255) with a constraint of all the usual enum options. + //NOTE: In this one instance, we are including the table name in the values array + + return "varchar(255) not null default '" . $values['default'] . "' check (\"" . $values['name'] . "\" in ('" . implode('\', \'', $values['enums']) . "'))"; + + } + + /** + * Return a float type-formatted string + * For MySQL, we simply return the word 'date', no other parameters are necessary + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function float($values, $asDbValue=false){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'float'); + //DB::requireField($this->tableName, $this->name, "float"); + + if($asDbValue) + return Array('data_type'=>'double precision'); + else return 'float'; + } + + /** + * Return a int type-formatted string + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function int($values, $asDbValue=false){ + //We'll be using an 8 digit precision to keep it in line with the serial8 datatype for ID columns + + if($asDbValue) + return Array('data_type'=>'numeric', 'numeric_precision'=>'8'); + else + return 'numeric(8) not null default ' . (int)$values['default']; + } + + /** + * Return a datetime type-formatted string + * For MS SQL, we simply return the word 'timestamp', no other parameters are necessary + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function ssdatetime($values, $asDbValue=false){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'datetime'); + //DB::requireField($this->tableName, $this->name, $values); + + if($asDbValue) + return Array('data_type'=>'datetime without time zone'); + else + return 'datetime'; + } + + /** + * Return a text type-formatted string + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function text($values, $asDbValue=false){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'mediumtext', 'character set'=>'utf8', 'collate'=>'utf8_general_ci'); + //DB::requireField($this->tableName, $this->name, "mediumtext character set utf8 collate utf8_general_ci"); + + if($asDbValue) + return Array('data_type'=>'text'); + else + return 'text'; + } + + /** + * Return a time type-formatted string + * For MySQL, we simply return the word 'time', no other parameters are necessary + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function time($values){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'time'); + //DB::requireField($this->tableName, $this->name, "time"); + + return 'time'; + } + + /** + * Return a varchar type-formatted string + * + * @params array $values Contains a tokenised list of info about this data type + * @return string + */ + public function varchar($values, $asDbValue=false){ + //For reference, this is what typically gets passed to this function: + //$parts=Array('datatype'=>'varchar', 'precision'=>$this->size, 'character set'=>'utf8', 'collate'=>'utf8_general_ci'); + //DB::requireField($this->tableName, $this->name, "varchar($this->size) character set utf8 collate utf8_general_ci"); + if($asDbValue) + return Array('data_type'=>'character varying', 'character_maximum_length'=>'255'); + else + return 'varchar(' . $values['precision'] . ')'; + } + + /* + * Return a 4 digit numeric type. MySQL has a proprietary 'Year' type. + */ + public function year($values, $asDbValue=false){ + if($asDbValue) + return Array('data_type'=>'numeric', 'numeric_precision'=>'4'); + else return 'numeric(4)'; + } + + function escape_character($escape=false){ + if($escape) + return "\\\""; + else + return "\""; + } + + /** + * Create a fulltext search datatype for MySQL + * + * @param array $spec + */ + function fulltext($table, $spec){ + //$spec['name'] is the column we've created that holds all the words we want to index. + //This is a coalesced collection of multiple columns if necessary + $spec='create index ix_' . $table . '_' . $spec['name'] . ' on ' . $table . ' using gist(' . $spec['name'] . ');'; + + return $spec; + } + + /** + * This returns the column which is the primary key for each table + * In Postgres, it is a SERIAL8, which is the equivalent of an auto_increment + * + * @return string + */ + function IdColumn($asDbValue=false){ + return 'bigint identity(1,1)'; + } + + /** + * Returns true if this table exists + * @todo Make a proper implementation + */ + function hasTable($tableName) { + return true; + } + + /** + * Return enum values for the given field + * @todo Make a proper implementation + */ + function enumValuesForField($tableName, $fieldName) { + return array('SiteTree','Page'); + } + + /** + * Convert a SQLQuery object into a SQL statement + * @todo There is a lot of duplication between this and MySQLDatabase::sqlQueryToString(). Perhaps they could both call a common + * helper function in Database? + */ + public function sqlQueryToString(SQLQuery $sqlQuery) { + if (!$sqlQuery->from) return ''; + $distinct = $sqlQuery->distinct ? "DISTINCT " : ""; + if($sqlQuery->delete) { + $text = "DELETE "; + } else if($sqlQuery->select) { + $text = "SELECT $distinct" . implode(", ", $sqlQuery->select); + } + $text .= " FROM " . implode(" ", $sqlQuery->from); + + if($sqlQuery->where) $text .= " WHERE (" . $sqlQuery->getFilter(). ")"; + if($sqlQuery->groupby) $text .= " GROUP BY " . implode(", ", $sqlQuery->groupby); + if($sqlQuery->having) $text .= " HAVING ( " . implode(" ) AND ( ", $sqlQuery->having) . " )"; + if($sqlQuery->orderby) $text .= " ORDER BY " . $sqlQuery->orderby; + + // Limit not implemented + /* + if($sqlQuery->limit) { + $limit = $sqlQuery->limit; + // Pass limit as array or SQL string value + if(is_array($limit)) { + if(!array_key_exists('limit',$limit)) user_error('SQLQuery::limit(): Wrong format for $limit', E_USER_ERROR); + + if(isset($limit['start']) && is_numeric($limit['start']) && isset($limit['limit']) && is_numeric($limit['limit'])) { + $combinedLimit = (int)$limit['start'] . ',' . (int)$limit['limit']; + } elseif(isset($limit['limit']) && is_numeric($limit['limit'])) { + $combinedLimit = (int)$limit['limit']; + } else { + $combinedLimit = false; + } + if(!empty($combinedLimit)) $this->limit = $combinedLimit; + + } else { + $text .= " LIMIT " . $sqlQuery->limit; + } + } + */ + + return $text; + } +} + +/** + * A result-set from a MySQL database. + * @package sapphire + * @subpackage model + */ +class MSSQLQuery extends Query { + /** + * The MySQLDatabase object that created this result set. + * @var MySQLDatabase + */ + private $database; + + /** + * The internal MySQL handle that points to the result set. + * @var resource + */ + private $handle; + + /** + * Hook the result-set given into a Query class, suitable for use by sapphire. + * @param database The database object that created this query. + * @param handle the internal mysql handle that is points to the resultset. + */ + public function __construct(MSSQLDatabase $database, $handle) { + $this->database = $database; + $this->handle = $handle; + parent::__construct(); + } + + public function __destroy() { + //mysql_free_result($this->handle); + } + + public function seek($row) { + //return mysql_data_seek($this->handle, $row); + //This is unnecessary in postgres. You can just provide a row number with the fetch + //command. + } + + public function numRecords() { + return mssql_num_rows($this->handle); + } + + public function nextRecord() { + // Coalesce rather than replace common fields. + if($data = mssql_fetch_row($this->handle)) { + foreach($data as $columnIdx => $value) { + $columnName = mssql_field_name($this->handle, $columnIdx); + // $value || !$ouput[$columnName] means that the *last* occurring value is shown + // !$ouput[$columnName] means that the *first* occurring value is shown + if(isset($value) || !isset($output[$columnName])) { + $output[$columnName] = $value; + } + } + return $output; + } else { + return false; + } + } + + +} + +?> \ No newline at end of file