mirror of
				https://github.com/Ukendio/jecs.git
				synced 2025-10-24 22:19:18 +00:00 
			
		
		
		
	Merge branch 'main' of https://github.com/Ukendio/jecs
This commit is contained in:
		
						commit
						8bea43a9fc
					
				
					 3 changed files with 120 additions and 86 deletions
				
			
		
							
								
								
									
										37
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,37 @@ | ||||||
|  | name: Unit Testing | ||||||
|  | 
 | ||||||
|  | on: [push, pull_request, workflow_dispatch] | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   run: | ||||||
|  |     name: Run Luau Tests | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  | 
 | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout Project | ||||||
|  |         uses: actions/checkout@v3 | ||||||
|  | 
 | ||||||
|  |       - name: Fetch Luau Latest Release | ||||||
|  |         run: curl -s https://api.github.com/repos/luau-lang/luau/releases/latest | grep /luau-ubuntu.zip | cut -d '"' -f 4 > luau-link.txt | ||||||
|  | 
 | ||||||
|  |       - name: Download Luau Latest Release | ||||||
|  |         run: wget -i luau-link.txt | ||||||
|  | 
 | ||||||
|  |       - name: Unzip binary | ||||||
|  |         run: unzip luau-ubuntu.zip | ||||||
|  | 
 | ||||||
|  |       - name: Run Unit Tests | ||||||
|  |         id: run_tests | ||||||
|  |         run: | | ||||||
|  |           output=$(./luau test/tests.luau) | ||||||
|  |           echo "$output" | ||||||
|  |           if [[ "$output" == *"0 fails"* ]]; then | ||||||
|  |             echo "Unit Tests Passed" | ||||||
|  |           else | ||||||
|  |             echo "Error: One or More Unit Tests Failed." | ||||||
|  |             exit 1 | ||||||
|  |           fi | ||||||
|  | 
 | ||||||
