roblox-opencode 1.0.0

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.
Files changed (248) hide show
  1. package/README.md +122 -0
  2. package/commands/setup-game.md +108 -0
  3. package/commands/sync-check.md +53 -0
  4. package/core/roblox-core.md +93 -0
  5. package/dist/server.js +167 -0
  6. package/package.json +35 -0
  7. package/skills/roblox-analytics/SKILL.md +277 -0
  8. package/skills/roblox-analytics/references/event-batcher.luau +75 -0
  9. package/skills/roblox-animation-vfx/SKILL.md +1325 -0
  10. package/skills/roblox-architecture/SKILL.md +863 -0
  11. package/skills/roblox-architecture/references/combat-systems.md +1381 -0
  12. package/skills/roblox-code-review/SKILL.md +687 -0
  13. package/skills/roblox-data/SKILL.md +889 -0
  14. package/skills/roblox-data/references/inventory-systems.md +1729 -0
  15. package/skills/roblox-debug/SKILL.md +99 -0
  16. package/skills/roblox-gui/SKILL.md +1103 -0
  17. package/skills/roblox-gui-fusion/SKILL.md +150 -0
  18. package/skills/roblox-gui-fusion/references/inventory.luau +427 -0
  19. package/skills/roblox-gui-fusion/references/settings-menu.luau +579 -0
  20. package/skills/roblox-gui-fusion/references/shop.luau +411 -0
  21. package/skills/roblox-luau-mastery/SKILL.md +1519 -0
  22. package/skills/roblox-monetization/SKILL.md +1084 -0
  23. package/skills/roblox-monetization/references/process-receipt.luau +131 -0
  24. package/skills/roblox-networking/SKILL.md +669 -0
  25. package/skills/roblox-networking/references/remote-validator.luau +193 -0
  26. package/skills/roblox-publish-checklist/SKILL.md +128 -0
  27. package/skills/roblox-runtime/SKILL.md +753 -0
  28. package/skills/roblox-sharp-edges/SKILL.md +295 -0
  29. package/skills/roblox-sync/SKILL.md +126 -0
  30. package/skills/roblox-testing/SKILL.md +943 -0
  31. package/skills/roblox-tooling/SKILL.md +150 -0
  32. package/vendor/LICENSES/ProfileStore-LICENSE +201 -0
  33. package/vendor/LICENSES/RbxUtil-LICENSE +7 -0
  34. package/vendor/LICENSES/promise-LICENSE +21 -0
  35. package/vendor/LICENSES/t-LICENSE +21 -0
  36. package/vendor/LICENSES/testez-LICENSE +201 -0
  37. package/vendor/README.md +84 -0
  38. package/vendor/fusion/Animation/ExternalTime.luau +84 -0
  39. package/vendor/fusion/Animation/Spring.luau +322 -0
  40. package/vendor/fusion/Animation/Stopwatch.luau +128 -0
  41. package/vendor/fusion/Animation/Tween.luau +187 -0
  42. package/vendor/fusion/Animation/getTweenDuration.luau +27 -0
  43. package/vendor/fusion/Animation/getTweenRatio.luau +47 -0
  44. package/vendor/fusion/Animation/lerpType.luau +164 -0
  45. package/vendor/fusion/Animation/packType.luau +100 -0
  46. package/vendor/fusion/Animation/springCoefficients.luau +80 -0
  47. package/vendor/fusion/Animation/unpackType.luau +103 -0
  48. package/vendor/fusion/Colour/Oklab.luau +70 -0
  49. package/vendor/fusion/Colour/sRGB.luau +55 -0
  50. package/vendor/fusion/External.luau +168 -0
  51. package/vendor/fusion/ExternalDebug.luau +70 -0
  52. package/vendor/fusion/Graph/Observer.luau +114 -0
  53. package/vendor/fusion/Graph/castToGraph.luau +29 -0
  54. package/vendor/fusion/Graph/change.luau +81 -0
  55. package/vendor/fusion/Graph/depend.luau +33 -0
  56. package/vendor/fusion/Graph/evaluate.luau +56 -0
  57. package/vendor/fusion/Instances/Attribute.luau +58 -0
  58. package/vendor/fusion/Instances/AttributeChange.luau +47 -0
  59. package/vendor/fusion/Instances/AttributeOut.luau +63 -0
  60. package/vendor/fusion/Instances/Child.luau +21 -0
  61. package/vendor/fusion/Instances/Children.luau +148 -0
  62. package/vendor/fusion/Instances/Hydrate.luau +33 -0
  63. package/vendor/fusion/Instances/New.luau +53 -0
  64. package/vendor/fusion/Instances/OnChange.luau +50 -0
  65. package/vendor/fusion/Instances/OnEvent.luau +54 -0
  66. package/vendor/fusion/Instances/Out.luau +69 -0
  67. package/vendor/fusion/Instances/applyInstanceProps.luau +149 -0
  68. package/vendor/fusion/Instances/defaultProps.luau +194 -0
  69. package/vendor/fusion/LICENSE +21 -0
  70. package/vendor/fusion/Logging/formatError.luau +49 -0
  71. package/vendor/fusion/Logging/messages.luau +52 -0
  72. package/vendor/fusion/Logging/parseError.luau +25 -0
  73. package/vendor/fusion/Memory/checkLifetime.luau +134 -0
  74. package/vendor/fusion/Memory/deriveScope.luau +24 -0
  75. package/vendor/fusion/Memory/deriveScopeImpl.luau +45 -0
  76. package/vendor/fusion/Memory/doCleanup.luau +79 -0
  77. package/vendor/fusion/Memory/innerScope.luau +34 -0
  78. package/vendor/fusion/Memory/legacyCleanup.luau +18 -0
  79. package/vendor/fusion/Memory/needsDestruction.luau +17 -0
  80. package/vendor/fusion/Memory/poisonScope.luau +34 -0
  81. package/vendor/fusion/Memory/scopePool.luau +55 -0
  82. package/vendor/fusion/Memory/scoped.luau +27 -0
  83. package/vendor/fusion/Memory/whichLivesLonger.luau +75 -0
  84. package/vendor/fusion/RobloxExternal.luau +98 -0
  85. package/vendor/fusion/State/Computed.luau +139 -0
  86. package/vendor/fusion/State/For/Disassembly.luau +211 -0
  87. package/vendor/fusion/State/For/ForTypes.luau +30 -0
  88. package/vendor/fusion/State/For/init.luau +110 -0
  89. package/vendor/fusion/State/ForKeys.luau +94 -0
  90. package/vendor/fusion/State/ForPairs.luau +97 -0
  91. package/vendor/fusion/State/ForValues.luau +94 -0
  92. package/vendor/fusion/State/Value.luau +88 -0
  93. package/vendor/fusion/State/castToState.luau +26 -0
  94. package/vendor/fusion/State/peek.luau +31 -0
  95. package/vendor/fusion/State/updateAll.luau +1 -0
  96. package/vendor/fusion/Types.luau +314 -0
  97. package/vendor/fusion/Utility/Contextual.luau +91 -0
  98. package/vendor/fusion/Utility/Safe.luau +23 -0
  99. package/vendor/fusion/Utility/isSimilar.luau +29 -0
  100. package/vendor/fusion/Utility/merge.luau +35 -0
  101. package/vendor/fusion/Utility/nameOf.luau +35 -0
  102. package/vendor/fusion/Utility/never.luau +14 -0
  103. package/vendor/fusion/Utility/nicknames.luau +11 -0
  104. package/vendor/fusion/Utility/xtypeof.luau +27 -0
  105. package/vendor/fusion/init.luau +82 -0
  106. package/vendor/profilestore/init.luau +2243 -0
  107. package/vendor/promise/init.luau +1982 -0
  108. package/vendor/rbxutil/buffer-util/Buffer.test.luau +25 -0
  109. package/vendor/rbxutil/buffer-util/BufferReader.luau +228 -0
  110. package/vendor/rbxutil/buffer-util/BufferWriter.luau +269 -0
  111. package/vendor/rbxutil/buffer-util/DataTypeBuffer.luau +223 -0
  112. package/vendor/rbxutil/buffer-util/Types.luau +60 -0
  113. package/vendor/rbxutil/buffer-util/index.d.ts +153 -0
  114. package/vendor/rbxutil/buffer-util/init.luau +41 -0
  115. package/vendor/rbxutil/buffer-util/package.json +16 -0
  116. package/vendor/rbxutil/buffer-util/wally.toml +9 -0
  117. package/vendor/rbxutil/comm/Client/ClientComm.luau +232 -0
  118. package/vendor/rbxutil/comm/Client/ClientRemoteProperty.luau +156 -0
  119. package/vendor/rbxutil/comm/Client/ClientRemoteSignal.luau +109 -0
  120. package/vendor/rbxutil/comm/Client/init.luau +135 -0
  121. package/vendor/rbxutil/comm/Server/RemoteProperty.luau +295 -0
  122. package/vendor/rbxutil/comm/Server/RemoteSignal.luau +211 -0
  123. package/vendor/rbxutil/comm/Server/ServerComm.luau +211 -0
  124. package/vendor/rbxutil/comm/Server/init.luau +140 -0
  125. package/vendor/rbxutil/comm/Types.luau +18 -0
  126. package/vendor/rbxutil/comm/Util.luau +27 -0
  127. package/vendor/rbxutil/comm/init.luau +35 -0
  128. package/vendor/rbxutil/comm/wally.toml +13 -0
  129. package/vendor/rbxutil/component/init.luau +759 -0
  130. package/vendor/rbxutil/component/init.test.luau +311 -0
  131. package/vendor/rbxutil/component/wally.toml +14 -0
  132. package/vendor/rbxutil/concur/init.luau +542 -0
  133. package/vendor/rbxutil/concur/init.test.luau +364 -0
  134. package/vendor/rbxutil/concur/wally.toml +8 -0
  135. package/vendor/rbxutil/enum-list/init.luau +101 -0
  136. package/vendor/rbxutil/enum-list/init.test.luau +91 -0
  137. package/vendor/rbxutil/enum-list/wally.toml +8 -0
  138. package/vendor/rbxutil/find/index.d.ts +20 -0
  139. package/vendor/rbxutil/find/init.luau +44 -0
  140. package/vendor/rbxutil/find/package.json +17 -0
  141. package/vendor/rbxutil/find/wally.toml +8 -0
  142. package/vendor/rbxutil/input/Gamepad.luau +559 -0
  143. package/vendor/rbxutil/input/Keyboard.luau +124 -0
  144. package/vendor/rbxutil/input/Mouse.luau +278 -0
  145. package/vendor/rbxutil/input/PreferredInput.luau +91 -0
  146. package/vendor/rbxutil/input/Touch.luau +120 -0
  147. package/vendor/rbxutil/input/init.luau +33 -0
  148. package/vendor/rbxutil/input/wally.toml +12 -0
  149. package/vendor/rbxutil/loader/index.d.ts +15 -0
  150. package/vendor/rbxutil/loader/init.luau +137 -0
  151. package/vendor/rbxutil/loader/wally.toml +8 -0
  152. package/vendor/rbxutil/log/index.d.ts +38 -0
  153. package/vendor/rbxutil/log/init.luau +746 -0
  154. package/vendor/rbxutil/log/wally.toml +8 -0
  155. package/vendor/rbxutil/net/init.luau +190 -0
  156. package/vendor/rbxutil/net/wally.toml +8 -0
  157. package/vendor/rbxutil/option/index.d.ts +44 -0
  158. package/vendor/rbxutil/option/init.luau +489 -0
  159. package/vendor/rbxutil/option/init.test.luau +342 -0
  160. package/vendor/rbxutil/option/wally.toml +8 -0
  161. package/vendor/rbxutil/pid/index.d.ts +53 -0
  162. package/vendor/rbxutil/pid/init.luau +195 -0
  163. package/vendor/rbxutil/pid/package.json +16 -0
  164. package/vendor/rbxutil/pid/wally.toml +9 -0
  165. package/vendor/rbxutil/quaternion/index.d.ts +117 -0
  166. package/vendor/rbxutil/quaternion/init.luau +570 -0
  167. package/vendor/rbxutil/quaternion/package.json +16 -0
  168. package/vendor/rbxutil/quaternion/wally.toml +9 -0
  169. package/vendor/rbxutil/query/index.d.ts +43 -0
  170. package/vendor/rbxutil/query/init.luau +117 -0
  171. package/vendor/rbxutil/query/package.json +18 -0
  172. package/vendor/rbxutil/query/wally.toml +9 -0
  173. package/vendor/rbxutil/sequent/index.d.ts +28 -0
  174. package/vendor/rbxutil/sequent/init.luau +340 -0
  175. package/vendor/rbxutil/sequent/package.json +16 -0
  176. package/vendor/rbxutil/sequent/wally.toml +9 -0
  177. package/vendor/rbxutil/ser/init.luau +175 -0
  178. package/vendor/rbxutil/ser/init.test.luau +50 -0
  179. package/vendor/rbxutil/ser/wally.toml +11 -0
  180. package/vendor/rbxutil/shake/index.d.ts +36 -0
  181. package/vendor/rbxutil/shake/init.luau +532 -0
  182. package/vendor/rbxutil/shake/init.test.luau +267 -0
  183. package/vendor/rbxutil/shake/package.json +16 -0
  184. package/vendor/rbxutil/shake/wally.toml +9 -0
  185. package/vendor/rbxutil/signal/index.d.ts +100 -0
  186. package/vendor/rbxutil/signal/init.luau +432 -0
  187. package/vendor/rbxutil/signal/init.test.luau +190 -0
  188. package/vendor/rbxutil/signal/package.json +17 -0
  189. package/vendor/rbxutil/signal/wally.toml +9 -0
  190. package/vendor/rbxutil/silo/TableWatcher.luau +65 -0
  191. package/vendor/rbxutil/silo/Util.luau +55 -0
  192. package/vendor/rbxutil/silo/init.luau +338 -0
  193. package/vendor/rbxutil/silo/init.test.luau +215 -0
  194. package/vendor/rbxutil/silo/wally.toml +8 -0
  195. package/vendor/rbxutil/spring/index.d.ts +40 -0
  196. package/vendor/rbxutil/spring/init.luau +97 -0
  197. package/vendor/rbxutil/spring/package.json +17 -0
  198. package/vendor/rbxutil/spring/wally.toml +8 -0
  199. package/vendor/rbxutil/stream/index.d.ts +88 -0
  200. package/vendor/rbxutil/stream/init.luau +597 -0
  201. package/vendor/rbxutil/stream/package.json +18 -0
  202. package/vendor/rbxutil/stream/wally.toml +9 -0
  203. package/vendor/rbxutil/streamable/Streamable.luau +202 -0
  204. package/vendor/rbxutil/streamable/StreamableUtil.luau +80 -0
  205. package/vendor/rbxutil/streamable/init.luau +8 -0
  206. package/vendor/rbxutil/streamable/wally.toml +12 -0
  207. package/vendor/rbxutil/symbol/init.luau +56 -0
  208. package/vendor/rbxutil/symbol/init.test.luau +37 -0
  209. package/vendor/rbxutil/symbol/wally.toml +8 -0
  210. package/vendor/rbxutil/table-util/init.luau +938 -0
  211. package/vendor/rbxutil/table-util/init.test.luau +439 -0
  212. package/vendor/rbxutil/table-util/wally.toml +8 -0
  213. package/vendor/rbxutil/task-queue/index.d.ts +27 -0
  214. package/vendor/rbxutil/task-queue/init.luau +97 -0
  215. package/vendor/rbxutil/task-queue/wally.toml +8 -0
  216. package/vendor/rbxutil/timer/index.d.ts +81 -0
  217. package/vendor/rbxutil/timer/init.luau +249 -0
  218. package/vendor/rbxutil/timer/init.test.luau +73 -0
  219. package/vendor/rbxutil/timer/wally.toml +11 -0
  220. package/vendor/rbxutil/tree/index.d.ts +15 -0
  221. package/vendor/rbxutil/tree/init.luau +137 -0
  222. package/vendor/rbxutil/tree/wally.toml +8 -0
  223. package/vendor/rbxutil/trove/index.d.ts +46 -0
  224. package/vendor/rbxutil/trove/init.luau +787 -0
  225. package/vendor/rbxutil/trove/init.test.luau +203 -0
  226. package/vendor/rbxutil/trove/wally.toml +8 -0
  227. package/vendor/rbxutil/typed-remote/init.luau +196 -0
  228. package/vendor/rbxutil/typed-remote/wally.toml +8 -0
  229. package/vendor/rbxutil/wait-for/index.d.ts +17 -0
  230. package/vendor/rbxutil/wait-for/init.luau +257 -0
  231. package/vendor/rbxutil/wait-for/init.test.luau +182 -0
  232. package/vendor/rbxutil/wait-for/wally.toml +11 -0
  233. package/vendor/t/t.lua +1350 -0
  234. package/vendor/testez/Context.lua +26 -0
  235. package/vendor/testez/Expectation.lua +311 -0
  236. package/vendor/testez/ExpectationContext.lua +38 -0
  237. package/vendor/testez/LifecycleHooks.lua +89 -0
  238. package/vendor/testez/Reporters/TeamCityReporter.lua +102 -0
  239. package/vendor/testez/Reporters/TextReporter.lua +106 -0
  240. package/vendor/testez/Reporters/TextReporterQuiet.lua +97 -0
  241. package/vendor/testez/TestBootstrap.lua +147 -0
  242. package/vendor/testez/TestEnum.lua +28 -0
  243. package/vendor/testez/TestPlan.lua +304 -0
  244. package/vendor/testez/TestPlanner.lua +40 -0
  245. package/vendor/testez/TestResults.lua +112 -0
  246. package/vendor/testez/TestRunner.lua +188 -0
  247. package/vendor/testez/TestSession.lua +243 -0
  248. package/vendor/testez/init.lua +40 -0
