WindowExpressionTest.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. <?php
  2. declare(strict_types=1);
  3. /**
  4. * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
  5. * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  6. *
  7. * Licensed under The Open Group Test Suite License
  8. * Redistributions of files must retain the above copyright notice.
  9. *
  10. * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  11. * @link https://cakephp.org CakePHP(tm) Project
  12. * @since 4.1.0
  13. * @license https://opensource.org/licenses/mit-license.php MIT License
  14. */
  15. namespace Cake\Test\TestCase\Database\Expression;
  16. use Cake\Database\Expression\AggregateExpression;
  17. use Cake\Database\Expression\IdentifierExpression;
  18. use Cake\Database\Expression\OrderByExpression;
  19. use Cake\Database\Expression\OrderClauseExpression;
  20. use Cake\Database\Expression\QueryExpression;
  21. use Cake\Database\Expression\WindowExpression;
  22. use Cake\Database\ValueBinder;
  23. use Cake\TestSuite\TestCase;
  24. /**
  25. * Tests WindowExpression class
  26. */
  27. class WindowExpressionTest extends TestCase
  28. {
  29. /**
  30. * Tests an empty window expression
  31. */
  32. public function testEmptyWindow(): void
  33. {
  34. $w = new WindowExpression();
  35. $this->assertSame('', $w->sql(new ValueBinder()));
  36. $w->partition('')->orderBy([]);
  37. $this->assertSame('', $w->sql(new ValueBinder()));
  38. }
  39. /**
  40. * Tests windows with partitions
  41. */
  42. public function testPartitions(): void
  43. {
  44. $w = (new WindowExpression())->partition('test');
  45. $this->assertEqualsSql(
  46. 'PARTITION BY test',
  47. $w->sql(new ValueBinder())
  48. );
  49. $w->partition(new IdentifierExpression('identifier'));
  50. $this->assertEqualsSql(
  51. 'PARTITION BY test, identifier',
  52. $w->sql(new ValueBinder())
  53. );
  54. $w = (new WindowExpression())->partition(new AggregateExpression('MyAggregate', ['param']));
  55. $this->assertEqualsSql(
  56. 'PARTITION BY MyAggregate(:param0)',
  57. $w->sql(new ValueBinder())
  58. );
  59. $w = (new WindowExpression())->partition(function (QueryExpression $expr) {
  60. return $expr->add(new AggregateExpression('MyAggregate', ['param']));
  61. });
  62. $this->assertEqualsSql(
  63. 'PARTITION BY MyAggregate(:param0)',
  64. $w->sql(new ValueBinder())
  65. );
  66. }
  67. /**
  68. * Tests windows with order by
  69. */
  70. public function testOrderBy(): void
  71. {
  72. $w = (new WindowExpression())->orderBy('test');
  73. $this->assertEqualsSql(
  74. 'ORDER BY test',
  75. $w->sql(new ValueBinder())
  76. );
  77. $w->orderBy(['test2' => 'DESC']);
  78. $this->assertEqualsSql(
  79. 'ORDER BY test, test2 DESC',
  80. $w->sql(new ValueBinder())
  81. );
  82. $w->partition('test');
  83. $this->assertEqualsSql(
  84. 'PARTITION BY test ORDER BY test, test2 DESC',
  85. $w->sql(new ValueBinder())
  86. );
  87. $w = (new WindowExpression())
  88. ->orderBy(function () {
  89. return 'test';
  90. })
  91. ->orderBy(function (QueryExpression $expr) {
  92. return [$expr->add('test2'), new OrderClauseExpression(new IdentifierExpression('test3'), 'DESC')];
  93. });
  94. $this->assertEqualsSql(
  95. 'ORDER BY test, test2, test3 DESC',
  96. $w->sql(new ValueBinder())
  97. );
  98. }
  99. public function testOrderDeprecated(): void
  100. {
  101. $w = (new WindowExpression())->order('test');
  102. $this->assertEqualsSql(
  103. 'ORDER BY test',
  104. $w->sql(new ValueBinder())
  105. );
  106. }
  107. /**
  108. * Tests windows with range frames
  109. */
  110. public function testRange(): void
  111. {
  112. $w = (new WindowExpression())->range(null);
  113. $this->assertEqualsSql(
  114. 'RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW',
  115. $w->sql(new ValueBinder())
  116. );
  117. $w = (new WindowExpression())->range(0);
  118. $this->assertEqualsSql(
  119. 'RANGE BETWEEN CURRENT ROW AND CURRENT ROW',
  120. $w->sql(new ValueBinder())
  121. );
  122. $w = (new WindowExpression())->range(2);
  123. $this->assertEqualsSql(
  124. 'RANGE BETWEEN 2 PRECEDING AND CURRENT ROW',
  125. $w->sql(new ValueBinder())
  126. );
  127. $w = (new WindowExpression())->range(null, null);
  128. $this->assertEqualsSql(
  129. 'RANGE BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING',
  130. $w->sql(new ValueBinder())
  131. );
  132. $w = (new WindowExpression())->range(0, null);
  133. $this->assertEqualsSql(
  134. 'RANGE BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING',
  135. $w->sql(new ValueBinder())
  136. );
  137. $w = (new WindowExpression())->range(0, 0);
  138. $this->assertEqualsSql(
  139. 'RANGE BETWEEN CURRENT ROW AND CURRENT ROW',
  140. $w->sql(new ValueBinder())
  141. );
  142. $w = (new WindowExpression())->range(1, 2);
  143. $this->assertEqualsSql(
  144. 'RANGE BETWEEN 1 PRECEDING AND 2 FOLLOWING',
  145. $w->sql(new ValueBinder())
  146. );
  147. $w = (new WindowExpression())->range("'1 day'", "'10 days'");
  148. $this->assertRegExpSql(
  149. "RANGE BETWEEN '1 day' PRECEDING AND '10 days' FOLLOWING",
  150. $w->sql(new ValueBinder())
  151. );
  152. $w = (new WindowExpression())->range(new QueryExpression("'1 day'"), new QueryExpression("'10 days'"));
  153. $this->assertRegExpSql(
  154. "RANGE BETWEEN '1 day' PRECEDING AND '10 days' FOLLOWING",
  155. $w->sql(new ValueBinder())
  156. );
  157. $w = (new WindowExpression())->frame(
  158. WindowExpression::RANGE,
  159. 2,
  160. WindowExpression::PRECEDING,
  161. 1,
  162. WindowExpression::PRECEDING
  163. );
  164. $this->assertEqualsSql(
  165. 'RANGE BETWEEN 2 PRECEDING AND 1 PRECEDING',
  166. $w->sql(new ValueBinder())
  167. );
  168. }
  169. /**
  170. * Tests windows with rows frames
  171. */
  172. public function testRows(): void
  173. {
  174. $w = (new WindowExpression())->rows(null);
  175. $this->assertEqualsSql(
  176. 'ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW',
  177. $w->sql(new ValueBinder())
  178. );
  179. $w = (new WindowExpression())->rows(0);
  180. $this->assertEqualsSql(
  181. 'ROWS BETWEEN CURRENT ROW AND CURRENT ROW',
  182. $w->sql(new ValueBinder())
  183. );
  184. $w = (new WindowExpression())->rows(2);
  185. $this->assertEqualsSql(
  186. 'ROWS BETWEEN 2 PRECEDING AND CURRENT ROW',
  187. $w->sql(new ValueBinder())
  188. );
  189. $w = (new WindowExpression())->rows(null, null);
  190. $this->assertEqualsSql(
  191. 'ROWS BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING',
  192. $w->sql(new ValueBinder())
  193. );
  194. $w = (new WindowExpression())->rows(0, null);
  195. $this->assertEqualsSql(
  196. 'ROWS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING',
  197. $w->sql(new ValueBinder())
  198. );
  199. $w = (new WindowExpression())->rows(0, 0);
  200. $this->assertEqualsSql(
  201. 'ROWS BETWEEN CURRENT ROW AND CURRENT ROW',
  202. $w->sql(new ValueBinder())
  203. );
  204. $w = (new WindowExpression())->rows(1, 2);
  205. $this->assertEqualsSql(
  206. 'ROWS BETWEEN 1 PRECEDING AND 2 FOLLOWING',
  207. $w->sql(new ValueBinder())
  208. );
  209. $w = (new WindowExpression())->frame(
  210. WindowExpression::ROWS,
  211. 2,
  212. WindowExpression::PRECEDING,
  213. 1,
  214. WindowExpression::PRECEDING
  215. );
  216. $b = new ValueBinder();
  217. $this->assertEqualsSql(
  218. 'ROWS BETWEEN 2 PRECEDING AND 1 PRECEDING',
  219. $w->sql($b)
  220. );
  221. }
  222. /**
  223. * Tests windows with groups frames
  224. */
  225. public function testGroups(): void
  226. {
  227. $w = (new WindowExpression())->groups(null);
  228. $this->assertEqualsSql(
  229. 'GROUPS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW',
  230. $w->sql(new ValueBinder())
  231. );
  232. $w = (new WindowExpression())->groups(0);
  233. $this->assertEqualsSql(
  234. 'GROUPS BETWEEN CURRENT ROW AND CURRENT ROW',
  235. $w->sql(new ValueBinder())
  236. );
  237. $w = (new WindowExpression())->groups(2);
  238. $this->assertEqualsSql(
  239. 'GROUPS BETWEEN 2 PRECEDING AND CURRENT ROW',
  240. $w->sql(new ValueBinder())
  241. );
  242. $w = (new WindowExpression())->groups(null, null);
  243. $this->assertEqualsSql(
  244. 'GROUPS BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING',
  245. $w->sql(new ValueBinder())
  246. );
  247. $w = (new WindowExpression())->groups(0, null);
  248. $this->assertEqualsSql(
  249. 'GROUPS BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING',
  250. $w->sql(new ValueBinder())
  251. );
  252. $w = (new WindowExpression())->groups(0, 0);
  253. $this->assertEqualsSql(
  254. 'GROUPS BETWEEN CURRENT ROW AND CURRENT ROW',
  255. $w->sql(new ValueBinder())
  256. );
  257. $w = (new WindowExpression())->groups(1, 2);
  258. $this->assertEqualsSql(
  259. 'GROUPS BETWEEN 1 PRECEDING AND 2 FOLLOWING',
  260. $w->sql(new ValueBinder())
  261. );
  262. $w = (new WindowExpression())->frame(
  263. WindowExpression::GROUPS,
  264. 2,
  265. WindowExpression::PRECEDING,
  266. 1,
  267. WindowExpression::PRECEDING
  268. );
  269. $b = new ValueBinder();
  270. $this->assertEqualsSql(
  271. 'GROUPS BETWEEN 2 PRECEDING AND 1 PRECEDING',
  272. $w->sql($b)
  273. );
  274. }
  275. /**
  276. * Tests windows with frame exclusion
  277. */
  278. public function testExclusion(): void
  279. {
  280. $w = (new WindowExpression())->excludeCurrent();
  281. $this->assertEqualsSql(
  282. '',
  283. $w->sql(new ValueBinder())
  284. );
  285. $w = (new WindowExpression())->range(null)->excludeCurrent();
  286. $this->assertEqualsSql(
  287. 'RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW EXCLUDE CURRENT ROW',
  288. $w->sql(new ValueBinder())
  289. );
  290. $w = (new WindowExpression())->range(null)->excludeGroup();
  291. $this->assertEqualsSql(
  292. 'RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW EXCLUDE GROUP',
  293. $w->sql(new ValueBinder())
  294. );
  295. $w = (new WindowExpression())->range(null)->excludeTies();
  296. $this->assertEqualsSql(
  297. 'RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW EXCLUDE TIES',
  298. $w->sql(new ValueBinder())
  299. );
  300. }
  301. /**
  302. * Tests windows with partition, order and frames
  303. */
  304. public function testCombined(): void
  305. {
  306. $w = (new WindowExpression())->partition('test')->range(null);
  307. $this->assertEqualsSql(
  308. 'PARTITION BY test RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW',
  309. $w->sql(new ValueBinder())
  310. );
  311. $w = (new WindowExpression())->orderBy('test')->range(null);
  312. $this->assertEqualsSql(
  313. 'ORDER BY test RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW',
  314. $w->sql(new ValueBinder())
  315. );
  316. $w = (new WindowExpression())->partition('test')->orderBy('test')->range(null);
  317. $this->assertEqualsSql(
  318. 'PARTITION BY test ORDER BY test RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW',
  319. $w->sql(new ValueBinder())
  320. );
  321. $w = (new WindowExpression())->partition('test')->orderBy('test')->range(null)->excludeCurrent();
  322. $this->assertEqualsSql(
  323. 'PARTITION BY test ORDER BY test RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW EXCLUDE CURRENT ROW',
  324. $w->sql(new ValueBinder())
  325. );
  326. }
  327. /**
  328. * Tests named windows
  329. */
  330. public function testNamedWindow(): void
  331. {
  332. $w = new WindowExpression();
  333. $this->assertFalse($w->isNamedOnly());
  334. $w->name('name');
  335. $this->assertTrue($w->isNamedOnly());
  336. $this->assertEqualsSql(
  337. 'name',
  338. $w->sql(new ValueBinder())
  339. );
  340. $w->name('new_name');
  341. $this->assertEqualsSql(
  342. 'new_name',
  343. $w->sql(new ValueBinder())
  344. );
  345. $w->orderBy('test');
  346. $this->assertFalse($w->isNamedOnly());
  347. $this->assertEqualsSql(
  348. 'new_name ORDER BY test',
  349. $w->sql(new ValueBinder())
  350. );
  351. }
  352. /**
  353. * Tests traversing window expressions.
  354. */
  355. public function testTraverse(): void
  356. {
  357. $w = (new WindowExpression('test1'))
  358. ->partition('test2')
  359. ->orderBy('test3')
  360. ->range(new QueryExpression("'1 day'"));
  361. $expressions = [];
  362. $w->traverse(function ($expression) use (&$expressions): void {
  363. $expressions[] = $expression;
  364. });
  365. $this->assertEquals(new IdentifierExpression('test1'), $expressions[0]);
  366. $this->assertEquals(new IdentifierExpression('test2'), $expressions[1]);
  367. $this->assertEquals((new OrderByExpression())->add('test3'), $expressions[2]);
  368. $this->assertEquals(new QueryExpression("'1 day'"), $expressions[3]);
  369. $w->range(new QueryExpression("'1 day'"), new QueryExpression("'10 days'"));
  370. $expressions = [];
  371. $w->traverse(function ($expression) use (&$expressions): void {
  372. $expressions[] = $expression;
  373. });
  374. $this->assertEquals(new QueryExpression("'1 day'"), $expressions[3]);
  375. $this->assertEquals(new QueryExpression("'10 days'"), $expressions[4]);
  376. }
  377. /**
  378. * Tests cloning window expressions
  379. */
  380. public function testCloning(): void
  381. {
  382. $w1 = (new WindowExpression())->name('test');
  383. $w2 = (clone $w1)->name('test2');
  384. $this->assertNotSame($w1->sql(new ValueBinder()), $w2->sql(new ValueBinder()));
  385. $w1 = (new WindowExpression())->partition('test');
  386. $w2 = (clone $w1)->partition('new');
  387. $this->assertNotSame($w1->sql(new ValueBinder()), $w2->sql(new ValueBinder()));
  388. $w1 = (new WindowExpression())->orderBy('test');
  389. $w2 = (clone $w1)->orderBy('new');
  390. $this->assertNotSame($w1->sql(new ValueBinder()), $w2->sql(new ValueBinder()));
  391. $w1 = (new WindowExpression())->rows(0, null);
  392. $w2 = (clone $w1)->rows(0, 0);
  393. $this->assertNotSame($w1->sql(new ValueBinder()), $w2->sql(new ValueBinder()));
  394. }
  395. }