roblox-opencode 1.0.0 → 1.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +112 -122
- package/commands/setup-game.md +108 -108
- package/commands/sync-check.md +53 -53
- package/core/roblox-core.md +93 -93
- package/dist/server.js +189 -167
- package/package.json +35 -35
- package/skills/roblox-analytics/SKILL.md +277 -277
- package/skills/roblox-analytics/references/event-batcher.luau +75 -75
- package/skills/roblox-animation-vfx/SKILL.md +1325 -1325
- package/skills/roblox-architecture/SKILL.md +877 -863
- package/skills/roblox-architecture/references/combat-systems.md +1381 -1381
- package/skills/roblox-code-review/SKILL.md +686 -686
- package/skills/roblox-data/SKILL.md +889 -889
- package/skills/roblox-data/references/inventory-systems.md +1729 -1729
- package/skills/roblox-debug/SKILL.md +98 -98
- package/skills/roblox-gui/SKILL.md +1103 -1103
- package/skills/roblox-gui-fusion/SKILL.md +150 -150
- package/skills/roblox-gui-fusion/references/inventory.luau +427 -427
- package/skills/roblox-gui-fusion/references/settings-menu.luau +579 -579
- package/skills/roblox-gui-fusion/references/shop.luau +411 -411
- package/skills/roblox-luau-mastery/SKILL.md +1618 -1519
- package/skills/roblox-monetization/SKILL.md +1084 -1084
- package/skills/roblox-monetization/references/process-receipt.luau +131 -131
- package/skills/roblox-networking/SKILL.md +669 -669
- package/skills/roblox-networking/references/remote-validator.luau +193 -193
- package/skills/roblox-publish-checklist/SKILL.md +127 -127
- package/skills/roblox-runtime/SKILL.md +753 -753
- package/skills/roblox-sharp-edges/SKILL.md +294 -294
- package/skills/roblox-sync/SKILL.md +126 -126
- package/skills/roblox-testing/SKILL.md +943 -943
- package/skills/roblox-tooling/SKILL.md +149 -149
- package/vendor/LICENSES/ProfileStore-LICENSE +201 -201
- package/vendor/LICENSES/RbxUtil-LICENSE +7 -7
- package/vendor/LICENSES/promise-LICENSE +20 -20
- package/vendor/LICENSES/t-LICENSE +21 -21
- package/vendor/LICENSES/testez-LICENSE +200 -200
- package/vendor/README.md +83 -83
- package/vendor/fusion/Animation/ExternalTime.luau +83 -83
- package/vendor/fusion/Animation/Spring.luau +321 -321
- package/vendor/fusion/Animation/Stopwatch.luau +127 -127
- package/vendor/fusion/Animation/Tween.luau +187 -187
- package/vendor/fusion/Animation/getTweenDuration.luau +27 -27
- package/vendor/fusion/Animation/getTweenRatio.luau +47 -47
- package/vendor/fusion/Animation/lerpType.luau +163 -163
- package/vendor/fusion/Animation/packType.luau +99 -99
- package/vendor/fusion/Animation/springCoefficients.luau +80 -80
- package/vendor/fusion/Animation/unpackType.luau +102 -102
- package/vendor/fusion/Colour/Oklab.luau +70 -70
- package/vendor/fusion/Colour/sRGB.luau +54 -54
- package/vendor/fusion/External.luau +167 -167
- package/vendor/fusion/ExternalDebug.luau +69 -69
- package/vendor/fusion/Graph/Observer.luau +113 -113
- package/vendor/fusion/Graph/castToGraph.luau +28 -28
- package/vendor/fusion/Graph/change.luau +80 -80
- package/vendor/fusion/Graph/depend.luau +32 -32
- package/vendor/fusion/Graph/evaluate.luau +55 -55
- package/vendor/fusion/Instances/Attribute.luau +57 -57
- package/vendor/fusion/Instances/AttributeChange.luau +46 -46
- package/vendor/fusion/Instances/AttributeOut.luau +63 -63
- package/vendor/fusion/Instances/Child.luau +21 -21
- package/vendor/fusion/Instances/Children.luau +147 -147
- package/vendor/fusion/Instances/Hydrate.luau +32 -32
- package/vendor/fusion/Instances/New.luau +52 -52
- package/vendor/fusion/Instances/OnChange.luau +49 -49
- package/vendor/fusion/Instances/OnEvent.luau +53 -53
- package/vendor/fusion/Instances/Out.luau +69 -69
- package/vendor/fusion/Instances/applyInstanceProps.luau +148 -148
- package/vendor/fusion/Instances/defaultProps.luau +194 -194
- package/vendor/fusion/LICENSE +21 -21
- package/vendor/fusion/Logging/formatError.luau +48 -48
- package/vendor/fusion/Logging/messages.luau +51 -51
- package/vendor/fusion/Logging/parseError.luau +24 -24
- package/vendor/fusion/Memory/checkLifetime.luau +133 -133
- package/vendor/fusion/Memory/deriveScope.luau +23 -23
- package/vendor/fusion/Memory/deriveScopeImpl.luau +44 -44
- package/vendor/fusion/Memory/doCleanup.luau +78 -78
- package/vendor/fusion/Memory/innerScope.luau +33 -33
- package/vendor/fusion/Memory/legacyCleanup.luau +17 -17
- package/vendor/fusion/Memory/needsDestruction.luau +16 -16
- package/vendor/fusion/Memory/poisonScope.luau +33 -33
- package/vendor/fusion/Memory/scopePool.luau +54 -54
- package/vendor/fusion/Memory/scoped.luau +26 -26
- package/vendor/fusion/Memory/whichLivesLonger.luau +74 -74
- package/vendor/fusion/RobloxExternal.luau +97 -97
- package/vendor/fusion/State/Computed.luau +138 -138
- package/vendor/fusion/State/For/Disassembly.luau +210 -210
- package/vendor/fusion/State/For/ForTypes.luau +30 -30
- package/vendor/fusion/State/For/init.luau +109 -109
- package/vendor/fusion/State/ForKeys.luau +93 -93
- package/vendor/fusion/State/ForPairs.luau +96 -96
- package/vendor/fusion/State/ForValues.luau +93 -93
- package/vendor/fusion/State/Value.luau +87 -87
- package/vendor/fusion/State/castToState.luau +25 -25
- package/vendor/fusion/State/peek.luau +30 -30
- package/vendor/fusion/Types.luau +314 -314
- package/vendor/fusion/Utility/Contextual.luau +90 -90
- package/vendor/fusion/Utility/Safe.luau +22 -22
- package/vendor/fusion/Utility/isSimilar.luau +29 -29
- package/vendor/fusion/Utility/merge.luau +35 -35
- package/vendor/fusion/Utility/nameOf.luau +34 -34
- package/vendor/fusion/Utility/never.luau +13 -13
- package/vendor/fusion/Utility/nicknames.luau +10 -10
- package/vendor/fusion/Utility/xtypeof.luau +26 -26
- package/vendor/fusion/init.luau +82 -82
- package/vendor/profilestore/init.luau +2242 -2242
- package/vendor/promise/init.luau +1982 -1982
- package/vendor/rbxutil/buffer-util/Buffer.test.luau +25 -25
- package/vendor/rbxutil/buffer-util/BufferReader.luau +228 -228
- package/vendor/rbxutil/buffer-util/BufferWriter.luau +269 -269
- package/vendor/rbxutil/buffer-util/DataTypeBuffer.luau +223 -223
- package/vendor/rbxutil/buffer-util/Types.luau +60 -60
- package/vendor/rbxutil/buffer-util/index.d.ts +153 -153
- package/vendor/rbxutil/buffer-util/init.luau +41 -41
- package/vendor/rbxutil/buffer-util/package.json +16 -16
- package/vendor/rbxutil/buffer-util/wally.toml +9 -9
- package/vendor/rbxutil/comm/Client/ClientComm.luau +232 -232
- package/vendor/rbxutil/comm/Client/ClientRemoteProperty.luau +156 -156
- package/vendor/rbxutil/comm/Client/ClientRemoteSignal.luau +109 -109
- package/vendor/rbxutil/comm/Client/init.luau +135 -135
- package/vendor/rbxutil/comm/Server/RemoteProperty.luau +295 -295
- package/vendor/rbxutil/comm/Server/RemoteSignal.luau +211 -211
- package/vendor/rbxutil/comm/Server/ServerComm.luau +211 -211
- package/vendor/rbxutil/comm/Server/init.luau +140 -140
- package/vendor/rbxutil/comm/Types.luau +18 -18
- package/vendor/rbxutil/comm/Util.luau +27 -27
- package/vendor/rbxutil/comm/init.luau +35 -35
- package/vendor/rbxutil/comm/wally.toml +13 -13
- package/vendor/rbxutil/component/init.luau +759 -759
- package/vendor/rbxutil/component/init.test.luau +311 -311
- package/vendor/rbxutil/component/wally.toml +14 -14
- package/vendor/rbxutil/concur/init.luau +542 -542
- package/vendor/rbxutil/concur/init.test.luau +364 -364
- package/vendor/rbxutil/concur/wally.toml +8 -8
- package/vendor/rbxutil/enum-list/init.luau +101 -101
- package/vendor/rbxutil/enum-list/init.test.luau +91 -91
- package/vendor/rbxutil/enum-list/wally.toml +8 -8
- package/vendor/rbxutil/find/index.d.ts +20 -20
- package/vendor/rbxutil/find/init.luau +44 -44
- package/vendor/rbxutil/find/package.json +17 -17
- package/vendor/rbxutil/find/wally.toml +8 -8
- package/vendor/rbxutil/input/Gamepad.luau +559 -559
- package/vendor/rbxutil/input/Keyboard.luau +124 -124
- package/vendor/rbxutil/input/Mouse.luau +278 -278
- package/vendor/rbxutil/input/PreferredInput.luau +91 -91
- package/vendor/rbxutil/input/Touch.luau +120 -120
- package/vendor/rbxutil/input/init.luau +33 -33
- package/vendor/rbxutil/input/wally.toml +12 -12
- package/vendor/rbxutil/loader/index.d.ts +15 -15
- package/vendor/rbxutil/loader/init.luau +137 -137
- package/vendor/rbxutil/loader/wally.toml +8 -8
- package/vendor/rbxutil/log/index.d.ts +38 -38
- package/vendor/rbxutil/log/init.luau +746 -746
- package/vendor/rbxutil/log/wally.toml +8 -8
- package/vendor/rbxutil/net/init.luau +190 -190
- package/vendor/rbxutil/net/wally.toml +8 -8
- package/vendor/rbxutil/option/index.d.ts +44 -44
- package/vendor/rbxutil/option/init.luau +489 -489
- package/vendor/rbxutil/option/init.test.luau +342 -342
- package/vendor/rbxutil/option/wally.toml +8 -8
- package/vendor/rbxutil/pid/index.d.ts +53 -53
- package/vendor/rbxutil/pid/init.luau +195 -195
- package/vendor/rbxutil/pid/package.json +16 -16
- package/vendor/rbxutil/pid/wally.toml +9 -9
- package/vendor/rbxutil/quaternion/index.d.ts +117 -117
- package/vendor/rbxutil/quaternion/init.luau +570 -570
- package/vendor/rbxutil/quaternion/package.json +16 -16
- package/vendor/rbxutil/quaternion/wally.toml +9 -9
- package/vendor/rbxutil/query/index.d.ts +43 -43
- package/vendor/rbxutil/query/init.luau +117 -117
- package/vendor/rbxutil/query/package.json +18 -18
- package/vendor/rbxutil/query/wally.toml +9 -9
- package/vendor/rbxutil/sequent/index.d.ts +28 -28
- package/vendor/rbxutil/sequent/init.luau +340 -340
- package/vendor/rbxutil/sequent/package.json +16 -16
- package/vendor/rbxutil/sequent/wally.toml +9 -9
- package/vendor/rbxutil/ser/init.luau +175 -175
- package/vendor/rbxutil/ser/init.test.luau +50 -50
- package/vendor/rbxutil/ser/wally.toml +11 -11
- package/vendor/rbxutil/shake/index.d.ts +36 -36
- package/vendor/rbxutil/shake/init.luau +532 -532
- package/vendor/rbxutil/shake/init.test.luau +267 -267
- package/vendor/rbxutil/shake/package.json +16 -16
- package/vendor/rbxutil/shake/wally.toml +9 -9
- package/vendor/rbxutil/signal/index.d.ts +100 -100
- package/vendor/rbxutil/signal/init.luau +432 -432
- package/vendor/rbxutil/signal/init.test.luau +190 -190
- package/vendor/rbxutil/signal/package.json +17 -17
- package/vendor/rbxutil/signal/wally.toml +9 -9
- package/vendor/rbxutil/silo/TableWatcher.luau +65 -65
- package/vendor/rbxutil/silo/Util.luau +55 -55
- package/vendor/rbxutil/silo/init.luau +338 -338
- package/vendor/rbxutil/silo/init.test.luau +215 -215
- package/vendor/rbxutil/silo/wally.toml +8 -8
- package/vendor/rbxutil/spring/index.d.ts +40 -40
- package/vendor/rbxutil/spring/init.luau +97 -97
- package/vendor/rbxutil/spring/package.json +17 -17
- package/vendor/rbxutil/spring/wally.toml +8 -8
- package/vendor/rbxutil/stream/index.d.ts +88 -88
- package/vendor/rbxutil/stream/init.luau +597 -597
- package/vendor/rbxutil/stream/package.json +18 -18
- package/vendor/rbxutil/stream/wally.toml +9 -9
- package/vendor/rbxutil/streamable/Streamable.luau +202 -202
- package/vendor/rbxutil/streamable/StreamableUtil.luau +80 -80
- package/vendor/rbxutil/streamable/init.luau +8 -8
- package/vendor/rbxutil/streamable/wally.toml +12 -12
- package/vendor/rbxutil/symbol/init.luau +56 -56
- package/vendor/rbxutil/symbol/init.test.luau +37 -37
- package/vendor/rbxutil/symbol/wally.toml +8 -8
- package/vendor/rbxutil/table-util/init.luau +938 -938
- package/vendor/rbxutil/table-util/init.test.luau +439 -439
- package/vendor/rbxutil/task-queue/index.d.ts +27 -27
- package/vendor/rbxutil/task-queue/init.luau +97 -97
- package/vendor/rbxutil/task-queue/wally.toml +8 -8
- package/vendor/rbxutil/timer/index.d.ts +81 -81
- package/vendor/rbxutil/timer/init.luau +249 -249
- package/vendor/rbxutil/timer/init.test.luau +73 -73
- package/vendor/rbxutil/timer/wally.toml +11 -11
- package/vendor/rbxutil/tree/index.d.ts +15 -15
- package/vendor/rbxutil/tree/init.luau +137 -137
- package/vendor/rbxutil/tree/wally.toml +8 -8
- package/vendor/rbxutil/trove/index.d.ts +46 -46
- package/vendor/rbxutil/trove/init.luau +787 -787
- package/vendor/rbxutil/trove/init.test.luau +203 -203
- package/vendor/rbxutil/trove/wally.toml +8 -8
- package/vendor/rbxutil/typed-remote/init.luau +196 -196
- package/vendor/rbxutil/typed-remote/wally.toml +8 -8
- package/vendor/rbxutil/wait-for/index.d.ts +17 -17
- package/vendor/rbxutil/wait-for/init.luau +257 -257
- package/vendor/rbxutil/wait-for/init.test.luau +182 -182
- package/vendor/rbxutil/wait-for/wally.toml +11 -11
- package/vendor/t/t.lua +1350 -1350
- package/vendor/testez/Context.lua +26 -26
- package/vendor/testez/Expectation.lua +311 -311
- package/vendor/testez/ExpectationContext.lua +38 -38
- package/vendor/testez/LifecycleHooks.lua +89 -89
- package/vendor/testez/Reporters/TeamCityReporter.lua +101 -101
- package/vendor/testez/Reporters/TextReporter.lua +105 -105
- package/vendor/testez/Reporters/TextReporterQuiet.lua +96 -96
- package/vendor/testez/TestBootstrap.lua +146 -146
- package/vendor/testez/TestEnum.lua +27 -27
- package/vendor/testez/TestPlan.lua +304 -304
- package/vendor/testez/TestPlanner.lua +39 -39
- package/vendor/testez/TestResults.lua +111 -111
- package/vendor/testez/TestRunner.lua +188 -188
- package/vendor/testez/TestSession.lua +243 -243
- package/vendor/testez/init.lua +39 -39
|
@@ -1,112 +1,112 @@
|
|
|
1
|
-
--[[
|
|
2
|
-
Represents a tree of test results.
|
|
3
|
-
|
|
4
|
-
Each node in the tree corresponds directly to a node in a corresponding
|
|
5
|
-
TestPlan, accessible via the 'planNode' field.
|
|
6
|
-
|
|
7
|
-
TestResults objects are produced by TestRunner using TestSession as state.
|
|
8
|
-
]]
|
|
9
|
-
|
|
10
|
-
local TestEnum = require(script.Parent.TestEnum)
|
|
11
|
-
|
|
12
|
-
local STATUS_SYMBOLS = {
|
|
13
|
-
[TestEnum.TestStatus.Success] = "+",
|
|
14
|
-
[TestEnum.TestStatus.Failure] = "-",
|
|
15
|
-
[TestEnum.TestStatus.Skipped] = "~"
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
local TestResults = {}
|
|
19
|
-
|
|
20
|
-
TestResults.__index = TestResults
|
|
21
|
-
|
|
22
|
-
--[[
|
|
23
|
-
Create a new TestResults tree that's linked to the given TestPlan.
|
|
24
|
-
]]
|
|
25
|
-
function TestResults.new(plan)
|
|
26
|
-
local self = {
|
|
27
|
-
successCount = 0,
|
|
28
|
-
failureCount = 0,
|
|
29
|
-
skippedCount = 0,
|
|
30
|
-
planNode = plan,
|
|
31
|
-
children = {},
|
|
32
|
-
errors = {}
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
setmetatable(self, TestResults)
|
|
36
|
-
|
|
37
|
-
return self
|
|
38
|
-
end
|
|
39
|
-
|
|
40
|
-
--[[
|
|
41
|
-
Create a new result node that can be inserted into a TestResult tree.
|
|
42
|
-
]]
|
|
43
|
-
function TestResults.createNode(planNode)
|
|
44
|
-
local node = {
|
|
45
|
-
planNode = planNode,
|
|
46
|
-
children = {},
|
|
47
|
-
errors = {},
|
|
48
|
-
status = nil
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
return node
|
|
52
|
-
end
|
|
53
|
-
|
|
54
|
-
--[[
|
|
55
|
-
Visit all test result nodes, depth-first.
|
|
56
|
-
]]
|
|
57
|
-
function TestResults:visitAllNodes(callback, root)
|
|
58
|
-
root = root or self
|
|
59
|
-
|
|
60
|
-
for _, child in ipairs(root.children) do
|
|
61
|
-
callback(child)
|
|
62
|
-
|
|
63
|
-
self:visitAllNodes(callback, child)
|
|
64
|
-
end
|
|
65
|
-
end
|
|
66
|
-
|
|
67
|
-
--[[
|
|
68
|
-
Creates a debug visualization of the test results.
|
|
69
|
-
]]
|
|
70
|
-
function TestResults:visualize(root, level)
|
|
71
|
-
root = root or self
|
|
72
|
-
level = level or 0
|
|
73
|
-
|
|
74
|
-
local buffer = {}
|
|
75
|
-
|
|
76
|
-
for _, child in ipairs(root.children) do
|
|
77
|
-
if child.planNode.type == TestEnum.NodeType.It then
|
|
78
|
-
local symbol = STATUS_SYMBOLS[child.status] or "?"
|
|
79
|
-
local str = ("%s[%s] %s"):format(
|
|
80
|
-
(" "):rep(3 * level),
|
|
81
|
-
symbol,
|
|
82
|
-
child.planNode.phrase
|
|
83
|
-
)
|
|
84
|
-
|
|
85
|
-
if child.messages and #child.messages > 0 then
|
|
86
|
-
str = str .. "\n " .. (" "):rep(3 * level) .. table.concat(child.messages, "\n " .. (" "):rep(3 * level))
|
|
87
|
-
end
|
|
88
|
-
|
|
89
|
-
table.insert(buffer, str)
|
|
90
|
-
else
|
|
91
|
-
local str = ("%s%s"):format(
|
|
92
|
-
(" "):rep(3 * level),
|
|
93
|
-
child.planNode.phrase or ""
|
|
94
|
-
)
|
|
95
|
-
|
|
96
|
-
if child.status then
|
|
97
|
-
str = str .. (" (%s)"):format(child.status)
|
|
98
|
-
end
|
|
99
|
-
|
|
100
|
-
table.insert(buffer, str)
|
|
101
|
-
|
|
102
|
-
if #child.children > 0 then
|
|
103
|
-
local text = self:visualize(child, level + 1)
|
|
104
|
-
table.insert(buffer, text)
|
|
105
|
-
end
|
|
106
|
-
end
|
|
107
|
-
end
|
|
108
|
-
|
|
109
|
-
return table.concat(buffer, "\n")
|
|
110
|
-
end
|
|
111
|
-
|
|
1
|
+
--[[
|
|
2
|
+
Represents a tree of test results.
|
|
3
|
+
|
|
4
|
+
Each node in the tree corresponds directly to a node in a corresponding
|
|
5
|
+
TestPlan, accessible via the 'planNode' field.
|
|
6
|
+
|
|
7
|
+
TestResults objects are produced by TestRunner using TestSession as state.
|
|
8
|
+
]]
|
|
9
|
+
|
|
10
|
+
local TestEnum = require(script.Parent.TestEnum)
|
|
11
|
+
|
|
12
|
+
local STATUS_SYMBOLS = {
|
|
13
|
+
[TestEnum.TestStatus.Success] = "+",
|
|
14
|
+
[TestEnum.TestStatus.Failure] = "-",
|
|
15
|
+
[TestEnum.TestStatus.Skipped] = "~"
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
local TestResults = {}
|
|
19
|
+
|
|
20
|
+
TestResults.__index = TestResults
|
|
21
|
+
|
|
22
|
+
--[[
|
|
23
|
+
Create a new TestResults tree that's linked to the given TestPlan.
|
|
24
|
+
]]
|
|
25
|
+
function TestResults.new(plan)
|
|
26
|
+
local self = {
|
|
27
|
+
successCount = 0,
|
|
28
|
+
failureCount = 0,
|
|
29
|
+
skippedCount = 0,
|
|
30
|
+
planNode = plan,
|
|
31
|
+
children = {},
|
|
32
|
+
errors = {}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
setmetatable(self, TestResults)
|
|
36
|
+
|
|
37
|
+
return self
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
--[[
|
|
41
|
+
Create a new result node that can be inserted into a TestResult tree.
|
|
42
|
+
]]
|
|
43
|
+
function TestResults.createNode(planNode)
|
|
44
|
+
local node = {
|
|
45
|
+
planNode = planNode,
|
|
46
|
+
children = {},
|
|
47
|
+
errors = {},
|
|
48
|
+
status = nil
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
return node
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
--[[
|
|
55
|
+
Visit all test result nodes, depth-first.
|
|
56
|
+
]]
|
|
57
|
+
function TestResults:visitAllNodes(callback, root)
|
|
58
|
+
root = root or self
|
|
59
|
+
|
|
60
|
+
for _, child in ipairs(root.children) do
|
|
61
|
+
callback(child)
|
|
62
|
+
|
|
63
|
+
self:visitAllNodes(callback, child)
|
|
64
|
+
end
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
--[[
|
|
68
|
+
Creates a debug visualization of the test results.
|
|
69
|
+
]]
|
|
70
|
+
function TestResults:visualize(root, level)
|
|
71
|
+
root = root or self
|
|
72
|
+
level = level or 0
|
|
73
|
+
|
|
74
|
+
local buffer = {}
|
|
75
|
+
|
|
76
|
+
for _, child in ipairs(root.children) do
|
|
77
|
+
if child.planNode.type == TestEnum.NodeType.It then
|
|
78
|
+
local symbol = STATUS_SYMBOLS[child.status] or "?"
|
|
79
|
+
local str = ("%s[%s] %s"):format(
|
|
80
|
+
(" "):rep(3 * level),
|
|
81
|
+
symbol,
|
|
82
|
+
child.planNode.phrase
|
|
83
|
+
)
|
|
84
|
+
|
|
85
|
+
if child.messages and #child.messages > 0 then
|
|
86
|
+
str = str .. "\n " .. (" "):rep(3 * level) .. table.concat(child.messages, "\n " .. (" "):rep(3 * level))
|
|
87
|
+
end
|
|
88
|
+
|
|
89
|
+
table.insert(buffer, str)
|
|
90
|
+
else
|
|
91
|
+
local str = ("%s%s"):format(
|
|
92
|
+
(" "):rep(3 * level),
|
|
93
|
+
child.planNode.phrase or ""
|
|
94
|
+
)
|
|
95
|
+
|
|
96
|
+
if child.status then
|
|
97
|
+
str = str .. (" (%s)"):format(child.status)
|
|
98
|
+
end
|
|
99
|
+
|
|
100
|
+
table.insert(buffer, str)
|
|
101
|
+
|
|
102
|
+
if #child.children > 0 then
|
|
103
|
+
local text = self:visualize(child, level + 1)
|
|
104
|
+
table.insert(buffer, text)
|
|
105
|
+
end
|
|
106
|
+
end
|
|
107
|
+
end
|
|
108
|
+
|
|
109
|
+
return table.concat(buffer, "\n")
|
|
110
|
+
end
|
|
111
|
+
|
|
112
112
|
return TestResults
|
|
@@ -1,188 +1,188 @@
|
|
|
1
|
-
--[[
|
|
2
|
-
Contains the logic to run a test plan and gather test results from it.
|
|
3
|
-
|
|
4
|
-
TestRunner accepts a TestPlan object, executes the planned tests, and
|
|
5
|
-
produces a TestResults object. While the tests are running, the system's
|
|
6
|
-
state is contained inside a TestSession object.
|
|
7
|
-
]]
|
|
8
|
-
|
|
9
|
-
local TestEnum = require(script.Parent.TestEnum)
|
|
10
|
-
local TestSession = require(script.Parent.TestSession)
|
|
11
|
-
local LifecycleHooks = require(script.Parent.LifecycleHooks)
|
|
12
|
-
|
|
13
|
-
local RUNNING_GLOBAL = "__TESTEZ_RUNNING_TEST__"
|
|
14
|
-
|
|
15
|
-
local TestRunner = {
|
|
16
|
-
environment = {}
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
local function wrapExpectContextWithPublicApi(expectationContext)
|
|
20
|
-
return setmetatable({
|
|
21
|
-
extend = function(...)
|
|
22
|
-
expectationContext:extend(...)
|
|
23
|
-
end,
|
|
24
|
-
}, {
|
|
25
|
-
__call = function(_self, ...)
|
|
26
|
-
return expectationContext:startExpectationChain(...)
|
|
27
|
-
end,
|
|
28
|
-
})
|
|
29
|
-
end
|
|
30
|
-
|
|
31
|
-
--[[
|
|
32
|
-
Runs the given TestPlan and returns a TestResults object representing the
|
|
33
|
-
results of the run.
|
|
34
|
-
]]
|
|
35
|
-
function TestRunner.runPlan(plan)
|
|
36
|
-
local session = TestSession.new(plan)
|
|
37
|
-
local lifecycleHooks = LifecycleHooks.new()
|
|
38
|
-
|
|
39
|
-
local exclusiveNodes = plan:findNodes(function(node)
|
|
40
|
-
return node.modifier == TestEnum.NodeModifier.Focus
|
|
41
|
-
end)
|
|
42
|
-
|
|
43
|
-
session.hasFocusNodes = #exclusiveNodes > 0
|
|
44
|
-
|
|
45
|
-
TestRunner.runPlanNode(session, plan, lifecycleHooks)
|
|
46
|
-
|
|
47
|
-
return session:finalize()
|
|
48
|
-
end
|
|
49
|
-
|
|
50
|
-
--[[
|
|
51
|
-
Run the given test plan node and its descendants, using the given test
|
|
52
|
-
session to store all of the results.
|
|
53
|
-
]]
|
|
54
|
-
function TestRunner.runPlanNode(session, planNode, lifecycleHooks)
|
|
55
|
-
local function runCallback(callback, messagePrefix)
|
|
56
|
-
local success = true
|
|
57
|
-
local errorMessage
|
|
58
|
-
-- Any code can check RUNNING_GLOBAL to fork behavior based on
|
|
59
|
-
-- whether a test is running. We use this to avoid accessing
|
|
60
|
-
-- protected APIs; it's a workaround that will go away someday.
|
|
61
|
-
_G[RUNNING_GLOBAL] = true
|
|
62
|
-
|
|
63
|
-
messagePrefix = messagePrefix or ""
|
|
64
|
-
|
|
65
|
-
local testEnvironment = getfenv(callback)
|
|
66
|
-
|
|
67
|
-
for key, value in pairs(TestRunner.environment) do
|
|
68
|
-
testEnvironment[key] = value
|
|
69
|
-
end
|
|
70
|
-
|
|
71
|
-
testEnvironment.fail = function(message)
|
|
72
|
-
if message == nil then
|
|
73
|
-
message = "fail() was called."
|
|
74
|
-
end
|
|
75
|
-
|
|
76
|
-
success = false
|
|
77
|
-
errorMessage = messagePrefix .. debug.traceback(tostring(message), 2)
|
|
78
|
-
end
|
|
79
|
-
|
|
80
|
-
testEnvironment.expect = wrapExpectContextWithPublicApi(session:getExpectationContext())
|
|
81
|
-
|
|
82
|
-
local context = session:getContext()
|
|
83
|
-
|
|
84
|
-
local nodeSuccess, nodeResult = xpcall(
|
|
85
|
-
function()
|
|
86
|
-
callback(context)
|
|
87
|
-
end,
|
|
88
|
-
function(message)
|
|
89
|
-
return messagePrefix .. debug.traceback(tostring(message), 2)
|
|
90
|
-
end
|
|
91
|
-
)
|
|
92
|
-
|
|
93
|
-
-- If a node threw an error, we prefer to use that message over
|
|
94
|
-
-- one created by fail() if it was set.
|
|
95
|
-
if not nodeSuccess then
|
|
96
|
-
success = false
|
|
97
|
-
errorMessage = nodeResult
|
|
98
|
-
end
|
|
99
|
-
|
|
100
|
-
_G[RUNNING_GLOBAL] = nil
|
|
101
|
-
|
|
102
|
-
return success, errorMessage
|
|
103
|
-
end
|
|
104
|
-
|
|
105
|
-
local function runNode(childPlanNode)
|
|
106
|
-
-- Errors can be set either via `error` propagating upwards or
|
|
107
|
-
-- by a test calling fail([message]).
|
|
108
|
-
|
|
109
|
-
for _, hook in ipairs(lifecycleHooks:getBeforeEachHooks()) do
|
|
110
|
-
local success, errorMessage = runCallback(hook, "beforeEach hook: ")
|
|
111
|
-
if not success then
|
|
112
|
-
return false, errorMessage
|
|
113
|
-
end
|
|
114
|
-
end
|
|
115
|
-
|
|
116
|
-
local testSuccess, testErrorMessage = runCallback(childPlanNode.callback)
|
|
117
|
-
|
|
118
|
-
for _, hook in ipairs(lifecycleHooks:getAfterEachHooks()) do
|
|
119
|
-
local success, errorMessage = runCallback(hook, "afterEach hook: ")
|
|
120
|
-
if not success then
|
|
121
|
-
if not testSuccess then
|
|
122
|
-
return false, testErrorMessage .. "\nWhile cleaning up the failed test another error was found:\n" .. errorMessage
|
|
123
|
-
end
|
|
124
|
-
return false, errorMessage
|
|
125
|
-
end
|
|
126
|
-
end
|
|
127
|
-
|
|
128
|
-
if not testSuccess then
|
|
129
|
-
return false, testErrorMessage
|
|
130
|
-
end
|
|
131
|
-
|
|
132
|
-
return true, nil
|
|
133
|
-
end
|
|
134
|
-
|
|
135
|
-
lifecycleHooks:pushHooksFrom(planNode)
|
|
136
|
-
|
|
137
|
-
local halt = false
|
|
138
|
-
for _, hook in ipairs(lifecycleHooks:getBeforeAllHooks()) do
|
|
139
|
-
local success, errorMessage = runCallback(hook, "beforeAll hook: ")
|
|
140
|
-
if not success then
|
|
141
|
-
session:addDummyError("beforeAll", errorMessage)
|
|
142
|
-
halt = true
|
|
143
|
-
end
|
|
144
|
-
end
|
|
145
|
-
|
|
146
|
-
if not halt then
|
|
147
|
-
for _, childPlanNode in ipairs(planNode.children) do
|
|
148
|
-
if childPlanNode.type == TestEnum.NodeType.It then
|
|
149
|
-
session:pushNode(childPlanNode)
|
|
150
|
-
if session:shouldSkip() then
|
|
151
|
-
session:setSkipped()
|
|
152
|
-
else
|
|
153
|
-
local success, errorMessage = runNode(childPlanNode)
|
|
154
|
-
|
|
155
|
-
if success then
|
|
156
|
-
session:setSuccess()
|
|
157
|
-
else
|
|
158
|
-
session:setError(errorMessage)
|
|
159
|
-
end
|
|
160
|
-
end
|
|
161
|
-
session:popNode()
|
|
162
|
-
elseif childPlanNode.type == TestEnum.NodeType.Describe then
|
|
163
|
-
session:pushNode(childPlanNode)
|
|
164
|
-
TestRunner.runPlanNode(session, childPlanNode, lifecycleHooks)
|
|
165
|
-
|
|
166
|
-
-- Did we have an error trying build a test plan?
|
|
167
|
-
if childPlanNode.loadError then
|
|
168
|
-
local message = "Error during planning: " .. childPlanNode.loadError
|
|
169
|
-
session:setError(message)
|
|
170
|
-
else
|
|
171
|
-
session:setStatusFromChildren()
|
|
172
|
-
end
|
|
173
|
-
session:popNode()
|
|
174
|
-
end
|
|
175
|
-
end
|
|
176
|
-
end
|
|
177
|
-
|
|
178
|
-
for _, hook in ipairs(lifecycleHooks:getAfterAllHooks()) do
|
|
179
|
-
local success, errorMessage = runCallback(hook, "afterAll hook: ")
|
|
180
|
-
if not success then
|
|
181
|
-
session:addDummyError("afterAll", errorMessage)
|
|
182
|
-
end
|
|
183
|
-
end
|
|
184
|
-
|
|
185
|
-
lifecycleHooks:popHooks()
|
|
186
|
-
end
|
|
187
|
-
|
|
188
|
-
return TestRunner
|
|
1
|
+
--[[
|
|
2
|
+
Contains the logic to run a test plan and gather test results from it.
|
|
3
|
+
|
|
4
|
+
TestRunner accepts a TestPlan object, executes the planned tests, and
|
|
5
|
+
produces a TestResults object. While the tests are running, the system's
|
|
6
|
+
state is contained inside a TestSession object.
|
|
7
|
+
]]
|
|
8
|
+
|
|
9
|
+
local TestEnum = require(script.Parent.TestEnum)
|
|
10
|
+
local TestSession = require(script.Parent.TestSession)
|
|
11
|
+
local LifecycleHooks = require(script.Parent.LifecycleHooks)
|
|
12
|
+
|
|
13
|
+
local RUNNING_GLOBAL = "__TESTEZ_RUNNING_TEST__"
|
|
14
|
+
|
|
15
|
+
local TestRunner = {
|
|
16
|
+
environment = {}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
local function wrapExpectContextWithPublicApi(expectationContext)
|
|
20
|
+
return setmetatable({
|
|
21
|
+
extend = function(...)
|
|
22
|
+
expectationContext:extend(...)
|
|
23
|
+
end,
|
|
24
|
+
}, {
|
|
25
|
+
__call = function(_self, ...)
|
|
26
|
+
return expectationContext:startExpectationChain(...)
|
|
27
|
+
end,
|
|
28
|
+
})
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
--[[
|
|
32
|
+
Runs the given TestPlan and returns a TestResults object representing the
|
|
33
|
+
results of the run.
|
|
34
|
+
]]
|
|
35
|
+
function TestRunner.runPlan(plan)
|
|
36
|
+
local session = TestSession.new(plan)
|
|
37
|
+
local lifecycleHooks = LifecycleHooks.new()
|
|
38
|
+
|
|
39
|
+
local exclusiveNodes = plan:findNodes(function(node)
|
|
40
|
+
return node.modifier == TestEnum.NodeModifier.Focus
|
|
41
|
+
end)
|
|
42
|
+
|
|
43
|
+
session.hasFocusNodes = #exclusiveNodes > 0
|
|
44
|
+
|
|
45
|
+
TestRunner.runPlanNode(session, plan, lifecycleHooks)
|
|
46
|
+
|
|
47
|
+
return session:finalize()
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
--[[
|
|
51
|
+
Run the given test plan node and its descendants, using the given test
|
|
52
|
+
session to store all of the results.
|
|
53
|
+
]]
|
|
54
|
+
function TestRunner.runPlanNode(session, planNode, lifecycleHooks)
|
|
55
|
+
local function runCallback(callback, messagePrefix)
|
|
56
|
+
local success = true
|
|
57
|
+
local errorMessage
|
|
58
|
+
-- Any code can check RUNNING_GLOBAL to fork behavior based on
|
|
59
|
+
-- whether a test is running. We use this to avoid accessing
|
|
60
|
+
-- protected APIs; it's a workaround that will go away someday.
|
|
61
|
+
_G[RUNNING_GLOBAL] = true
|
|
62
|
+
|
|
63
|
+
messagePrefix = messagePrefix or ""
|
|
64
|
+
|
|
65
|
+
local testEnvironment = getfenv(callback)
|
|
66
|
+
|
|
67
|
+
for key, value in pairs(TestRunner.environment) do
|
|
68
|
+
testEnvironment[key] = value
|
|
69
|
+
end
|
|
70
|
+
|
|
71
|
+
testEnvironment.fail = function(message)
|
|
72
|
+
if message == nil then
|
|
73
|
+
message = "fail() was called."
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
success = false
|
|
77
|
+
errorMessage = messagePrefix .. debug.traceback(tostring(message), 2)
|
|
78
|
+
end
|
|
79
|
+
|
|
80
|
+
testEnvironment.expect = wrapExpectContextWithPublicApi(session:getExpectationContext())
|
|
81
|
+
|
|
82
|
+
local context = session:getContext()
|
|
83
|
+
|
|
84
|
+
local nodeSuccess, nodeResult = xpcall(
|
|
85
|
+
function()
|
|
86
|
+
callback(context)
|
|
87
|
+
end,
|
|
88
|
+
function(message)
|
|
89
|
+
return messagePrefix .. debug.traceback(tostring(message), 2)
|
|
90
|
+
end
|
|
91
|
+
)
|
|
92
|
+
|
|
93
|
+
-- If a node threw an error, we prefer to use that message over
|
|
94
|
+
-- one created by fail() if it was set.
|
|
95
|
+
if not nodeSuccess then
|
|
96
|
+
success = false
|
|
97
|
+
errorMessage = nodeResult
|
|
98
|
+
end
|
|
99
|
+
|
|
100
|
+
_G[RUNNING_GLOBAL] = nil
|
|
101
|
+
|
|
102
|
+
return success, errorMessage
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
local function runNode(childPlanNode)
|
|
106
|
+
-- Errors can be set either via `error` propagating upwards or
|
|
107
|
+
-- by a test calling fail([message]).
|
|
108
|
+
|
|
109
|
+
for _, hook in ipairs(lifecycleHooks:getBeforeEachHooks()) do
|
|
110
|
+
local success, errorMessage = runCallback(hook, "beforeEach hook: ")
|
|
111
|
+
if not success then
|
|
112
|
+
return false, errorMessage
|
|
113
|
+
end
|
|
114
|
+
end
|
|
115
|
+
|
|
116
|
+
local testSuccess, testErrorMessage = runCallback(childPlanNode.callback)
|
|
117
|
+
|
|
118
|
+
for _, hook in ipairs(lifecycleHooks:getAfterEachHooks()) do
|
|
119
|
+
local success, errorMessage = runCallback(hook, "afterEach hook: ")
|
|
120
|
+
if not success then
|
|
121
|
+
if not testSuccess then
|
|
122
|
+
return false, testErrorMessage .. "\nWhile cleaning up the failed test another error was found:\n" .. errorMessage
|
|
123
|
+
end
|
|
124
|
+
return false, errorMessage
|
|
125
|
+
end
|
|
126
|
+
end
|
|
127
|
+
|
|
128
|
+
if not testSuccess then
|
|
129
|
+
return false, testErrorMessage
|
|
130
|
+
end
|
|
131
|
+
|
|
132
|
+
return true, nil
|
|
133
|
+
end
|
|
134
|
+
|
|
135
|
+
lifecycleHooks:pushHooksFrom(planNode)
|
|
136
|
+
|
|
137
|
+
local halt = false
|
|
138
|
+
for _, hook in ipairs(lifecycleHooks:getBeforeAllHooks()) do
|
|
139
|
+
local success, errorMessage = runCallback(hook, "beforeAll hook: ")
|
|
140
|
+
if not success then
|
|
141
|
+
session:addDummyError("beforeAll", errorMessage)
|
|
142
|
+
halt = true
|
|
143
|
+
end
|
|
144
|
+
end
|
|
145
|
+
|
|
146
|
+
if not halt then
|
|
147
|
+
for _, childPlanNode in ipairs(planNode.children) do
|
|
148
|
+
if childPlanNode.type == TestEnum.NodeType.It then
|
|
149
|
+
session:pushNode(childPlanNode)
|
|
150
|
+
if session:shouldSkip() then
|
|
151
|
+
session:setSkipped()
|
|
152
|
+
else
|
|
153
|
+
local success, errorMessage = runNode(childPlanNode)
|
|
154
|
+
|
|
155
|
+
if success then
|
|
156
|
+
session:setSuccess()
|
|
157
|
+
else
|
|
158
|
+
session:setError(errorMessage)
|
|
159
|
+
end
|
|
160
|
+
end
|
|
161
|
+
session:popNode()
|
|
162
|
+
elseif childPlanNode.type == TestEnum.NodeType.Describe then
|
|
163
|
+
session:pushNode(childPlanNode)
|
|
164
|
+
TestRunner.runPlanNode(session, childPlanNode, lifecycleHooks)
|
|
165
|
+
|
|
166
|
+
-- Did we have an error trying build a test plan?
|
|
167
|
+
if childPlanNode.loadError then
|
|
168
|
+
local message = "Error during planning: " .. childPlanNode.loadError
|
|
169
|
+
session:setError(message)
|
|
170
|
+
else
|
|
171
|
+
session:setStatusFromChildren()
|
|
172
|
+
end
|
|
173
|
+
session:popNode()
|
|
174
|
+
end
|
|
175
|
+
end
|
|
176
|
+
end
|
|
177
|
+
|
|
178
|
+
for _, hook in ipairs(lifecycleHooks:getAfterAllHooks()) do
|
|
179
|
+
local success, errorMessage = runCallback(hook, "afterAll hook: ")
|
|
180
|
+
if not success then
|
|
181
|
+
session:addDummyError("afterAll", errorMessage)
|
|
182
|
+
end
|
|
183
|
+
end
|
|
184
|
+
|
|
185
|
+
lifecycleHooks:popHooks()
|
|
186
|
+
end
|
|
187
|
+
|
|
188
|
+
return TestRunner
|