This repository was archived by the owner on Aug 31, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 226
Expand file tree
/
Copy path_testerlib.livecodescript
More file actions
396 lines (325 loc) · 11.6 KB
/
_testerlib.livecodescript
File metadata and controls
396 lines (325 loc) · 11.6 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
script "TesterLib"
/*
Copyright (C) 2015-2016 LiveCode Ltd.
This file is part of LiveCode.
LiveCode is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License v3 as published by the Free
Software Foundation.
LiveCode is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License
along with LiveCode. If not see <http://www.gnu.org/licenses/>. */
/*
This script is a library of commands and functions that can be used by
test runner implementations.
*/
on revLoadLibrary
if the target is not me then
pass revLoadLibrary
end if
insert the script of me into back
end revLoadLibrary
local sLogCallbackURL
----------------------------------------------------------------
-- Script-only stack handling
----------------------------------------------------------------
-- Get all livecode script files beneath the CWD, apart from
-- filenames starting with "." or "_"
function TesterGetTestFileNames pBaseFolder, pExtension
local tFiles, tCount
put empty into tFiles
put 0 into tCount
if pBaseFolder is empty then
put the defaultfolder into pBaseFolder
end if
if pExtension is empty then
put "livecodescript" into pExtension
end if
TesterGetActualFileNames_Recursive pBaseFolder, empty, pExtension, tFiles, tCount
return tFiles
end TesterGetTestFileNames
-- Helper command used by runGetTestFileNames
private command TesterGetActualFileNames_Recursive pPath, pRelPath, pExtension, @xFiles, @xCount
-- Process files in the current directory
local tFile
repeat for each line tFile in files(pPath)
if tFile ends with ("." & pExtension) and \
not (tFile begins with "." or tFile begins with "_") then
if pRelPath is not empty then
put pRelPath & slash before tFile
end if
add 1 to xCount
put tFile into xFiles[xCount]
end if
end repeat
-- Process subdirectories
local tFolder, tFolderPath
repeat for each line tFolder in folders(pPath)
if tFolder begins with "." then
next repeat
end if
put pPath & slash & tFolder into tFolderPath
if pRelPath is not empty then
put pRelPath & slash before tFolder
end if
TesterGetActualFileNames_Recursive tFolderPath, tFolder, pExtension, xFiles, xCount
end repeat
end TesterGetActualFileNames_Recursive
-- Get a number-indexed array contain the names of all "test"
-- commands in pFilename.
function TesterParseTestCommandNames pFilename
local tScript
-- Get the contents of the file
open file pFilename for "UTF-8" text read
if the result is not empty then
throw the result
end if
read from file pFilename until end
put it into tScript
close file pFilename
-- Scan the file for "on Test*" definitions
local tCommandNames, tCount, tLine, tName
local tCommandNameSeen
repeat for each line tLine in tScript
if token 1 of tLine is not "on" then
next repeat
end if
put token 2 of tLine into tName
if not (tName begins with "Test") then
next repeat
end if
if tCommandNameSeen[tName] then
-- Duplicate test name
get merge("Error: duplicate test name [[tName]] in [[pFilename]]\n")
if sLogCallbackURL is not empty then
post "^err^N^"&it to sLogCallbackURL
else
write it to stderr
end if
quit 1
end if
-- Exclude the test setup message
if tName is "TestSetup" or tName is "TestTearDown" then
next repeat
end if
add 1 to tCount
put tName into tCommandNames[tCount]
put true into tCommandNameSeen[tName]
end repeat
return tCommandNames
end TesterParseTestCommandNames
-- Prettify a test name by removing a ".livecodescript" suffix
function TesterGetPrettyTestName pFilename, pExtension
if pExtension is empty then
put "livecodescript" into pExtension
end if
if pFilename ends with ("." & pExtension) then
set the itemDelimiter to "."
return item 1 to -2 of pFileName
end if
end TesterGetPrettyTestName
----------------------------------------------------------------
-- TAP support
----------------------------------------------------------------
-- Scan TAP output and extract summary of test results
function TesterTapAnalyse pTapData
local tStats
put 0 into tStats["pass"]
put 0 into tStats["xpass"]
put 0 into tStats["xfail"]
put 0 into tStats["fail"]
put 0 into tStats["skip"]
put pTapData into tStats["log"]
local tLine, tIsOkay, tIsTodo, tIsSkip
local tTail, tDirective
repeat for each line tLine in pTapData
-- Check if the line is a test result and, if so, whether
-- the test was successful
if token 1 of tLine is "ok" then
put true into tIsOkay
else if token 1 of tLine is "not" and token 2 of tLine is "ok" then
put false into tIsOkay
else
-- No test result on this line
next repeat
end if
-- Check if the test on this line was skipped or was expected to fail
put false into tIsTodo
put false into tIsSkip
put char (offset("#", tLine) + 1) to -1 of tLine into tTail
switch token 1 of tTail
case "TODO"
put true into tIsTodo
break
case "SKIP"
put true into tIsSkip
break
end switch
if tIsOkay then
if tIsTodo then
add 1 to tStats["xpass"]
else if tIsSkip then
add 1 to tStats["skip"]
else
add 1 to tStats["pass"]
end if
else
if tIsTodo then
add 1 to tStats["xfail"]
else
add 1 to tStats["fail"]
put tLine & return after tStats["failures"]
end if
end if
end repeat
return tStats
end TesterTapAnalyse
-- Examine the statistics generated by tapAnalyse() and return a string
-- describing the test log's worst result
function TesterTapGetWorstResult pTapAnalysis
local tResult
repeat for each item tResult in "fail,xpass,xfail,pass,skip"
if pTapAnalysis[tResult] > 0 then
return tResult
end if
end repeat
return "pass"
end TesterTapGetWorstResult
-- Get the total number of tests from the statistics generated by
-- tapAnalyse()
function TesterTapGetTestCount pAnalysis
return pAnalysis["pass"] + pAnalysis["xpass"] + pAnalysis["xfail"] + \
pAnalysis["fail"] + pAnalysis["skip"]
end TesterTapGetTestCount
-- Combine two sets of TAP results
function TesterTapCombine pAnalysisA, pAnalysisB
local tCombined, tKey
if pAnalysisA is an array and pAnalysisB is an array then
repeat for each key tKey in pAnalysisB
if tKey is among the words of "log failures" then
put pAnalysisA[tKey] & pAnalysisB[tKey] into tCombined[tKey]
else
put pAnalysisA[tKey] + pAnalysisB[tKey] into tCombined[tKey]
end if
end repeat
else if pAnalysisA is an array then
put pAnalysisA into tCombined
else if pAnalysisB is an array then
put pAnalysisB into tCombined
else
put empty into tCombined
end if
return tCombined
end TesterTapCombine
----------------------------------------------------------------
-- Logging helpers
----------------------------------------------------------------
local sLogInfo
-- Figure out what the highlighting escape codes are for the terminal
--
-- FIXME this really doesn't work properly if LiveCode's stdout
-- *isn't* a TTY.
private command logInit
-- Don't colour output from dev engine as it's processed by
-- standalone engine
if the environment contains "development" then
put false into sLogInfo
exit logInit
end if
-- We can only do colour on Linux and OS X
if the platform is not "Linux" and the platform is not "MacOS" then
put false into sLogInfo
exit logInit
end if
-- Check if colouring is possible
local tTput
put shell("tput colors") into tTput
if the result is not empty or tTput <= 8 then
put false into sLogInfo
exit logInit
end if
-- Get colours
put shell("tput sgr0") into sLogInfo["normal"]
put shell("tput bold") into sLogInfo["bold"]
put shell("tput setaf 1") into sLogInfo["red"]
put shell("tput setaf 2") into sLogInfo["green"]
put shell("tput setaf 3") into sLogInfo["yellow"]
put shell("tput setaf 6") into sLogInfo["cyan"]
end logInit
private function logHighlight pString
logInit
if pString is "fail" then
return sLogInfo["red"] & sLogInfo["bold"] & pString & sLogInfo["normal"]
else if pString is "xfail" or pString is "xpass" then
return sLogInfo["yellow"] & pString & sLogInfo["normal"]
else if pString is "pass" then
return sLogInfo["green"] & pString & sLogInfo["normal"]
else if pString is "skip" then
return sLogInfo["cyan"] & pString & sLogInfo["normal"]
else
return pString
end if
end logHighlight
command TesterLog pStatus, pDescription
local tMessage
put logHighLight(the toUpper of pStatus) into tMessage
put ":" after tMessage
if the number of chars in pStatus < 5 then
put space after tMessage
end if
get tMessage && pDescription & return
if sLogCallbackURL is not empty then
post "^out^N^"&it to sLogCallbackURL
else
write it to stdout
end if
end TesterLog
command TesterLogSummaryLine pTapResults, pTestName
local tTotal, tPassed, tWorst, tMessage
put pTapResults["xpass"] + pTapResults["pass"] + pTapResults["skip"] into tPassed
put TesterTapGetTestCount(pTapResults) into tTotal
put TesterTapGetWorstResult(pTapResults) into tWorst
put pTestName into tMessage
put space & "[" & tPassed & "/" & tTotal & "]" after tMessage
if pTapResults["failures"] is not empty then
put return & pTapResults["failures"] after tMessage
end if
TesterLog tWorst, tMessage
end TesterLogSummaryLine
command TesterLogSetCallbackURL pURL
put pURL into sLogCallbackURL
end TesterLogSetCallbackURL
-- Print out a table of statistics
command TesterRunPrintSummary pAnalysis
local tSummaryString, tTotal, tDecoration
put TesterTapGetTestCount(pAnalysis) into tTotal
-- Format basic summary information
if pAnalysis["xfail"] is 0 and pAnalysis["fail"] is 0 then
put "All" && tTotal && "tests passed" into tSummaryString
else if pAnalysis["fail"] is 0 then
put "All" && tTotal && "tests behaved as expected" into tSummaryString
else
put pAnalysis["fail"] && "OF" && tTotal && "TESTS FAILED" into tSummaryString
end if
put return after tSummaryString
-- Add extra summary info from expected failure & skip directives
if pAnalysis["xpass"] > 0 then
put tab & pAnalysis["xpass"] && "unexpected passes" & return after tSummaryString
end if
if pAnalysis["xfail"] > 0 then
put tab & pAnalysis["xfail"] && "expected failures" & return after tSummaryString
end if
if pAnalysis["skip"] > 0 then
put tab & pAnalysis["skip"] && "skipped" & return after tSummaryString
end if
put "================================================================" into tDecoration
put tDecoration & return before tSummaryString
put tDecoration & return after tSummaryString
if sLogCallbackURL is not empty then
post "^out^N^"&tSummaryString to sLogCallbackURL
else
write tSummaryString to stdout
end if
end TesterRunPrintSummary