forked from livecode/livecode
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy path_testlib.livecodescript
More file actions
338 lines (273 loc) · 9.8 KB
/
_testlib.livecodescript
File metadata and controls
338 lines (273 loc) · 9.8 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
script "TestLibrary"
/*
Copyright (C) 2015 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/>. */
on revLoadLibrary
insert the script of me into back
end revLoadLibrary
----------------------------------------------------------------
-- Helper functions
----------------------------------------------------------------
local sOutputToVariable
local sOutput
private function _TestValidateCount pCount
if pCount is not an integer or pCount <= 0 then
throw "Bad test count '" & pCount & "': must be positive integer"
end if
return pCount
end _TestValidateCount
private function _TestValidateDescription pDescription
if the number of lines in pDescription > 1 then
throw "Bad test description '" & line 1 of pDescription & "...': multiple lines"
end if
if "0123456789" contains codepoint 1 of pDescription then
throw "Bad test description '" & pDescription & "': starts with digit"
end if
if pDescription contains "#" then
throw "Bad test description '" & pDescription & "': contains '#'"
end if
return line 1 of pDescription
end _TestValidateDescription
private function _TestValidateReason pReason
if the number of lines in pReason > 1 then
throw "Bad test directive reason '" & line 1 of pReason & "...': multiple lines"
end if
if pReason contains "#" then
throw "Bad test directive reason '" & pReason & "': contains '#'"
end if
return line 1 of pReason
end _TestValidateReason
private function _TestValidateDirective pDirective
switch pDirective
case empty
return empty
case "TODO"
return "TODO"
case "SKIP"
return "SKIP"
default
throw "Bad test directive '" & line 1 of pDirective & "'"
end switch
end _TestValidateDirective
-- Used by top level assertion functions to generate output
private command _TestOutput pIsOkay, pDescription, pDirective, pReason
local tDescription, tDirective, tReason
put _TestValidateDescription(pDescription) into tDescription
put _TestValidateDirective(pDirective) into tDirective
put _TestValidateReason(pReason) into tReason
local tMessage
if pIsOkay then
put "ok" into tMessage
else
put "not ok" into tMessage
end if
if tDescription is not empty then
put " - " & tDEscription after tMessage
end if
if tDirective is not empty then
put " # " & tDirective after tMessage
if tReason is not empty then
put " " & tReason after tMessage
end if
end if
_TestWriteOutput tMessage & return
end _TestOutput
private command _TestWriteOutput pMessage
-- As stdout is considered to be a 'native' stream, we encode to UTF-8 first
-- then will unencode in the test runner.
if sOutputToVariable then
put pMessage after sOutput
else
write textEncode(pMessage, "UTF8") to stdout
end if
end _TestWriteOutput
private function _TestGetBuiltExtensionsFolder
local tPath, tRepoPath
put specialfolderpath("engine") into tPath
put TestGetEngineRepositoryPath() into tRepoPath
# Find the built extensions folder
set the itemdelimiter to slash
repeat while tPath is not tRepoPath and tPath is not empty
if there is a folder (tPath & slash & "packaged_extensions") then
return (tPath & slash & "packaged_extensions")
end if
delete item -1 of tPath
end repeat
end _TestGetBuiltExtensionsFolder
private command _TestLoadExtension pFolder, pPath
if there is a folder (pFolder & slash & "resources") then
do "load extension from file pPath with resource path (pFolder & slash & " & quote & "resources" & quote & ")"
else
do "load extension from file pPath"
end if
return the result
end _TestLoadExtension
----------------------------------------------------------------
-- Unit test library functions
----------------------------------------------------------------
on TestOutputToVariable
put true into sOutputToVariable
end TestOutputToVariable
function TestFetchAndClearOutput
if not sOutputToVariable then
return empty
end if
local tOutput
put sOutput into tOutput
put empty into sOutput
return tOutput
end TestFetchAndClearOutput
on TestPlan pCount
_TestWriteOutput "1.." & _TestValidateCount(pCount) & return
end TestPlan
on TestDiagnostic pMessage
local tLine
repeat for each line tLine in pMessage
_TestWriteOutput "#" && tLine & return
end repeat
end TestDiagnostic
on TestSkip pDescription, pReasonSkipped
_TestOutput true, pDescription, "SKIP", pReasonSkipped
end TestSkip
on TestAssert pDescription, pExpectTrue
_TestOutput pExpectTrue, pDescription, empty, empty
end TestAssert
on TestAssertBroken pDescription, pExpectTrue, pReasonBroken
_TestOutput pExpectTrue, pDescription, "TODO", pReasonBroken
end TestAssertBroken
on TestAssertThrow pDescription, pHandler, pTarget, pErrorCode, pParam
local tError
try
dispatch pHandler to pTarget with pParam
catch tError
end try
TestAssert pDescription, pErrorCode is item 1 of line 1 of tError
end TestAssertThrow
on ErrorDialog executionError, parseError
write executionError & return to stderr
quit 1
end ErrorDialog
function TestGetEngineRepositoryPath
set the itemdelimiter to "/"
return item 1 to -3 of the filename of me
end TestGetEngineRepositoryPath
function TestGetIDERepositoryPath
set the itemdelimiter to "/"
return item 1 to -3 of the filename of me & slash & "ide"
end TestGetIDERepositoryPath
on TestLoadExtension pName
set the itemdelimiter to "."
local tExtensionUnzipFolder
put pName into tExtensionUnzipFolder
local tError
put "extension" && pName && "not found" into tError
local tExtensionsFolder
put _TestGetBuiltExtensionsFolder() into tExtensionsFolder
local tExtensionFolder
if tExtensionsFolder is not empty then
if there is a folder (tExtensionsFolder & slash & tExtensionUnzipFolder) then
put (tExtensionsFolder & slash & tExtensionUnzipFolder) into tExtensionFolder
end if
end if
local tExtensionFile
if tExtensionFolder is not empty then
if there is a file (tExtensionFolder & slash & "module.lcm") then
put (tExtensionFolder & slash & "module.lcm") into tExtensionFile
end if
end if
if tExtensionFile is not empty then
_TestLoadExtension tExtensionFolder, tExtensionFile
end if
put the result into tError
if tError is not empty then
write tError & return to stderr
quit 1
end if
end TestLoadExtension
on TestLoadAllExtensions
local tDefaultFolder
put the defaultFolder into tDefaultFolder
local tExtFolder
put _TestGetBuiltExtensionsFolder() into tExtFolder
set the defaultFolder to tExtFolder
local tExtensions
put the folders into tExtensions
local tFiles, tAllExtensionsA
local tHasCompiled, tSource
# Collect all the valid (and compiled) extension files
repeat for each line tFolder in tExtensions
if tFolder begins with "." then
next repeat
end if
put false into tHasCompiled
put empty into tSource
put tExtFolder & slash before tFolder
set the defaultFolder to tFolder
put the files into tFiles
repeat for each line tFile in tFiles
if tFile ends with ".lcb" then
put tFile into tSource
else if tFile is "module.lcm" then
put true into tHasCompiled
end if
if tHasCompiled and tSource is not empty then
put tFolder into tAllExtensionsA[tFolder & slash & tSource]
exit repeat
end if
end repeat
end repeat
set the defaultFolder to tDefaultFolder
# Use the lc-compile --deps option to sort by dependency
local tLCCompile, tLCBFiles
put the keys of tAllExtensionsA into tLCBFiles
replace return with " " in tLCBFiles
set the itemdelimiter to slash
put item 1 to -2 of tExtFolder & slash & "lc-compile" into tLCCompile
put shell(tLCCompile && "--deps order --" && tLCBFiles) into tLCBFiles
if the result is not 0 then
write the result & return to stderr
quit 1
end if
# Load the extensions in order
repeat for each line tExtFile in tLCBFiles
_TestLoadExtension tAllExtensionsA[tExtFile], tExtFile
end repeat
end TestLoadAllExtensions
on TestRepeat pDesc, pHandler, pTarget, pTimeOut, pParamsArray
# Construct a dispatch command with all the desired parameters
local tDoString, tParams, tResult
put "dispatch pHandler to pTarget" into tDoString
repeat with tParam = 1 to the number of elements in pParamsArray
if tParams is empty then
put " with pParamsArray[" & tParam & "]" into tParams
else
put ", pParamsArray[" & tParam & "]" after tParams
end if
end repeat
put tParams after tDoString
put "; put the result into tResult" after tDoString
put false into tResult
local tTimer, tTimeout
put false into tTimeout
put the millisecs into tTimerStart
repeat while tResult is false and not tTimeout
wait 1 millisecs with messages
do tDoString
if the millisecs - tTimerStart > pTimeOut then
put true into tTimeOut
end if
end repeat
TestAssert pDesc, tResult
if (not tResult) and tTimeOut then
TestDiagnostic pDesc & "- timed out"
end if
end TestRepeat