sl@0
|
1 |
# 2007 Dec 4
|
sl@0
|
2 |
#
|
sl@0
|
3 |
# The author disclaims copyright to this source code. In place of
|
sl@0
|
4 |
# a legal notice, here is a blessing:
|
sl@0
|
5 |
#
|
sl@0
|
6 |
# May you do good and not evil.
|
sl@0
|
7 |
# May you find forgiveness for yourself and forgive others.
|
sl@0
|
8 |
# May you share freely, never taking more than you give.
|
sl@0
|
9 |
#
|
sl@0
|
10 |
#***********************************************************************
|
sl@0
|
11 |
#
|
sl@0
|
12 |
# This file is to test that the issues surrounding expressions in
|
sl@0
|
13 |
# ORDER BY clauses on compound SELECT statements raised by ticket
|
sl@0
|
14 |
# #2822 have been dealt with.
|
sl@0
|
15 |
#
|
sl@0
|
16 |
# $Id: tkt2822.test,v 1.6 2008/08/20 16:35:10 drh Exp $
|
sl@0
|
17 |
#
|
sl@0
|
18 |
|
sl@0
|
19 |
set testdir [file dirname $argv0]
|
sl@0
|
20 |
source $testdir/tester.tcl
|
sl@0
|
21 |
|
sl@0
|
22 |
ifcapable !compound {
|
sl@0
|
23 |
finish_test
|
sl@0
|
24 |
return
|
sl@0
|
25 |
}
|
sl@0
|
26 |
|
sl@0
|
27 |
# The ORDER BY matching algorithm is three steps:
|
sl@0
|
28 |
#
|
sl@0
|
29 |
# (1) If the ORDER BY term is an integer constant i, then
|
sl@0
|
30 |
# sort by the i-th column of the result set.
|
sl@0
|
31 |
#
|
sl@0
|
32 |
# (2) If the ORDER BY term is an identifier (not x.y or x.y.z
|
sl@0
|
33 |
# but simply x) then look for a column alias with the same
|
sl@0
|
34 |
# name. If found, then sort by that column.
|
sl@0
|
35 |
#
|
sl@0
|
36 |
# (3) Evaluate the term as an expression and sort by the
|
sl@0
|
37 |
# value of the expression.
|
sl@0
|
38 |
#
|
sl@0
|
39 |
# For a compound SELECT the rules are modified slightly.
|
sl@0
|
40 |
# In the third rule, the expression must exactly match one
|
sl@0
|
41 |
# of the result columns. The sequences of three rules is
|
sl@0
|
42 |
# attempted first on the left-most SELECT. If that doesn't
|
sl@0
|
43 |
# work, we move to the right, one by one.
|
sl@0
|
44 |
#
|
sl@0
|
45 |
# Rule (3) is not in standard SQL - it is an SQLite extension,
|
sl@0
|
46 |
# though one copied from PostgreSQL. The rule for compound
|
sl@0
|
47 |
# queries where a search is made of SELECTs to the right
|
sl@0
|
48 |
# if the left-most SELECT does not match is not a part of
|
sl@0
|
49 |
# standard SQL either. This extension is unique to SQLite
|
sl@0
|
50 |
# as far as we know.
|
sl@0
|
51 |
#
|
sl@0
|
52 |
# Rule (2) was added by the changes ticket #2822. Prior to
|
sl@0
|
53 |
# that changes, SQLite did not support rule (2), making it
|
sl@0
|
54 |
# technically in violation of standard SQL semantics.
|
sl@0
|
55 |
# No body noticed because rule (3) has the same effect as
|
sl@0
|
56 |
# rule (2) except in some obscure cases.
|
sl@0
|
57 |
#
|
sl@0
|
58 |
|
sl@0
|
59 |
|
sl@0
|
60 |
# Test plan:
|
sl@0
|
61 |
#
|
sl@0
|
62 |
# tkt2822-1.* - Simple identifier as ORDER BY expression.
|
sl@0
|
63 |
# tkt2822-2.* - More complex ORDER BY expressions.
|
sl@0
|
64 |
|
sl@0
|
65 |
do_test tkt2822-0.1 {
|
sl@0
|
66 |
execsql {
|
sl@0
|
67 |
CREATE TABLE t1(a, b, c);
|
sl@0
|
68 |
CREATE TABLE t2(a, b, c);
|
sl@0
|
69 |
|
sl@0
|
70 |
INSERT INTO t1 VALUES(1, 3, 9);
|
sl@0
|
71 |
INSERT INTO t1 VALUES(3, 9, 27);
|
sl@0
|
72 |
INSERT INTO t1 VALUES(5, 15, 45);
|
sl@0
|
73 |
|
sl@0
|
74 |
INSERT INTO t2 VALUES(2, 6, 18);
|
sl@0
|
75 |
INSERT INTO t2 VALUES(4, 12, 36);
|
sl@0
|
76 |
INSERT INTO t2 VALUES(6, 18, 54);
|
sl@0
|
77 |
}
|
sl@0
|
78 |
} {}
|
sl@0
|
79 |
|
sl@0
|
80 |
# Test the "ORDER BY <integer>" syntax.
|
sl@0
|
81 |
#
|
sl@0
|
82 |
do_test tkt2822-1.1 {
|
sl@0
|
83 |
execsql {
|
sl@0
|
84 |
SELECT a, b, c FROM t1 UNION ALL SELECT a, b, c FROM t2 ORDER BY 1;
|
sl@0
|
85 |
}
|
sl@0
|
86 |
} {1 3 9 2 6 18 3 9 27 4 12 36 5 15 45 6 18 54}
|
sl@0
|
87 |
do_test tkt2822-1.2 {
|
sl@0
|
88 |
execsql {
|
sl@0
|
89 |
SELECT a, CAST (b AS TEXT), c FROM t1
|
sl@0
|
90 |
UNION ALL
|
sl@0
|
91 |
SELECT a, b, c FROM t2
|
sl@0
|
92 |
ORDER BY 2;
|
sl@0
|
93 |
}
|
sl@0
|
94 |
} {2 6 18 4 12 36 6 18 54 5 15 45 1 3 9 3 9 27}
|
sl@0
|
95 |
|
sl@0
|
96 |
# Test the "ORDER BY <identifier>" syntax.
|
sl@0
|
97 |
#
|
sl@0
|
98 |
do_test tkt2822-2.1 {
|
sl@0
|
99 |
execsql {
|
sl@0
|
100 |
SELECT a, b, c FROM t1 UNION ALL SELECT a, b, c FROM t2 ORDER BY a;
|
sl@0
|
101 |
}
|
sl@0
|
102 |
} {1 3 9 2 6 18 3 9 27 4 12 36 5 15 45 6 18 54}
|
sl@0
|
103 |
|
sl@0
|
104 |
do_test tkt2822-2.2 {
|
sl@0
|
105 |
execsql {
|
sl@0
|
106 |
SELECT a, CAST (b AS TEXT) AS x, c FROM t1
|
sl@0
|
107 |
UNION ALL
|
sl@0
|
108 |
SELECT a, b, c FROM t2
|
sl@0
|
109 |
ORDER BY x;
|
sl@0
|
110 |
}
|
sl@0
|
111 |
} {2 6 18 4 12 36 6 18 54 5 15 45 1 3 9 3 9 27}
|
sl@0
|
112 |
do_test tkt2822-2.3 {
|
sl@0
|
113 |
execsql {
|
sl@0
|
114 |
SELECT t1.a, b, c FROM t1 UNION ALL SELECT t2.a, b, c FROM t2 ORDER BY a;
|
sl@0
|
115 |
}
|
sl@0
|
116 |
} {1 3 9 2 6 18 3 9 27 4 12 36 5 15 45 6 18 54}
|
sl@0
|
117 |
|
sl@0
|
118 |
# Test the "ORDER BY <expression>" syntax.
|
sl@0
|
119 |
#
|
sl@0
|
120 |
do_test tkt2822-3.1 {
|
sl@0
|
121 |
execsql {
|
sl@0
|
122 |
SELECT a, CAST (b AS TEXT) AS x, c FROM t1
|
sl@0
|
123 |
UNION ALL
|
sl@0
|
124 |
SELECT a, b, c FROM t2
|
sl@0
|
125 |
ORDER BY CAST (b AS TEXT);
|
sl@0
|
126 |
}
|
sl@0
|
127 |
} {2 6 18 4 12 36 6 18 54 5 15 45 1 3 9 3 9 27}
|
sl@0
|
128 |
do_test tkt2822-3.2 {
|
sl@0
|
129 |
execsql {
|
sl@0
|
130 |
SELECT t1.a, b, c FROM t1 UNION ALL SELECT t2.a, b, c FROM t2 ORDER BY t1.a;
|
sl@0
|
131 |
}
|
sl@0
|
132 |
} {1 3 9 2 6 18 3 9 27 4 12 36 5 15 45 6 18 54}
|
sl@0
|
133 |
|
sl@0
|
134 |
# Test that if a match cannot be found in the leftmost SELECT, an
|
sl@0
|
135 |
# attempt is made to find a match in subsequent SELECT statements.
|
sl@0
|
136 |
#
|
sl@0
|
137 |
do_test tkt2822-3.3 {
|
sl@0
|
138 |
execsql {
|
sl@0
|
139 |
SELECT a, b, c FROM t1 UNION ALL SELECT a AS x, b, c FROM t2 ORDER BY x;
|
sl@0
|
140 |
}
|
sl@0
|
141 |
} {1 3 9 2 6 18 3 9 27 4 12 36 5 15 45 6 18 54}
|
sl@0
|
142 |
do_test tkt2822-3.4 {
|
sl@0
|
143 |
# But the leftmost SELECT takes precedence.
|
sl@0
|
144 |
execsql {
|
sl@0
|
145 |
SELECT a AS b, CAST (b AS TEXT) AS a, c FROM t1
|
sl@0
|
146 |
UNION ALL
|
sl@0
|
147 |
SELECT a, b, c FROM t2
|
sl@0
|
148 |
ORDER BY a;
|
sl@0
|
149 |
}
|
sl@0
|
150 |
} {2 6 18 4 12 36 6 18 54 5 15 45 1 3 9 3 9 27}
|
sl@0
|
151 |
do_test tkt2822-3.5 {
|
sl@0
|
152 |
execsql {
|
sl@0
|
153 |
SELECT a, b, c FROM t2
|
sl@0
|
154 |
UNION ALL
|
sl@0
|
155 |
SELECT a AS b, CAST (b AS TEXT) AS a, c FROM t1
|
sl@0
|
156 |
ORDER BY a;
|
sl@0
|
157 |
}
|
sl@0
|
158 |
} {1 3 9 2 6 18 3 9 27 4 12 36 5 15 45 6 18 54}
|
sl@0
|
159 |
|
sl@0
|
160 |
# Test some error conditions (ORDER BY clauses that match no column).
|
sl@0
|
161 |
#
|
sl@0
|
162 |
do_test tkt2822-4.1 {
|
sl@0
|
163 |
catchsql {
|
sl@0
|
164 |
SELECT a, b, c FROM t1 UNION ALL SELECT a, b, c FROM t2 ORDER BY x
|
sl@0
|
165 |
}
|
sl@0
|
166 |
} {1 {1st ORDER BY term does not match any column in the result set}}
|
sl@0
|
167 |
do_test tkt2822-4.2 {
|
sl@0
|
168 |
catchsql {
|
sl@0
|
169 |
SELECT a, CAST (b AS TEXT) AS x, c FROM t1
|
sl@0
|
170 |
UNION ALL
|
sl@0
|
171 |
SELECT a, b, c FROM t2
|
sl@0
|
172 |
ORDER BY CAST (b AS INTEGER);
|
sl@0
|
173 |
}
|
sl@0
|
174 |
} {1 {1st ORDER BY term does not match any column in the result set}}
|
sl@0
|
175 |
|
sl@0
|
176 |
# Tests for rule (2).
|
sl@0
|
177 |
#
|
sl@0
|
178 |
# The "ORDER BY b" should match the column alias (rule 2), not the
|
sl@0
|
179 |
# the t3.b value (rule 3).
|
sl@0
|
180 |
#
|
sl@0
|
181 |
do_test tkt2822-5.1 {
|
sl@0
|
182 |
execsql {
|
sl@0
|
183 |
CREATE TABLE t3(a,b);
|
sl@0
|
184 |
INSERT INTO t3 VALUES(1,8);
|
sl@0
|
185 |
INSERT INTO t3 VALUES(9,2);
|
sl@0
|
186 |
|
sl@0
|
187 |
SELECT a AS b FROM t3 ORDER BY b;
|
sl@0
|
188 |
}
|
sl@0
|
189 |
} {1 9}
|
sl@0
|
190 |
do_test tkt2822-5.2 {
|
sl@0
|
191 |
# Case does not matter. b should match B
|
sl@0
|
192 |
execsql {
|
sl@0
|
193 |
SELECT a AS b FROM t3 ORDER BY B;
|
sl@0
|
194 |
}
|
sl@0
|
195 |
} {1 9}
|
sl@0
|
196 |
do_test tkt2822-5.3 {
|
sl@0
|
197 |
# Quoting should not matter
|
sl@0
|
198 |
execsql {
|
sl@0
|
199 |
SELECT a AS 'b' FROM t3 ORDER BY "B";
|
sl@0
|
200 |
}
|
sl@0
|
201 |
} {1 9}
|
sl@0
|
202 |
do_test tkt2822-5.4 {
|
sl@0
|
203 |
# Quoting should not matter
|
sl@0
|
204 |
execsql {
|
sl@0
|
205 |
SELECT a AS "b" FROM t3 ORDER BY [B];
|
sl@0
|
206 |
}
|
sl@0
|
207 |
} {1 9}
|
sl@0
|
208 |
|
sl@0
|
209 |
# In "ORDER BY +b" the term is now an expression rather than
|
sl@0
|
210 |
# a label. It therefore matches by rule (3) instead of rule (2).
|
sl@0
|
211 |
#
|
sl@0
|
212 |
do_test tkt2822-5.5 {
|
sl@0
|
213 |
execsql {
|
sl@0
|
214 |
SELECT a AS b FROM t3 ORDER BY +b;
|
sl@0
|
215 |
}
|
sl@0
|
216 |
} {9 1}
|
sl@0
|
217 |
|
sl@0
|
218 |
# Tests for rule 2 in compound queries
|
sl@0
|
219 |
#
|
sl@0
|
220 |
do_test tkt2822-6.1 {
|
sl@0
|
221 |
execsql {
|
sl@0
|
222 |
CREATE TABLE t6a(p,q);
|
sl@0
|
223 |
INSERT INTO t6a VALUES(1,8);
|
sl@0
|
224 |
INSERT INTO t6a VALUES(9,2);
|
sl@0
|
225 |
CREATE TABLE t6b(x,y);
|
sl@0
|
226 |
INSERT INTO t6b VALUES(1,7);
|
sl@0
|
227 |
INSERT INTO t6b VALUES(7,2);
|
sl@0
|
228 |
|
sl@0
|
229 |
SELECT p, q FROM t6a UNION ALL SELECT x, y FROM t6b ORDER BY 1, 2
|
sl@0
|
230 |
}
|
sl@0
|
231 |
} {1 7 1 8 7 2 9 2}
|
sl@0
|
232 |
do_test tkt2822-6.2 {
|
sl@0
|
233 |
execsql {
|
sl@0
|
234 |
SELECT p PX, q QX FROM t6a UNION ALL SELECT x XX, y YX FROM t6b
|
sl@0
|
235 |
ORDER BY PX, YX
|
sl@0
|
236 |
}
|
sl@0
|
237 |
} {1 7 1 8 7 2 9 2}
|
sl@0
|
238 |
do_test tkt2822-6.3 {
|
sl@0
|
239 |
execsql {
|
sl@0
|
240 |
SELECT p PX, q QX FROM t6a UNION ALL SELECT x XX, y YX FROM t6b
|
sl@0
|
241 |
ORDER BY XX, QX
|
sl@0
|
242 |
}
|
sl@0
|
243 |
} {1 7 1 8 7 2 9 2}
|
sl@0
|
244 |
do_test tkt2822-6.4 {
|
sl@0
|
245 |
execsql {
|
sl@0
|
246 |
SELECT p PX, q QX FROM t6a UNION ALL SELECT x XX, y YX FROM t6b
|
sl@0
|
247 |
ORDER BY QX, XX
|
sl@0
|
248 |
}
|
sl@0
|
249 |
} {7 2 9 2 1 7 1 8}
|
sl@0
|
250 |
do_test tkt2822-6.5 {
|
sl@0
|
251 |
execsql {
|
sl@0
|
252 |
SELECT p PX, q QX FROM t6a UNION ALL SELECT x XX, y YX FROM t6b
|
sl@0
|
253 |
ORDER BY t6b.x, QX
|
sl@0
|
254 |
}
|
sl@0
|
255 |
} {1 7 1 8 7 2 9 2}
|
sl@0
|
256 |
do_test tkt2822-6.6 {
|
sl@0
|
257 |
execsql {
|
sl@0
|
258 |
SELECT p PX, q QX FROM t6a UNION ALL SELECT x XX, y YX FROM t6b
|
sl@0
|
259 |
ORDER BY t6a.q, XX
|
sl@0
|
260 |
}
|
sl@0
|
261 |
} {7 2 9 2 1 7 1 8}
|
sl@0
|
262 |
|
sl@0
|
263 |
# More error message tests. This is really more of a test of the
|
sl@0
|
264 |
# %r ordinal value formatting capablity added to sqlite3_snprintf()
|
sl@0
|
265 |
# by ticket #2822.
|
sl@0
|
266 |
#
|
sl@0
|
267 |
do_test tkt2822-7.1 {
|
sl@0
|
268 |
execsql {
|
sl@0
|
269 |
CREATE TABLE t7(a1,a2,a3,a4,a5,a6,a7,a8,a9,a10,a11,a12,a13,a14,
|
sl@0
|
270 |
a15,a16,a17,a18,a19,a20,a21,a22,a23,a24,a25);
|
sl@0
|
271 |
}
|
sl@0
|
272 |
catchsql {
|
sl@0
|
273 |
SELECT * FROM t7 ORDER BY 0;
|
sl@0
|
274 |
}
|
sl@0
|
275 |
} {1 {1st ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
276 |
do_test tkt2822-7.2 {
|
sl@0
|
277 |
catchsql {
|
sl@0
|
278 |
SELECT * FROM t7 ORDER BY 1, 0;
|
sl@0
|
279 |
}
|
sl@0
|
280 |
} {1 {2nd ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
281 |
do_test tkt2822-7.3 {
|
sl@0
|
282 |
catchsql {
|
sl@0
|
283 |
SELECT * FROM t7 ORDER BY 1, 2, 0;
|
sl@0
|
284 |
}
|
sl@0
|
285 |
} {1 {3rd ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
286 |
do_test tkt2822-7.4 {
|
sl@0
|
287 |
catchsql {
|
sl@0
|
288 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 0;
|
sl@0
|
289 |
}
|
sl@0
|
290 |
} {1 {4th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
291 |
do_test tkt2822-7.9 {
|
sl@0
|
292 |
catchsql {
|
sl@0
|
293 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 0;
|
sl@0
|
294 |
}
|
sl@0
|
295 |
} {1 {9th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
296 |
do_test tkt2822-7.10 {
|
sl@0
|
297 |
catchsql {
|
sl@0
|
298 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 0;
|
sl@0
|
299 |
}
|
sl@0
|
300 |
} {1 {10th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
301 |
do_test tkt2822-7.11 {
|
sl@0
|
302 |
catchsql {
|
sl@0
|
303 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 0;
|
sl@0
|
304 |
}
|
sl@0
|
305 |
} {1 {11th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
306 |
do_test tkt2822-7.12 {
|
sl@0
|
307 |
catchsql {
|
sl@0
|
308 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 12, 0;
|
sl@0
|
309 |
}
|
sl@0
|
310 |
} {1 {12th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
311 |
do_test tkt2822-7.13 {
|
sl@0
|
312 |
catchsql {
|
sl@0
|
313 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 12, 13, 0;
|
sl@0
|
314 |
}
|
sl@0
|
315 |
} {1 {13th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
316 |
do_test tkt2822-7.20 {
|
sl@0
|
317 |
catchsql {
|
sl@0
|
318 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 10,
|
sl@0
|
319 |
11,12,13,14,15,16,17,18,19, 0
|
sl@0
|
320 |
}
|
sl@0
|
321 |
} {1 {20th ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
322 |
do_test tkt2822-7.21 {
|
sl@0
|
323 |
catchsql {
|
sl@0
|
324 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 10,
|
sl@0
|
325 |
11,12,13,14,15,16,17,18,19, 20, 0
|
sl@0
|
326 |
}
|
sl@0
|
327 |
} {1 {21st ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
328 |
do_test tkt2822-7.22 {
|
sl@0
|
329 |
catchsql {
|
sl@0
|
330 |
SELECT * FROM t7 ORDER BY 1, 2, 3, 4, 5, 6, 7, 8, 9, 10,
|
sl@0
|
331 |
11,12,13,14,15,16,17,18,19, 20, 21, 0
|
sl@0
|
332 |
}
|
sl@0
|
333 |
} {1 {22nd ORDER BY term out of range - should be between 1 and 25}}
|
sl@0
|
334 |
|
sl@0
|
335 |
|
sl@0
|
336 |
finish_test
|