os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/tkt2822.test
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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