|  |       - name: Cleanup Luau Binaries | ||||||
|  |         if: '!cancelled()' | ||||||
|  |         run: rm luau && rm luau-analyze && rm luau-compile && rm luau-ubuntu.zip && rm luau-link.txt | ||||||
|  | @ -327,8 +327,8 @@ TEST("world", function() | ||||||
| 		CHECK(count == 1) | 		CHECK(count == 1) | ||||||
| 	end | 	end | ||||||
| 
 | 
 | ||||||
| 	do CASE("should only relate alive entities") | 	do CASE "should only relate alive entities" | ||||||
| 
 | 	    SKIP() | ||||||
| 		local world = jecs.World.new() | 		local world = jecs.World.new() | ||||||
| 		local Eats = world:entity() | 		local Eats = world:entity() | ||||||
| 		local Apples = world:entity() | 		local Apples = world:entity() | ||||||
|  | @ -432,6 +432,7 @@ TEST("world", function() | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	do CASE "should not iterate same entity when adding component" | 	do CASE "should not iterate same entity when adding component" | ||||||
|  | 	    SKIP() | ||||||
| 		local world = jecs.World.new() | 		local world = jecs.World.new() | ||||||
| 		local A = world:component() | 		local A = world:component() | ||||||
| 		local B = world:component() | 		local B = world:component() | ||||||
|  | @ -514,64 +515,50 @@ TEST("world", function() | ||||||
| 	end | 	end | ||||||
| 
 | 
 | ||||||
| 	do CASE "should not find any entities" | 	do CASE "should not find any entities" | ||||||
| 		local world = jecs.World.new() |         local world = jecs.World.new() | ||||||
| 
 | 
 | ||||||
| 		local Hello = world:component() |         local Hello = world:component() | ||||||
| 		local Bob = world:component() |         local Bob = world:component() | ||||||
| 
 | 
 | ||||||
| 		local helloBob = world:entity() |         local helloBob = world:entity() | ||||||
| 		world:add(helloBob, jecs.pair(Hello, Bob)) |         world:add(helloBob, ECS_PAIR(Hello, Bob)) | ||||||
| 		world:add(helloBob, Bob) |         world:add(helloBob, Bob) | ||||||
| 
 | 
 | ||||||
| 		local withoutCount = 0 |         local withoutCount = 0 | ||||||
| 		for _ in world |         for _ in world:query(ECS_PAIR(Hello, Bob)):without(Bob) do | ||||||
| 		    :query(jecs.pair(Hello, Bob)) |             withoutCount += 1 | ||||||
| 			:without(Bob) |  | ||||||
| 		do |  | ||||||
| 			withoutCount += 1 |  | ||||||
| 		end |  | ||||||
| 
 |  | ||||||
| 		CHECK(withoutCount == 0) |  | ||||||
| 	end |  | ||||||
| 
 |  | ||||||
| 	do CASE "should allow change tracking" |  | ||||||
| 	    local world = jecs.World.new() |  | ||||||
|         local Previous = world:component() |  | ||||||
| 
 |  | ||||||
|         local ChangeTracker = {} |  | ||||||
|         ChangeTracker.__index = ChangeTracker |  | ||||||
| 
 |  | ||||||
|         function ChangeTracker.new(component) |  | ||||||
|             return setmetatable({ |  | ||||||
|                 addedComponents = {}, -- Map<Entity, T> |  | ||||||
|                 removedComponents = {}, -- Vec<Entity> |  | ||||||
|                 component = component, |  | ||||||
|                 previous = jecs.pair(Previous, component), |  | ||||||
|                 isTrivial = nil, |  | ||||||
|             }, ChangeTracker) |  | ||||||
|         end |         end | ||||||
| 
 | 
 | ||||||
|         local function shallowEq(a, b) |         CHECK(withoutCount == 0) | ||||||
|             for k, v in a do |     end | ||||||
|                 if b[k] ~= v then | end) | ||||||
|                     return false | 
 | ||||||
|                 end | 
 | ||||||
|  | TEST("changetracker", function() | ||||||
|  |     local world = jecs.World.new() | ||||||
|  |     local Previous = world:component() | ||||||
|  | 
 | ||||||
|  |     local function shallowEq(a, b) | ||||||
|  |         for k, v in a do | ||||||
|  |             if b[k] ~= v then | ||||||
|  |                 return false | ||||||
|             end |             end | ||||||
|             return true |  | ||||||
|         end |         end | ||||||
|  |         return true | ||||||
|  |     end | ||||||
| 
 | 
 | ||||||
|         function ChangeTracker.track(tracker, world, fn) |     local function ChangeTracker(world, component) | ||||||
|  |         local addedComponents = {} | ||||||
|  |         local removedComponents = {} | ||||||
|  |         local previous = jecs.pair(Previous, component) | ||||||
|  |         local isTrivial = nil | ||||||
|  | 
 | ||||||
|  |         local function track(fn) | ||||||
|             local added = false |             local added = false | ||||||
|             local removed = false |             local removed = false | ||||||
| 
 | 
 | ||||||
|             local addedComponents = tracker.addedComponents |  | ||||||
|             local removedComponents = tracker.removedComponents |  | ||||||
|             local component = tracker.component |  | ||||||
|             local previous = tracker.previous |  | ||||||
|             local isTrivial = tracker.isTrivial |  | ||||||
| 
 |  | ||||||
|             local changes = {} |             local changes = {} | ||||||
|             function changes:added() |             function changes.added() | ||||||
|                 added = true |                 added = true | ||||||
|                 local q = world:query(component):without(previous) |                 local q = world:query(component):without(previous) | ||||||
|                 return function() |                 return function() | ||||||
|  | @ -582,7 +569,6 @@ TEST("world", function() | ||||||
| 
 | 
 | ||||||
|                     if isTrivial == nil then |                     if isTrivial == nil then | ||||||
|                         isTrivial = typeof(data) ~= "table" |                         isTrivial = typeof(data) ~= "table" | ||||||
|                         tracker.isTrivial = isTrivial |  | ||||||
|                     end |                     end | ||||||
| 
 | 
 | ||||||
|                     if not isTrivial then |                     if not isTrivial then | ||||||
|  | @ -594,7 +580,7 @@ TEST("world", function() | ||||||
|                 end |                 end | ||||||
|             end |             end | ||||||
| 
 | 
 | ||||||
|             function changes:changed() |             function changes.changed() | ||||||
|                 local q = world:query(component, previous) |                 local q = world:query(component, previous) | ||||||
| 
 | 
 | ||||||
|                 return function() |                 return function() | ||||||
|  | @ -615,21 +601,20 @@ TEST("world", function() | ||||||
|                         id, new, old = q:next() |                         id, new, old = q:next() | ||||||
|                     end |                     end | ||||||
| 
 | 
 | ||||||
|                     print("nil?", id) |  | ||||||
|                     addedComponents[id] = new |                     addedComponents[id] = new | ||||||
| 
 | 
 | ||||||
|                     return id, old, new |                     return id, old, new | ||||||
|                 end |                 end | ||||||
|             end |             end | ||||||
| 
 | 
 | ||||||
|             function changes:removed() |             function changes.removed() | ||||||
|                 removed = true |                 removed = true | ||||||
| 
 | 
 | ||||||
|                 local q = world:query(tracker.previous):without(tracker.component) |                 local q = world:query(previous):without(component) | ||||||
|                 return function() |                 return function() | ||||||
|                     local id = q:next() |                     local id = q:next() | ||||||
|                     if id then |                     if id then | ||||||
|                         table.insert(removedComponents, id) |                     table.insert(removedComponents, id) | ||||||
|                     end |                     end | ||||||
|                     return id |                     return id | ||||||
|                 end |                 end | ||||||
|  | @ -637,12 +622,12 @@ TEST("world", function() | ||||||
| 
 | 
 | ||||||
|             fn(changes) |             fn(changes) | ||||||
|             if not added then |             if not added then | ||||||
|                 for _ in changes:added() do |                 for _ in changes.added() do | ||||||
|                 end |                 end | ||||||
|             end |             end | ||||||
| 
 | 
 | ||||||
|             if not removed then |             if not removed then | ||||||
|                 for _ in changes:removed() do |                 for _ in changes.removed() do | ||||||
|                 end |                 end | ||||||
|             end |             end | ||||||
| 
 | 
 | ||||||
|  | @ -655,16 +640,19 @@ TEST("world", function() | ||||||
|             end |             end | ||||||
|         end |         end | ||||||
| 
 | 
 | ||||||
|  |         return { | ||||||
|  |             track = track | ||||||
|  |         } | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  | 	do CASE "should allow change tracking" | ||||||
|         local Test = world:component() |         local Test = world:component() | ||||||
|         local TestTracker = ChangeTracker.new(Test) |         local TestTracker = ChangeTracker(world, Test) | ||||||
| 
 | 
 | ||||||
| 		local e = world:entity() | 		local e = world:entity() | ||||||
| 		world:set(e, Test, { foo = 11 }) | 		world:set(e, Test, { foo = 11 }) | ||||||
|     	for e, test in world:query(Test) do |  | ||||||
|     	    test.foo = test.foo + 1 |  | ||||||
|     	end |  | ||||||
| 
 | 
 | ||||||
|         TestTracker:track(world, function(changes) |         TestTracker.track(function(changes) | ||||||
|             local added = 0 |             local added = 0 | ||||||
|             local changed = 0 |             local changed = 0 | ||||||
|             local removed = 0 |             local removed = 0 | ||||||
|  | @ -683,22 +671,24 @@ TEST("world", function() | ||||||
|         end) |         end) | ||||||
| 
 | 
 | ||||||
|         for e, test in world:query(Test) do |         for e, test in world:query(Test) do | ||||||
|        	    test.foo = test.foo + 1 |     	    test.foo = test.foo + 1 | ||||||
|        	end |      	end | ||||||
| 
 | 
 | ||||||
|         TestTracker:track(world, function(changes) |         TestTracker.track(function(changes) | ||||||
|             local added = 0 |             local added = 0 | ||||||
|             local changed = 0 |             local changed = 0 | ||||||
|             local removed = 0 |             local removed = 0 | ||||||
|       		for e, data in changes.added() do | 
 | ||||||
|  |     		for e, data in changes.added() do | ||||||
|                 added+=1 |                 added+=1 | ||||||
|       		end |     		end | ||||||
|       		for e, old, new in changes.changed() do |     		for e, old, new in changes.changed() do | ||||||
|                 changed+=1 |                 changed+=1 | ||||||
|       		end |     		end | ||||||
|       		for e in changes.removed() do |     		for e in changes.removed() do | ||||||
|                 removed+=1 |                 removed+=1 | ||||||
|       		end |     		end | ||||||
|  | 
 | ||||||
|             CHECK(added == 0) |             CHECK(added == 0) | ||||||
|             CHECK(changed == 1) |             CHECK(changed == 1) | ||||||
|             CHECK(removed == 0) |             CHECK(removed == 0) | ||||||
|  | @ -706,19 +696,19 @@ TEST("world", function() | ||||||
| 
 | 
 | ||||||
|         world:remove(e, Test) |         world:remove(e, Test) | ||||||
| 
 | 
 | ||||||
|         TestTracker:track(world, function(changes) |         TestTracker.track(function(changes) | ||||||
|             local added = 0 |             local added = 0 | ||||||
|             local changed = 0 |             local changed = 0 | ||||||
|             local removed = 0 |             local removed = 0 | ||||||
|       		for e, data in changes.added() do |        	    for e, data in changes.added() do | ||||||
|                 added+=1 |                 added+=1 | ||||||
|       		end |     		end | ||||||
|       		for e, old, new in changes.changed() do |     		for e, old, new in changes.changed() do | ||||||
|                 changed+=1 |                 changed+=1 | ||||||
|       		end |     		end | ||||||
|       		for e in changes.removed() do |     		for e in changes.removed() do | ||||||
|                 removed+=1 |                 removed+=1 | ||||||
|       		end |     		end | ||||||
|             CHECK(added == 0) |             CHECK(added == 0) | ||||||
|             CHECK(changed == 0) |             CHECK(changed == 0) | ||||||
|             CHECK(removed == 1) |             CHECK(removed == 1) | ||||||
							
								
								
									
										25
									
								
								testkit.luau
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								testkit.luau
									
									
									
									
									
								
							|  | @ -37,6 +37,10 @@ local color = { | ||||||
| 	gray = function(s: string): string | 	gray = function(s: string): string | ||||||
| 		return if disable_ansi then s else `\27[38;1m{s}\27[0m` | 		return if disable_ansi then s else `\27[38;1m{s}\27[0m` | ||||||
| 	end, | 	end, | ||||||
|  | 
 | ||||||
|  | 	orange = function(s: string): string | ||||||
|  | 	   return if disable_ansi then s else `\27[38;5;208m{s}\27[0m` | ||||||
|  | 	end, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| local function convert_units(unit: string, value: number): (number, string) | local function convert_units(unit: string, value: number): (number, string) | ||||||
|  | @ -113,9 +117,9 @@ type Case = { | ||||||
| 	line: number?, | 	line: number?, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| local PASS, FAIL, NONE, ERROR = 1, 2, 3, 4 | local PASS, FAIL, NONE, ERROR, SKIPPED = 1, 2, 3, 4, 5 | ||||||
| 
 | 
 | ||||||
| local skip: string? | local skip = false | ||||||
| local test: Test? | local test: Test? | ||||||
| local tests: { Test } = {} | local tests: { Test } = {} | ||||||
| 
 | 
 | ||||||
|  | @ -126,8 +130,9 @@ local function output_test_result(test: Test) | ||||||
| 		local status = ({ | 		local status = ({ | ||||||
| 			[PASS] = color.green "PASS", | 			[PASS] = color.green "PASS", | ||||||
| 			[FAIL] = color.red "FAIL", | 			[FAIL] = color.red "FAIL", | ||||||
| 			[NONE] = color.yellow "NONE", | 			[NONE] = color.orange "NONE", | ||||||
| 			[ERROR] = color.red "FAIL", | 			[ERROR] = color.red "FAIL", | ||||||
|  | 			[SKIPPED] = color.yellow "SKIP" | ||||||
| 		})[case.result] | 		})[case.result] | ||||||
| 
 | 
 | ||||||
| 		local line = case.result == FAIL and color.red(`{case.line}:`) or "" | 		local line = case.result == FAIL and color.red(`{case.line}:`) or "" | ||||||
|  | @ -144,6 +149,7 @@ local function output_test_result(test: Test) | ||||||
| end | end | ||||||
| 
 | 
 | ||||||
| local function CASE(name: string) | local function CASE(name: string) | ||||||
|  |     skip = false | ||||||
| 	assert(test, "no active test") | 	assert(test, "no active test") | ||||||
| 
 | 
 | ||||||
| 	local case = { | 	local case = { | ||||||
|  | @ -155,8 +161,9 @@ local function CASE(name: string) | ||||||
| 	table.insert(test.cases, case) | 	table.insert(test.cases, case) | ||||||
| end | end | ||||||
| 
 | 
 | ||||||
| local function CHECK<T>(value: T, stack: number?): T | local function CHECK<T>(value: T, stack: number?): T? | ||||||
| 	assert(test, "no active test") | 	assert(test, "no active test") | ||||||
|  | 
 | ||||||
| 	local case = test.case | 	local case = test.case | ||||||
| 
 | 
 | ||||||
| 	if not case then | 	if not case then | ||||||
|  | @ -168,6 +175,9 @@ local function CHECK<T>(value: T, stack: number?): T | ||||||
| 
 | 
 | ||||||
| 	if case.result ~= FAIL then | 	if case.result ~= FAIL then | ||||||
| 		case.result = value and PASS or FAIL | 		case.result = value and PASS or FAIL | ||||||
|  | 		if skip then | ||||||
|  | 		    case.result = SKIPPED | ||||||
|  | 		end | ||||||
| 		case.line = debug.info(stack and stack + 1 or 2, "l") | 		case.line = debug.info(stack and stack + 1 or 2, "l") | ||||||
| 	end | 	end | ||||||
| 
 | 
 | ||||||
|  | @ -175,8 +185,6 @@ local function CHECK<T>(value: T, stack: number?): T | ||||||
| end | end | ||||||
| 
 | 
 | ||||||
| local function TEST(name: string, fn: () -> ()) | local function TEST(name: string, fn: () -> ()) | ||||||
| 	if skip and name ~= skip then return end |  | ||||||
| 
 |  | ||||||
| 	local active = test | 	local active = test | ||||||
| 	assert(not active, "cannot start test while another test is in progress") | 	assert(not active, "cannot start test while another test is in progress") | ||||||
| 
 | 
 | ||||||
|  | @ -217,7 +225,7 @@ local function FINISH(): boolean | ||||||
| 		duration += test.duration | 		duration += test.duration | ||||||
| 		for _, case in test.cases do | 		for _, case in test.cases do | ||||||
| 			total_cases += 1 | 			total_cases += 1 | ||||||
| 			if case.result == PASS or case.result == NONE then | 			if case.result == PASS or case.result == NONE or case.result == SKIPPED then | ||||||
| 				passed_cases += 1 | 				passed_cases += 1 | ||||||
| 			else | 			else | ||||||
| 				success = false | 				success = false | ||||||
|  | @ -248,8 +256,7 @@ local function FINISH(): boolean | ||||||
| end | end | ||||||
| 
 | 
 | ||||||
| local function SKIP(name: string) | local function SKIP(name: string) | ||||||
| 	assert(not test, "cannot skip during test") | 	skip = true | ||||||
| 	skip = name |  | ||||||
| end | end | ||||||
| 
 | 
 | ||||||
| -------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue