| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141 | <?php namespace Laravel\Database\Query\Grammars;use Laravel\Database\Query;class SQLServer extends Grammar {	/**	 * The keyword identifier for the database system.	 *	 * @var string	 */	protected $wrapper = '[%s]';	/**	 * Compile a SQL SELECT statement from a Query instance.	 *	 * @param  Query   $query	 * @return string	 */	public function select(Query $query)	{		$sql = parent::components($query);		// SQL Server does not currently implement an "OFFSET" type keyword, so we		// actually have to generate the ANSI standard SQL for doing offset like		// functionality. In the next version of SQL Server, an OFFSET like		// keyword is included for convenience.		if ($query->offset > 0)		{			return $this->ansi_offset($query, $sql);		}		// Once all of the clauses have been compiled, we can join them all as		// one statement. Any segments that are null or an empty string will		// be removed from the array of clauses before they are imploded.		return $this->concatenate($sql);	}	/**	 * Compile the SELECT clause for a query.	 *	 * @param  Query   $query	 * @return string	 */	protected function selects(Query $query)	{		if ( ! is_null($query->aggregate)) return;		$select = ($query->distinct) ? 'SELECT DISTINCT ' : 'SELECT ';		// Instead of using a "LIMIT" keyword, SQL Server uses the "TOP"		// keyword within the SELECT statement. So, if we have a limit,		// we will add it here.		//		// We will not add the TOP clause if there is an offset however,		// since we will have to handle offsets using the ANSI syntax		// and will need to remove the TOP clause in that situation.		if ($query->limit > 0 and $query->offset <= 0)		{			$select .= 'TOP '.$query->limit.' ';		}		return $select.$this->columnize($query->selects);	}	/**	 * Generate the ANSI standard SQL for an offset clause.	 *	 * @param  Query  $query	 * @param  array  $components	 * @return array	 */	protected function ansi_offset(Query $query, $components)	{		// An ORDER BY clause is required to make this offset query		// work, so if one doesn't exist, we'll just create a dummy		// clause to satisfy the database.		if ( ! isset($components['orderings']))		{			$components['orderings'] = 'ORDER BY (SELECT 0)';		}		// We need to add the row number to the query results so we		// can compare it against the offset and limit values given		// for the statement. To do that we'll add an expression to		// the select statement for the row number.		$orderings = $components['orderings'];		$components['selects'] .= ", ROW_NUMBER() OVER ({$orderings}) AS RowNum";		unset($components['orderings']);		$start = $query->offset + 1;		// Next we need to calculate the constraint that should be		// placed on the row number to get the correct offset and		// limit on the query. If a limit has not been set, we'll		// only add a constraint to handle offset.		if ($query->limit > 0)		{			$finish = $query->offset + $query->limit;			$constraint = "BETWEEN {$start} AND {$finish}";		}		else		{			$constraint = ">= {$start}";		}		// Now, we're finally ready to build the final SQL query.		// We'll create a common table expression with the query		// and then select all of the results from it where the		// row number is between oru given limit and offset.		$sql = $this->concatenate($components);		return "SELECT * FROM ($sql) AS TempTable WHERE RowNum {$constraint}";	}	/**	 * Compile the LIMIT clause for a query.	 *	 * @param  Query   $query	 * @return string	 */	protected function limit(Query $query)	{		return '';	}	/**	 * Compile the OFFSET clause for a query.	 *	 * @param  Query   $query	 * @return string	 */	protected function offset(Query $query)	{		return '';	}}
 |