sqlserver.php 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. <?php namespace Laravel\Database\Query\Grammars;
  2. use Laravel\Database\Query;
  3. class SQLServer extends Grammar {
  4. /**
  5. * The keyword identifier for the database system.
  6. *
  7. * @var string
  8. */
  9. protected $wrapper = '[%s]';
  10. /**
  11. * Compile a SQL SELECT statement from a Query instance.
  12. *
  13. * @param Query $query
  14. * @return string
  15. */
  16. public function select(Query $query)
  17. {
  18. $sql = parent::components($query);
  19. // SQL Server does not currently implement an "OFFSET" type keyword, so we
  20. // actually have to generate the ANSI standard SQL for doing offset like
  21. // functionality. In the next version of SQL Server, an OFFSET like
  22. // keyword is included for convenience.
  23. if ($query->offset > 0)
  24. {
  25. return $this->ansi_offset($query, $sql);
  26. }
  27. // Once all of the clauses have been compiled, we can join them all as
  28. // one statement. Any segments that are null or an empty string will
  29. // be removed from the array of clauses before they are imploded.
  30. return $this->concatenate($sql);
  31. }
  32. /**
  33. * Compile the SELECT clause for a query.
  34. *
  35. * @param Query $query
  36. * @return string
  37. */
  38. protected function selects(Query $query)
  39. {
  40. if ( ! is_null($query->aggregate)) return;
  41. $select = ($query->distinct) ? 'SELECT DISTINCT ' : 'SELECT ';
  42. // Instead of using a "LIMIT" keyword, SQL Server uses the "TOP"
  43. // keyword within the SELECT statement. So, if we have a limit,
  44. // we will add it here.
  45. //
  46. // We will not add the TOP clause if there is an offset however,
  47. // since we will have to handle offsets using the ANSI syntax
  48. // and will need to remove the TOP clause in that situation.
  49. if ($query->limit > 0 and $query->offset <= 0)
  50. {
  51. $select .= 'TOP '.$query->limit.' ';
  52. }
  53. return $select.$this->columnize($query->selects);
  54. }
  55. /**
  56. * Generate the ANSI standard SQL for an offset clause.
  57. *
  58. * @param Query $query
  59. * @param array $components
  60. * @return array
  61. */
  62. protected function ansi_offset(Query $query, $components)
  63. {
  64. // An ORDER BY clause is required to make this offset query
  65. // work, so if one doesn't exist, we'll just create a dummy
  66. // clause to satisfy the database.
  67. if ( ! isset($components['orderings']))
  68. {
  69. $components['orderings'] = 'ORDER BY (SELECT 0)';
  70. }
  71. // We need to add the row number to the query results so we
  72. // can compare it against the offset and limit values given
  73. // for the statement. To do that we'll add an expression to
  74. // the select statement for the row number.
  75. $orderings = $components['orderings'];
  76. $components['selects'] .= ", ROW_NUMBER() OVER ({$orderings}) AS RowNum";
  77. unset($components['orderings']);
  78. $start = $query->offset + 1;
  79. // Next we need to calculate the constraint that should be
  80. // placed on the row number to get the correct offset and
  81. // limit on the query. If a limit has not been set, we'll
  82. // only add a constraint to handle offset.
  83. if ($query->limit > 0)
  84. {
  85. $finish = $query->offset + $query->limit;
  86. $constraint = "BETWEEN {$start} AND {$finish}";
  87. }
  88. else
  89. {
  90. $constraint = ">= {$start}";
  91. }
  92. // Now, we're finally ready to build the final SQL query.
  93. // We'll create a common table expression with the query
  94. // and then select all of the results from it where the
  95. // row number is between oru given limit and offset.
  96. $sql = $this->concatenate($components);
  97. return "SELECT * FROM ($sql) AS TempTable WHERE RowNum {$constraint}";
  98. }
  99. /**
  100. * Compile the LIMIT clause for a query.
  101. *
  102. * @param Query $query
  103. * @return string
  104. */
  105. protected function limit(Query $query)
  106. {
  107. return '';
  108. }
  109. /**
  110. * Compile the OFFSET clause for a query.
  111. *
  112. * @param Query $query
  113. * @return string
  114. */
  115. protected function offset(Query $query)
  116. {
  117. return '';
  118. }
  119. }