@@ -0,0 +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
+
112
+ return TestResults
@@ -0,0 +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
@@ -0,0 +1,243 @@
1
+ --[[
2
+ Represents the state relevant while executing a test plan.
3
+
4
+ Used by TestRunner to produce a TestResults object.
5
+
6
+ Uses the same tree building structure as TestPlanBuilder; TestSession keeps
7
+ track of a stack of nodes that represent the current path through the tree.
8
+ ]]
9
+
10
+ local TestEnum = require(script.Parent.TestEnum)
11
+ local TestResults = require(script.Parent.TestResults)
12
+ local Context = require(script.Parent.Context)
13
+ local ExpectationContext = require(script.Parent.ExpectationContext)
14
+
15
+ local TestSession = {}
16
+
17
+ TestSession.__index = TestSession
18
+
19
+ --[[
20
+ Create a TestSession related to the given TestPlan.
21
+
22
+ The resulting TestResults object will be linked to this TestPlan.
23
+ ]]
24
+ function TestSession.new(plan)
25
+ local self = {
26
+ results = TestResults.new(plan),
27
+ nodeStack = {},
28
+ contextStack = {},
29
+ expectationContextStack = {},
30
+ hasFocusNodes = false
31
+ }
32
+
33
+ setmetatable(self, TestSession)
34
+
35
+ return self
36
+ end
37
+
38
+ --[[
39
+ Calculate success, failure, and skipped test counts in the tree at the
40
+ current point in the execution.
41
+ ]]
42
+ function TestSession:calculateTotals()
43
+ local results = self.results
44
+
45
+ results.successCount = 0
46
+ results.failureCount = 0
47
+ results.skippedCount = 0
48
+
49
+ results:visitAllNodes(function(node)
50
+ local status = node.status
51
+ local nodeType = node.planNode.type
52
+
53
+ if nodeType == TestEnum.NodeType.It then
54
+ if status == TestEnum.TestStatus.Success then
55
+ results.successCount = results.successCount + 1
56
+ elseif status == TestEnum.TestStatus.Failure then
57
+ results.failureCount = results.failureCount + 1
58
+ elseif status == TestEnum.TestStatus.Skipped then
59
+ results.skippedCount = results.skippedCount + 1
60
+ end
61
+ end
62
+ end)
63
+ end
64
+
65
+ --[[
66
+ Gathers all of the errors reported by tests and puts them at the top level
67
+ of the TestResults object.
68
+ ]]
69
+ function TestSession:gatherErrors()
70
+ local results = self.results
71
+
72
+ results.errors = {}
73
+
74
+ results:visitAllNodes(function(node)
75
+ if #node.errors > 0 then
76
+ for _, message in ipairs(node.errors) do
77
+ table.insert(results.errors, message)
78
+ end
79
+ end
80
+ end)
81
+ end
82
+
83
+ --[[
84
+ Calculates test totals, verifies the tree is valid, and returns results.
85
+ ]]
86
+ function TestSession:finalize()
87
+ if #self.nodeStack ~= 0 then
88
+ error("Cannot finalize TestResults with nodes still on the stack!", 2)
89
+ end
90
+
91
+ self:calculateTotals()
92
+ self:gatherErrors()
93
+
94
+ return self.results
95
+ end
96
+
97
+ --[[
98
+ Create a new test result node and push it onto the navigation stack.
99
+ ]]
100
+ function TestSession:pushNode(planNode)
101
+ local node = TestResults.createNode(planNode)
102
+ local lastNode = self.nodeStack[#self.nodeStack] or self.results
103
+ table.insert(lastNode.children, node)
104
+ table.insert(self.nodeStack, node)
105
+
106
+ local lastContext = self.contextStack[#self.contextStack]
107
+ local context = Context.new(lastContext)
108
+ table.insert(self.contextStack, context)
109
+
110
+ local lastExpectationContext = self.expectationContextStack[#self.expectationContextStack]
111
+ local expectationContext = ExpectationContext.new(lastExpectationContext)
112
+ table.insert(self.expectationContextStack, expectationContext)
113
+ end
114
+
115
+ --[[
116
+ Pops a node off of the navigation stack.
117
+ ]]
118
+ function TestSession:popNode()
119
+ assert(#self.nodeStack > 0, "Tried to pop from an empty node stack!")
120
+ table.remove(self.nodeStack, #self.nodeStack)
121
+ table.remove(self.contextStack, #self.contextStack)
122
+ table.remove(self.expectationContextStack, #self.expectationContextStack)
123
+ end
124
+
125
+ --[[
126
+ Gets the Context object for the current node.
127
+ ]]
128
+ function TestSession:getContext()
129
+ assert(#self.contextStack > 0, "Tried to get context from an empty stack!")
130
+ return self.contextStack[#self.contextStack]
131
+ end
132
+
133
+
134
+ function TestSession:getExpectationContext()
135
+ assert(#self.expectationContextStack > 0, "Tried to get expectationContext from an empty stack!")
136
+ return self.expectationContextStack[#self.expectationContextStack]
137
+ end
138
+
139
+ --[[
140
+ Tells whether the current test we're in should be skipped.
141
+ ]]
142
+ function TestSession:shouldSkip()
143
+ -- If our test tree had any exclusive tests, then normal tests are skipped!
144
+ if self.hasFocusNodes then
145
+ for i = #self.nodeStack, 1, -1 do
146
+ local node = self.nodeStack[i]
147
+
148
+ -- Skipped tests are still skipped
149
+ if node.planNode.modifier == TestEnum.NodeModifier.Skip then
150
+ return true
151
+ end
152
+
153
+ -- Focused tests are the only ones that aren't skipped
154
+ if node.planNode.modifier == TestEnum.NodeModifier.Focus then
155
+ return false
156
+ end
157
+ end
158
+
159
+ return true
160
+ else
161
+ for i = #self.nodeStack, 1, -1 do
162
+ local node = self.nodeStack[i]
163
+
164
+ if node.planNode.modifier == TestEnum.NodeModifier.Skip then
165
+ return true
166
+ end
167
+ end
168
+ end
169
+
170
+ return false
171
+ end
172
+
173
+ --[[
174
+ Set the current node's status to Success.
175
+ ]]
176
+ function TestSession:setSuccess()
177
+ assert(#self.nodeStack > 0, "Attempting to set success status on empty stack")
178
+ self.nodeStack[#self.nodeStack].status = TestEnum.TestStatus.Success
179
+ end
180
+
181
+ --[[
182
+ Set the current node's status to Skipped.
183
+ ]]
184
+ function TestSession:setSkipped()
185
+ assert(#self.nodeStack > 0, "Attempting to set skipped status on empty stack")
186
+ self.nodeStack[#self.nodeStack].status = TestEnum.TestStatus.Skipped
187
+ end
188
+
189
+ --[[
190
+ Set the current node's status to Failure and adds a message to its list of
191
+ errors.
192
+ ]]
193
+ function TestSession:setError(message)
194
+ assert(#self.nodeStack > 0, "Attempting to set error status on empty stack")
195
+ local last = self.nodeStack[#self.nodeStack]
196
+ last.status = TestEnum.TestStatus.Failure
197
+ table.insert(last.errors, message)
198
+ end
199
+
200
+ --[[
201
+ Add a dummy child node to the current node to hold the given error. This
202
+ allows an otherwise empty describe node to report an error in a more natural
203
+ way.
204
+ ]]
205
+ function TestSession:addDummyError(phrase, message)
206
+ self:pushNode({type = TestEnum.NodeType.It, phrase = phrase})
207
+ self:setError(message)
208
+ self:popNode()
209
+ self.nodeStack[#self.nodeStack].status = TestEnum.TestStatus.Failure
210
+ end
211
+
212
+ --[[
213
+ Set the current node's status based on that of its children. If all children
214
+ are skipped, mark it as skipped. If any are fails, mark it as failed.
215
+ Otherwise, mark it as success.
216
+ ]]
217
+ function TestSession:setStatusFromChildren()
218
+ assert(#self.nodeStack > 0, "Attempting to set status from children on empty stack")
219
+
220
+ local last = self.nodeStack[#self.nodeStack]
221
+ local status = TestEnum.TestStatus.Success
222
+ local skipped = true
223
+
224
+ -- If all children were skipped, then we were skipped
225
+ -- If any child failed, then we failed!
226
+ for _, child in ipairs(last.children) do
227
+ if child.status ~= TestEnum.TestStatus.Skipped then
228
+ skipped = false
229
+
230
+ if child.status == TestEnum.TestStatus.Failure then
231
+ status = TestEnum.TestStatus.Failure
232
+ end
233
+ end
234
+ end
235
+
236
+ if skipped then
237
+ status = TestEnum.TestStatus.Skipped
238
+ end
239
+
240
+ last.status = status
241
+ end
242
+
243
+ return TestSession
@@ -0,0 +1,40 @@
1
+ local Expectation = require(script.Expectation)
2
+ local TestBootstrap = require(script.TestBootstrap)
3
+ local TestEnum = require(script.TestEnum)
4
+ local TestPlan = require(script.TestPlan)
5
+ local TestPlanner = require(script.TestPlanner)
6
+ local TestResults = require(script.TestResults)
7
+ local TestRunner = require(script.TestRunner)
8
+ local TestSession = require(script.TestSession)
9
+ local TextReporter = require(script.Reporters.TextReporter)
10
+ local TextReporterQuiet = require(script.Reporters.TextReporterQuiet)
11
+ local TeamCityReporter = require(script.Reporters.TeamCityReporter)
12
+
13
+ local function run(testRoot, callback)
14
+ local modules = TestBootstrap:getModules(testRoot)
15
+ local plan = TestPlanner.createPlan(modules)
16
+ local results = TestRunner.runPlan(plan)
17
+
18
+ callback(results)
19
+ end
20
+
21
+ local TestEZ = {
22
+ run = run,
23
+
24
+ Expectation = Expectation,
25
+ TestBootstrap = TestBootstrap,
26
+ TestEnum = TestEnum,
27
+ TestPlan = TestPlan,
28
+ TestPlanner = TestPlanner,
29
+ TestResults = TestResults,
30
+ TestRunner = TestRunner,
31
+ TestSession = TestSession,
32
+
33
+ Reporters = {
34
+ TextReporter = TextReporter,
35
+ TextReporterQuiet = TextReporterQuiet,
36
+ TeamCityReporter = TeamCityReporter,
37
+ },
38
+ }
39
+
40
+ return TestEZ