blob: 0db6a26e4c2c32c4cadfc219df8cd65f1853d345 [file] [log] [blame]
Denis Vlasenko960eca62007-07-18 18:33:18 +00001#!/bin/sh
2
Denis Vlasenko960eca62007-07-18 18:33:18 +00003# Copyright 2007 by Denys Vlasenko <vda.linux@googlemail.com>
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02004# Licensed under GPLv2, see file LICENSE in this source tree.
Denis Vlasenko960eca62007-07-18 18:33:18 +00005
Mike Frysingercaa79402009-11-04 18:41:22 -05006. ./testing.sh
Denis Vlasenko960eca62007-07-18 18:33:18 +00007
Denis Vlasenkobb130792008-07-15 06:45:14 +00008# testing "description" "command" "result" "infile" "stdin"
Denis Vlasenko960eca62007-07-18 18:33:18 +00009
10testing "awk -F case 0" "awk -F '[#]' '{ print NF }'" "" "" ""
11testing "awk -F case 1" "awk -F '[#]' '{ print NF }'" "0\n" "" "\n"
12testing "awk -F case 2" "awk -F '[#]' '{ print NF }'" "2\n" "" "#\n"
13testing "awk -F case 3" "awk -F '[#]' '{ print NF }'" "3\n" "" "#abc#\n"
14testing "awk -F case 4" "awk -F '[#]' '{ print NF }'" "3\n" "" "#abc#zz\n"
15testing "awk -F case 5" "awk -F '[#]' '{ print NF }'" "4\n" "" "#abc##zz\n"
16testing "awk -F case 6" "awk -F '[#]' '{ print NF }'" "4\n" "" "z#abc##zz\n"
17testing "awk -F case 7" "awk -F '[#]' '{ print NF }'" "5\n" "" "z##abc##zz\n"
18
Tanguy Pruvot0a393cf2012-06-11 10:33:45 +020019# conditions and operators
20testing "awk if operator == " "awk 'BEGIN{if(23==23) print \"foo\"}'" "foo\n" "" ""
21testing "awk if operator != " "awk 'BEGIN{if(23!=23) print \"bar\"}'" "" "" ""
22testing "awk if operator >= " "awk 'BEGIN{if(23>=23) print \"foo\"}'" "foo\n" "" ""
23testing "awk if operator < " "awk 'BEGIN{if(2 < 13) print \"foo\"}'" "foo\n" "" ""
24testing "awk if string == " "awk 'BEGIN{if(\"a\"==\"ab\") print \"bar\"}'" "" "" ""
25
Denis Vlasenko66496d72008-08-29 08:37:07 +000026# 4294967295 = 0xffffffff
Denys Vlasenkob8554602013-07-22 11:49:06 +020027testing "awk bitwise op" "awk '{ print or(4294967295,1) }'" "4294967295\n" "" "\n"
Bernhard Reutner-Fischerb79a0fe2013-03-06 21:01:05 +010028
29# we were testing for a non-empty body when deciding if a function was
30# defined or not. The testcase below caused:
31# awk: cmd. line:8: Call to undefined function
32prg='
33function empty_fun(count) {
34 # empty
35}
36END {
37 i=1
38 print "L" i "\n"
39 empty_fun(i + i + ++i)
40 print "L" i "\n"
41}'
42testing "awk handles empty function f(arg){}" \
43 "awk '$prg'" \
44 "L1\n\nL2\n\n" \
45 "" ""
46
Bernhard Reutner-Fischera060a1a2013-07-31 15:29:20 +020047prg='
48function outer_fun() {
49 return 1
50}
51END {
52 i=1
53 print "L" i "\n"
54 i += outer_fun()
55 print "L" i "\n"
56}'
57testing "awk properly handles function from other scope" \
58 "awk '$prg'" \
59 "L1\n\nL2\n\n" \
60 "" ""
61
62prg='
63END {
64 i=1
65 print "L" i "\n"
66 i + trigger_error_fun()
67 print "L" i "\n"
68}'
69testing "awk properly handles undefined function" \
70 "awk '$prg' 2>&1" \
71 "L1\n\nawk: cmd. line:5: Call to undefined function\n" \
72 "" ""
73
74
Denys Vlasenkobfa1b2e2010-05-11 03:53:57 +020075optional DESKTOP
Denys Vlasenkob8554602013-07-22 11:49:06 +020076testing "awk hex const 1" "awk '{ print or(0xffffffff,1) }'" "4294967295\n" "" "\n"
77testing "awk hex const 2" "awk '{ print or(0x80000000,1) }'" "2147483649\n" "" "\n"
Denis Vlasenkoa2e1eea2008-09-02 09:00:23 +000078testing "awk oct const" "awk '{ print or(01234,1) }'" "669\n" "" "\n"
Denys Vlasenkobfa1b2e2010-05-11 03:53:57 +020079SKIP=
Denis Vlasenko66496d72008-08-29 08:37:07 +000080
Denys Vlasenkod527e0c2010-10-05 13:22:11 +020081# check that "hex/oct integer" heuristic doesn't kick in on 00NN.NNN
82testing "awk floating const with leading zeroes" \
83 "awk '{ printf \"%f %f\n\", \"000.123\", \"009.123\" }'" \
84 "0.123000 9.123000\n" \
85 "" "\n"
86
Denis Vlasenko67b5eeb2009-04-12 13:54:13 +000087# long field seps requiring regex
88testing "awk long field sep" "awk -F-- '{ print NF, length(\$NF), \$NF }'" \
89 "2 0 \n3 0 \n4 0 \n5 0 \n" \
90 "" \
91 "a--\na--b--\na--b--c--\na--b--c--d--"
92
Denys Vlasenkoea664dd2012-06-22 18:41:01 +020093testing "awk -F handles escapes" "awk -F'\\x21' '{print \$1}'" \
94 "a\n" \
95 "" \
96 "a!b\n"
97
Denis Vlasenko7a676642009-03-15 22:20:31 +000098# '@(samp|code|file)\{' is an invalid extended regex (unmatched '{'),
99# but gawk 3.1.5 does not bail out on it.
100testing "awk gsub falls back to non-extended-regex" \
101 "awk 'gsub(\"@(samp|code|file)\{\",\"\");'; echo \$?" "0\n" "" "Hi\n"
102
Denys Vlasenko8e3aff02010-05-10 11:00:11 +0200103optional TAR BUNZIP2 FEATURE_SEAMLESS_BZ2
Denys Vlasenkoe3d90a92010-05-10 05:53:16 +0200104test x"$SKIP" != x"1" && tar xjf awk_t1.tar.bz2
Denis Vlasenko3bb2bbd2008-07-01 01:57:36 +0000105testing "awk 'gcc build bug'" \
106 "awk -f awk_t1_opt-functions.awk -f awk_t1_opth-gen.awk <awk_t1_input | md5sum" \
107 "f842e256461a5ab1ec60b58d16f1114f -\n" \
108 "" ""
Denys Vlasenkoe3d90a92010-05-10 05:53:16 +0200109rm -rf awk_t1_* 2>/dev/null
110SKIP=
Denis Vlasenko3bb2bbd2008-07-01 01:57:36 +0000111
Denis Vlasenko41d5ebe2009-01-25 01:00:15 +0000112Q='":"'
113
114testing "awk NF in BEGIN" \
115 "awk 'BEGIN { print ${Q} NF ${Q} \$0 ${Q} \$1 ${Q} \$2 ${Q} }'" \
116 ":0::::\n" \
117 "" ""
118
Denys Vlasenko12847742009-11-30 01:15:04 +0100119prg='
120function b(tmp) {
121 tmp = 0;
122 print "" tmp; #this line causes the bug
123 return tmp;
124}
125function c(tmpc) {
126 tmpc = b(); return tmpc;
127}
128BEGIN {
129 print (c() ? "string" : "number");
130}'
131testing "awk string cast (bug 725)" \
132 "awk '$prg'" \
133 "0\nnumber\n" \
134 "" ""
135
Alexander Shishkind03cd3b2010-02-25 17:55:40 +0200136testing "awk handles whitespace before array subscript" \
137 "awk 'BEGIN { arr [3] = 1; print arr [3] }'" "1\n" "" ""
138
Denys Vlasenko6a0d7492010-10-23 21:02:15 +0200139# GNU awk 3.1.5's "print ERRNO" prints "No such file or directory" instead of "2",
140# do we need to emulate that as well?
141testing "awk handles non-existing file correctly" \
142 "awk 'BEGIN { getline line <\"doesnt_exist\"; print ERRNO; ERRNO=0; close(\"doesnt_exist\"); print ERRNO; print \"Ok\" }'" \
143 "2\n0\nOk\n" "" ""
144
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100145prg='
146BEGIN {
Denys Vlasenko90f19fa2010-03-11 08:27:53 +0100147 u["a"]=1
148 u["b"]=1
149 u["c"]=1
150 v["d"]=1
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100151 v["e"]=1
Denys Vlasenko90f19fa2010-03-11 08:27:53 +0100152 v["f"]=1
153 for (l in u) {
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100154 print "outer1", l;
155 for (l in v) {
156 print " inner", l;
157 }
158 print "outer2", l;
159 }
160 print "end", l;
161 l="a"
162 exit;
163}'
164testing "awk nested loops with the same variable" \
165 "awk '$prg'" \
166 "\
Denys Vlasenko90f19fa2010-03-11 08:27:53 +0100167outer1 a
168 inner d
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100169 inner e
Denys Vlasenko90f19fa2010-03-11 08:27:53 +0100170 inner f
171outer2 f
172outer1 b
173 inner d
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100174 inner e
Denys Vlasenko90f19fa2010-03-11 08:27:53 +0100175 inner f
176outer2 f
177outer1 c
178 inner d
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100179 inner e
Denys Vlasenko90f19fa2010-03-11 08:27:53 +0100180 inner f
181outer2 f
182end f
183" \
184 "" ""
185
186prg='
187BEGIN {
188 u["a"]=1
189 u["b"]=1
190 u["c"]=1
191 v["d"]=1
192 v["e"]=1
193 v["f"]=1
194 for (l in u) {
195 print "outer1", l;
196 for (l in v) {
197 print " inner", l;
198 break;
199 }
200 print "outer2", l;
201 }
202 print "end", l;
203 l="a"
204 exit;
205}'
206# It's not just buggy, it enters infinite loop. Thus disabled
207false && test x"$SKIP_KNOWN_BUGS" = x"" && testing "awk nested loops with the same variable and break" \
208 "awk '$prg'" \
209 "\
210outer1 a
211 inner d
212outer2 d
213outer1 b
214 inner d
215outer2 d
216outer1 c
217 inner d
218outer2 d
219end d
220" \
221 "" ""
222
223prg='
224function f() {
225 for (l in v) {
226 print " inner", l;
227 return;
228 }
229}
230
231BEGIN {
232 u["a"]=1
233 u["b"]=1
234 u["c"]=1
235 v["d"]=1
236 v["e"]=1
237 v["f"]=1
238 for (l in u) {
239 print "outer1", l;
240 f();
241 print "outer2", l;
242 }
243 print "end", l;
244 l="a"
245 exit;
246}'
247# It's not just buggy, it enters infinite loop. Thus disabled
248false && test x"$SKIP_KNOWN_BUGS" = x"" && testing "awk nested loops with the same variable and return" \
249 "awk '$prg'" \
250 "\
251outer1 a
252 inner d
253outer2 d
254outer1 b
255 inner d
256outer2 d
257outer1 c
258 inner d
259outer2 d
260end d
Denys Vlasenko3cb60c32010-03-10 19:20:32 +0100261" \
262 "" ""
263
Denys Vlasenko6f4a7852018-01-07 01:19:08 +0100264prg='
265BEGIN{
266cnt = 0
267a[cnt] = "zeroth"
268a[++cnt] = "first"
269delete a[cnt--]
270print cnt
271print "[0]:" a[0]
272print "[1]:" a[1]
273}'
274testing "awk 'delete a[v--]' evaluates v-- once" \
275 "awk '$prg'" \
276 "\
2770
278[0]:zeroth
279[1]:
280" \
281 "" ""
282
Brian Foley1c42c182019-01-06 18:32:59 -0800283testing "awk func arg parsing 1" \
284 "awk 'func f(,) { }' 2>&1" "awk: cmd. line:1: Unexpected token\n" "" ""
285
286testing "awk func arg parsing 2" \
287 "awk 'func f(a,,b) { }' 2>&1" "awk: cmd. line:1: Unexpected token\n" "" ""
288
289testing "awk func arg parsing 3" \
290 "awk 'func f(a,) { }' 2>&1" "awk: cmd. line:1: Unexpected token\n" "" ""
291
292testing "awk func arg parsing 4" \
293 "awk 'func f(a b) { }' 2>&1" "awk: cmd. line:1: Unexpected token\n" "" ""
294
Denys Vlasenko7b46d112011-09-11 00:30:56 +0200295testing "awk handles empty ()" \
296 "awk 'BEGIN {print()}' 2>&1" "awk: cmd. line:1: Empty sequence\n" "" ""
297
Denys Vlasenkodf8066a2012-07-11 01:27:15 +0200298testing "awk FS assignment" "awk '{FS=\":\"; print \$1}'" \
299 "a:b\ne\n" \
300 "" \
301 "a:b c:d\ne:f g:h"
302
Denys Vlasenko198b02f2013-12-31 23:22:36 +0100303optional FEATURE_AWK_LIBM
Denys Vlasenkob8554602013-07-22 11:49:06 +0200304testing "awk large integer" \
305 "awk 'BEGIN{n=(2^31)-1; print n, int(n), n%1, ++n, int(n), n%1}'" \
306 "2147483647 2147483647 0 2147483648 2147483648 0\n" \
307 "" ""
Denys Vlasenko198b02f2013-12-31 23:22:36 +0100308SKIP=
Denys Vlasenkob8554602013-07-22 11:49:06 +0200309
Denys Vlasenko7985bc12013-10-12 04:51:54 +0200310testing "awk length(array)" \
Denys Vlasenkobd0e2212013-11-21 15:09:55 +0100311 "awk 'BEGIN{ A[1]=2; A[\"qwe\"]=\"asd\"; print length(A)}'" \
Denys Vlasenko7985bc12013-10-12 04:51:54 +0200312 "2\n" \
313 "" ""
314
Denys Vlasenko28b00ce2015-10-02 02:41:39 +0200315testing "awk length()" \
316 "awk '{print length; print length(); print length(\"qwe\"); print length(99+9)}'" \
317 "3\n3\n3\n3\n" \
318 "" "qwe"
319
Denys Vlasenkobd0e2212013-11-21 15:09:55 +0100320testing "awk -f and ARGC" \
321 "awk -f - input" \
322 "re\n2\n" \
323 "do re mi\n" \
324 '{print $2; print ARGC;}' \
325
Denys Vlasenko198b02f2013-12-31 23:22:36 +0100326optional FEATURE_AWK_GNU_EXTENSIONS
Denys Vlasenkobd0e2212013-11-21 15:09:55 +0100327testing "awk -e and ARGC" \
328 "awk -e '{print \$2; print ARGC;}' input" \
329 "re\n2\n" \
330 "do re mi\n" \
Denys Vlasenko198b02f2013-12-31 23:22:36 +0100331 ""
332SKIP=
Denys Vlasenkobd0e2212013-11-21 15:09:55 +0100333
Denys Vlasenko5f8daef2014-06-26 16:40:28 +0200334# The examples are in fact not valid awk programs (break/continue
335# can only be used inside loops).
336# But we do accept them outside of loops.
337# We had a bug with misparsing "break ; else" sequence.
338# Test that *that* bug is fixed, using simplest possible scripts:
339testing "awk break" \
340 "awk -f - 2>&1; echo \$?" \
341 "0\n" \
342 "" \
343 'BEGIN { if (1) break; else a = 1 }'
344testing "awk continue" \
345 "awk -f - 2>&1; echo \$?" \
346 "0\n" \
347 "" \
348 'BEGIN { if (1) continue; else a = 1 }'
349
Brian Foley61d59972016-10-15 14:45:40 +0100350testing "awk handles invalid for loop" \
Brian Foley08a514c2019-01-01 13:40:59 -0800351 "awk -e '{ for() }' 2>&1" "awk: cmd. line:1: Unexpected token\n" "" ""
Brian Foley61d59972016-10-15 14:45:40 +0100352
Brian Foleydac15a12019-01-01 13:40:58 -0800353testing "awk handles colon not preceded by ternary" \
Brian Foley08a514c2019-01-01 13:40:59 -0800354 "awk -e foo:bar: 2>&1" "awk: cmd. line:1: Unexpected token\n" "" ""
355
356testing "awk errors on missing delete arg" \
357 "awk -e '{delete}' 2>&1" "awk: cmd. line:1: Too few arguments\n" "" ""
Brian Foleydac15a12019-01-01 13:40:58 -0800358
Denys Vlasenkodf8066a2012-07-11 01:27:15 +0200359# testing "description" "command" "result" "infile" "stdin"
Denys Vlasenko2454e672018-04-23 10:53:18 +0200360testing 'awk negative field access' \
361 'awk 2>&1 -- '\''{ $(-1) }'\' \
362 "awk: cmd. line:1: Access to negative field\n" \
363 '' \
364 'anything'
365
Denys Vlasenkodf8066a2012-07-11 01:27:15 +0200366
Denis Vlasenko960eca62007-07-18 18:33:18 +0000367exit $FAILCOUNT