diff --git a/.forgejo/workflows/release.yml b/.forgejo/workflows/release.yml index 0e33a44..abf57e5 100644 --- a/.forgejo/workflows/release.yml +++ b/.forgejo/workflows/release.yml @@ -6,7 +6,7 @@ on: - cron: "10 0 * * *" # Runs at 00:10 UTC every day jobs: - sync: + release: name: Sync runs-on: docker container: @@ -20,7 +20,13 @@ jobs: with: token: ${{ secrets.githubtoken }} - - name: Install fj + - name: Install Luau + uses: https://github.com/EncodedVenom/install-luau@v4 + with: + version: "latest" + verbose: "true" + + - name: Install forgejo-cli run: | wget https://codeberg.org/Cyborus/forgejo-cli/releases/download/v0.2.0/forgejo-cli-linux.gz -O fj.gz gunzip fj.gz @@ -36,8 +42,19 @@ jobs: wally login --token "${{ secrets.wally_auth_token }}" rm wally.toml - - name: Sync & Release - run: lune run src/init + - name: Synchronize + run: lune run src/init -- sync jecs + + - name: Run Unit Tests + run: | + output=$(lune run src/init -- test jecs) + echo "$output" + if [[ "$output" == *"0 fails"* ]]; then + echo "Unit Tests Passed" + else + echo "Error: One or More Unit Tests Failed" + exit 1 + fi - name: Read Jecs Version id: read_jecs_version @@ -45,6 +62,22 @@ jobs: version=$(lune run src/read_version | tr '\n' ' ') echo "JECS_VERSION=$version" >> $GITHUB_OUTPUT + - name: Build With Rojo + run: | + cd jecs + rojo build default.project.json -o build.rbxm + mv build.rbxm ../jecs_nightly.rbxm + cd .. + + - name: Upload Build Artifact + uses: https://git.devmarked.win/actions/upload-artifact@v4 + with: + name: build-${{ steps.read_jecs_version.outputs.JECS_VERSION }} + path: jecs_nightly.rbxm + + - name: Release + run: lune run src/init -- release jecs --pesde-scope marked/jecs_nightly --wally-scope mark-marks/jecs-nightly + - name: Create Pull Request id: create_pull_request uses: https://git.devmarked.win/actions/create-pull-request@7174d368c2e4450dea17b297819eb28ae93ee645 diff --git a/jecs/.luaurc b/jecs/.luaurc deleted file mode 100644 index 07221f7..0000000 --- a/jecs/.luaurc +++ /dev/null @@ -1,8 +0,0 @@ -{ - "aliases": { - "jecs": "jecs", - "testkit": "test/testkit", - "mirror": "mirror" - }, - "languageMode": "strict" -} diff --git a/jecs/CHANGELOG.md b/jecs/CHANGELOG.md deleted file mode 100644 index 52f87cc..0000000 --- a/jecs/CHANGELOG.md +++ /dev/null @@ -1,205 +0,0 @@ -# Jecs Changelog - -All notable changes to this project will be documented in this file. - -The format is based on [Keep a Changelog][kac], and this project adheres to -[Semantic Versioning][semver]. - -[kac]: https://keepachangelog.com/en/1.1.0/ -[semver]: https://semver.org/spec/v2.0.0.html - -## [Unreleased] - -- `[world]`: - - 16% faster `world:get` - - `world:has` no longer typechecks components after the 8th one. -- `[typescript]` - - - Fixed Entity type to default to `undefined | unknown` instead of just `undefined` - -- `[query]`: - - Fixed bug where `world:clear` did not invoke `jecs.OnRemove` hooks - - Changed `query.__iter` to drain on iteration - - It will initialize once wherever you left iteration off at last time - - Changed `query:iter` to restart the iterator - - Removed `query:drain` and `query:next` - - If you want to get individual results outside of a for-loop, you need to call `query:iter` to initialize the iterator and then call the iterator function manually - ```lua - local it = world:query(A, B, C):iter() - local entity, a, b, c = it() - entity, a, b, c = it() -- get next results - ``` -- `[world` - - Fixed a bug with `world:clear` not invoking `jecs.OnRemove` hooks -- `[typescript]`: - - Changed pair to accept generics - - Improved handling of Tags - -## [0.3.2] - 2024-10-01 - -- `[world]`: - - Changed `world:cleanup` to traverse a header type for graph edges. (Edit) - - Fixed a regression that occurred when you call `world:set` following a `world:remove` using the same component - - Remove explicit error in JECS_DEBUG for `world:target` when not applying an index parameter -- `[typescript]` : - - Fixed `world.set` with NoInfer - -## [0.3.1] - 2024-10-01 - -- `[world]`: - - Added an index parameter to `world:target` - - Added a way to change the components limit via `_G.JECS_HI_COMPONENT_ID` - - Set it to whatever number you want but try to make it as close to the number of components you will use as possible - - Make sure to set this before importing jecs or else it will not work - - Added debug mode, enable via setting `_G.JECS_DEBUG` to true - - Make sure to set this before importing jecs or else it will not work - - Added `world:cleanup` which is called to cleanup empty archetypes manually - - Changed `world:delete` to delete archetypes that are dependent on the passed entity - - Changed `world:delete` to delete entity's children before the entity to prevent cycles -- `[query]`: - - Fixed the iterator to not drain by default -- `[typescript]` - - Fixed entry point of the package.json file to be `src` rather than `src/init` - - Fixed `query.next` returning a query object whereas it would be expected to return a tuple containing the entity and the corresponding component values - - Exported `query.archetypes` - - Changed `pair` to return a number instead of an entity - - Preventing direct usage of a pair as an entity while still allowing it to be used as a component - - Exported built-in components `ChildOf` and `Name` - - Exported `world.parent` - -## [0.2.10] - 2024-09-07 - -- `[world]`: - - Improved performance for hooks - - Changed `world:set` to be idempotent when setting tags -- `[traits]`: - - Added cleanup condition `jecs.OnDelete` for when the entity or component is deleted - - Added cleanup action `jecs.Remove` which removes instances of the specified (component) id from all entities - - This is the default cleanup action - - Added component trait `jecs.Tag` which allows for zero-cost components used as tags - - Setting data to a component with this trait will do nothing -- `[luau]`: - - Exported `world:contains()` - - Exported `query:drain()` - - Exported `Query` - - Improved types for the hook `OnAdd`, `OnSet`, `OnRemove` - - Changed functions to accept any ID including pairs in type parameters - - Applies to `world:add()`, `world:set()`, `world:remove()`, `world:get()`, `world:has()` and `world:query()` - - New exported type `Id = Entity | Pair` - - Changed `world:contains()` to return a `boolean` instead of an entity which may or may not exist - - Fixed `world:has()` to take the correct parameters - -## [0.2.2] - 2024-07-07 - -### Added - -- Added `query:replace(function(...T) return ...U end)` for replacing components in place - - Method is fast pathed to replace the data to the components for each corresponding entity - -### Changed - -- Iterator now goes backwards instead to prevent common cases of iterator invalidation - -## [0.2.1] - 2024-07-06 - -### Added - -- Added `jecs.Component` built-in component which will be added to ids created with `world:component()`. - - Used to find every component id with `query(jecs.Component) - -## [0.2.0] - 2024-07-03 - -### Added - -- Added `world:parent(entity)` and `jecs.ChildOf` respectively as first class citizen for building parent-child relationships. - - Give a parent to an entity with `world:add($source, pair(ChildOf, $target))` - - Use `world:parent(entity)` to find the target of the relationship -- Added user-facing Luau types - -### Changed - -- Improved iteration speeds 20-40% by manually indexing rather than using `next()` :scream: - -## [0.1.1] - 2024-05-19 - -### Added - -- Added `world:clear(entity)` for removing the components to the corresponding entity -- Added Typescript Types - -## [0.1.0] - 2024-05-13 - -### Changed - -- Optimized iterator - -## [0.1.0-rc.6] - 2024-05-13 - -### Added - -- Added a `jecs.Wildcard` term - - it lets you query any partially matched pairs - -## [0.1.0-rc.5] - 2024-05-10 - -### Added - -- Added Entity relationships for creating logical connections between entities -- Added `world:__iter method` which allows for iteration over the whole world to get every entity - - used for reconciling whole worlds such as via replication, saving/loading, etc -- Added `world:add(entity, component)` which adds a component to the entity - - it is an idempotent function, so calling it twice and in any order should be fine - -### Fixed - -- Fixed component overriding when in disorder - - Previously setting the components in different order results in it overriding component data because it incorrectly mapped the index of the column. So it took the index from the source archetype rather than the destination archetype - -## [0.0.0-prototype.rc.3] - 2024-05-01 - -### Added - -- Added observers -- Added an arm to query `query:without()` for chaining invariants. - -### Changed - -- Separates ranges for components and entity IDs. - - - IDs created with `world:component()` will promote array lookups rather than map lookups in the `component_index` which is a significant boost - -- No longer caches the column pointers directly and instead the column indices which stay persistent even when data is reallocated during swap-removals - - This was an issue with the iterator being invalidated when you move an entity to a different archetype. - -### Fixedhttps://github.com/Ukendio/jecs/releases/tag/v0.0.0-prototype.rc.3 - -- Fixed a bug where changing an existing component would be slow because it was always appending changing the row of the entity record - - The fix dramatically improves times where it is basically down to just the speed of setting a field in a table - -## [0.0.0-prototype.rc.2] - 2024-04-26 - -### Changed - -- Optimized the creation of the query - - It will now finds the smallest archetype map to iterate over -- Optimized the query iterator - - - It will now populates iterator with columns for faster indexing - -- Renamed the insertion method from world:add to world:set to better reflect what it does. - -## [0.0.0-prototype.rc.2] - 2024-04-23 - -- Initial release - -[unreleased]: https://github.com/ukendio/jecs/compare/v0.0.0.0-prototype.rc.2...HEAD -[0.2.2]: https://github.com/ukendio/jecs/releases/tag/v0.2.2 -[0.2.1]: https://github.com/ukendio/jecs/releases/tag/v0.2.1 -[0.2.0]: https://github.com/ukendio/jecs/releases/tag/v0.2.0 -[0.1.1]: https://github.com/ukendio/jecs/releases/tag/v0.1.1 -[0.1.0]: https://github.com/ukendio/jecs/releases/tag/v0.1.0 -[0.1.0-rc.6]: https://github.com/ukendio/jecs/releases/tag/v0.1.0-rc.6 -[0.1.0-rc.5]: https://github.com/ukendio/jecs/releases/tag/v0.1.0-rc.5 -[0.0.0-prototype-rc.3]: https://github.com/ukendio/jecs/releases/tag/v0.0.0-prototype.rc.3 -[0.0.0-prototype.rc.2]: https://github.com/ukendio/jecs/releases/tag/v0.0.0-prototype.rc.2 -[0.0.0-prototype-rc.1]: https://github.com/ukendio/jecs/releases/tag/v0.0.0-prototype.rc.1 diff --git a/jecs/LICENSE b/jecs/LICENSE deleted file mode 100644 index 605eef8..0000000 --- a/jecs/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -MIT License - -Copyright (c) 2024 jecs authors - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. diff --git a/jecs/README.md b/jecs/README.md deleted file mode 100644 index d44797c..0000000 --- a/jecs/README.md +++ /dev/null @@ -1,64 +0,0 @@ -

- -

- -[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg?style=for-the-badge)](LICENSE) [![Wally](https://img.shields.io/github/v/tag/ukendio/jecs?&style=for-the-badge)](https://wally.run/package/ukendio/jecs) - -Just a stupidly fast Entity Component System - -- [Entity Relationships](https://ajmmertens.medium.com/building-games-in-ecs-with-entity-relationships-657275ba2c6c) as first class citizens -- Iterate 800,000 entities at 60 frames per second -- Type-safe [Luau](https://luau-lang.org/) API -- Zero-dependency package -- Optimized for column-major operations -- Cache friendly [archetype/SoA](https://ajmmertens.medium.com/building-an-ecs-2-archetypes-and-vectorization-fe21690805f9) storage -- Rigorously [unit tested](https://github.com/Ukendio/jecs/actions/workflows/ci.yaml) for stability - -### Example - -```lua -local world = jecs.World.new() -local pair = jecs.pair - --- These components and functions are actually already builtin --- but have been illustrated for demonstration purposes -local ChildOf = world:component() -local Name = world:component() - -local function parent(entity) - return world:target(entity, ChildOf) -end -local function getName(entity) - return world:get(entity, Name) -end - -local alice = world:entity() -world:set(alice, Name, "alice") - -local bob = world:entity() -world:add(bob, pair(ChildOf, alice)) -world:set(bob, Name, "bob") - -local sara = world:entity() -world:add(sara, pair(ChildOf, alice)) -world:set(sara, Name, "sara") - -print(getName(parent(sara))) - -for e in world:query(pair(ChildOf, alice)) do - print(getName(e), "is the child of alice") -end - --- Output --- "alice" --- bob is the child of alice --- sara is the child of alice -``` - -21,000 entities 125 archetypes 4 random components queried. -![Queries](assets/image-3.png) -Can be found under /benches/visual/query.luau - -Inserting 8 components to an entity and updating them over 50 times. -![Insertions](assets/image-4.png) -Can be found under /benches/visual/insertions.luau diff --git a/jecs/default.project.json b/jecs/default.project.json deleted file mode 100644 index d4531a0..0000000 --- a/jecs/default.project.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "name": "jecs", - "tree": { - "$path": "jecs.luau" - } -} diff --git a/jecs/jecs.luau b/jecs/jecs.luau deleted file mode 100644 index e53d366..0000000 --- a/jecs/jecs.luau +++ /dev/null @@ -1,2564 +0,0 @@ ---!optimize 2 ---!native ---!strict ---draft 4 - -type i53 = number -type i24 = number - -type Ty = { i53 } -type ArchetypeId = number - -type Column = { any } - -type Map = { [K]: V } - -type GraphEdge = { - from: Archetype, - to: Archetype?, - id: number, - prev: GraphEdge?, - next: GraphEdge?, -} - -type GraphEdges = Map - -type GraphNode = { - add: GraphEdges, - remove: GraphEdges, - refs: GraphEdge, -} - -export type Archetype = { - id: number, - types: Ty, - type: string, - entities: { number }, - columns: { Column }, - records: { number }, - counts: { number }, -} & GraphNode - -export type Record = { - archetype: Archetype, - row: number, - dense: i24, -} - -type IdRecord = { - columns: { number }, - counts: { number }, - flags: number, - size: number, - hooks: { - on_add: ((entity: i53) -> ())?, - on_set: ((entity: i53, data: any) -> ())?, - on_remove: ((entity: i53) -> ())?, - }, -} - -type ComponentIndex = Map - -type Archetypes = { [ArchetypeId]: Archetype } - -type ArchetypeDiff = { - added: Ty, - removed: Ty, -} - -type EntityIndex = { - dense_array: Map, - sparse_array: Map, - alive_count: number, - max_id: number, -} - -local HI_COMPONENT_ID = _G.__JECS_HI_COMPONENT_ID or 256 --- stylua: ignore start -local EcsOnAdd = HI_COMPONENT_ID + 1 -local EcsOnRemove = HI_COMPONENT_ID + 2 -local EcsOnSet = HI_COMPONENT_ID + 3 -local EcsWildcard = HI_COMPONENT_ID + 4 -local EcsChildOf = HI_COMPONENT_ID + 5 -local EcsComponent = HI_COMPONENT_ID + 6 -local EcsOnDelete = HI_COMPONENT_ID + 7 -local EcsOnDeleteTarget = HI_COMPONENT_ID + 8 -local EcsDelete = HI_COMPONENT_ID + 9 -local EcsRemove = HI_COMPONENT_ID + 10 -local EcsName = HI_COMPONENT_ID + 11 -local EcsOnArchetypeCreate = HI_COMPONENT_ID + 12 -local EcsOnArchetypeDelete = HI_COMPONENT_ID + 13 -local EcsRest = HI_COMPONENT_ID + 14 - -local ECS_PAIR_FLAG = 0x8 -local ECS_ID_FLAGS_MASK = 0x10 -local ECS_ENTITY_MASK = bit32.lshift(1, 24) -local ECS_GENERATION_MASK = bit32.lshift(1, 16) - -local ECS_ID_DELETE = 0b0000_0001 -local ECS_ID_IS_TAG = 0b0000_0010 -local ECS_ID_HAS_ON_ADD = 0b0000_0100 -local ECS_ID_HAS_ON_SET = 0b0000_1000 -local ECS_ID_HAS_ON_REMOVE = 0b0001_0000 -local ECS_ID_MASK = 0b0000_0000 --- stylua: ignore end -local NULL_ARRAY = table.freeze({}) :: Column - -local function FLAGS_ADD(is_pair: boolean): number - local flags = 0x0 - - if is_pair then - flags = bit32.bor(flags, ECS_PAIR_FLAG) -- HIGHEST bit in the ID. - end - if false then - flags = bit32.bor(flags, 0x4) -- Set the second flag to true - end - if false then - flags = bit32.bor(flags, 0x2) -- Set the third flag to true - end - if false then - flags = bit32.bor(flags, 0x1) -- LAST BIT in the ID. - end - - return flags -end - -local function ECS_COMBINE(source: number, target: number): i53 - return (source * 268435456) + (target * ECS_ID_FLAGS_MASK) -end - -local function ECS_IS_PAIR(e: number): boolean - return if e > ECS_ENTITY_MASK then (e % ECS_ID_FLAGS_MASK) // ECS_PAIR_FLAG ~= 0 else false -end - --- HIGH 24 bits LOW 24 bits -local function ECS_GENERATION(e: i53): i24 - return if e > ECS_ENTITY_MASK then (e // ECS_ID_FLAGS_MASK) % ECS_GENERATION_MASK else 0 -end - -local function ECS_GENERATION_INC(e: i53) - if e > ECS_ENTITY_MASK then - local flags = e // ECS_ID_FLAGS_MASK - local id = flags // ECS_ENTITY_MASK - local generation = flags % ECS_GENERATION_MASK - - local next_gen = generation + 1 - if next_gen > ECS_GENERATION_MASK then - return id - end - - return ECS_COMBINE(id, next_gen) - end - return ECS_COMBINE(e, 1) -end - --- FIRST gets the high ID -local function ECS_ENTITY_T_HI(e: i53): i24 - return if e > ECS_ENTITY_MASK then (e // ECS_ID_FLAGS_MASK) % ECS_ENTITY_MASK else e -end - --- SECOND -local function ECS_ENTITY_T_LO(e: i53): i24 - return if e > ECS_ENTITY_MASK then (e // ECS_ID_FLAGS_MASK) // ECS_ENTITY_MASK else e -end - -local function _STRIP_GENERATION(e: i53): i24 - return ECS_ENTITY_T_LO(e) -end - -local function ECS_PAIR(pred: i53, obj: i53): i53 - return ECS_COMBINE(ECS_ENTITY_T_LO(pred), ECS_ENTITY_T_LO(obj)) + FLAGS_ADD(--[[isPair]] true) :: i53 -end - -local function entity_index_try_get_any(entity_index: EntityIndex, entity: number): Record? - local r = entity_index.sparse_array[ECS_ENTITY_T_LO(entity)] - if not r then - return nil - end - - if not r or r.dense == 0 then - return nil - end - - return r -end - -local function entity_index_try_get(entity_index: EntityIndex, entity: number): Record? - local r = entity_index_try_get_any(entity_index, entity) - if r then - local r_dense = r.dense - if r_dense > entity_index.alive_count then - return nil - end - if entity_index.dense_array[r_dense] ~= entity then - return nil - end - end - return r -end - -local function entity_index_try_get_fast(entity_index: EntityIndex, entity: number): Record? - local r = entity_index.sparse_array[ECS_ENTITY_T_LO(entity)] - if r then - if entity_index.dense_array[r.dense] ~= entity then - return nil - end - end - return r -end - -local function entity_index_get_alive(index: EntityIndex, e: i24): i53 - local r = entity_index_try_get_any(index, e) - if r then - return index.dense_array[r.dense] - end - return 0 -end - -local function entity_index_is_alive(entity_index: EntityIndex, entity: number) - return entity_index_try_get(entity_index, entity) ~= nil -end - -local function entity_index_new_id(entity_index: EntityIndex): i53 - local dense_array = entity_index.dense_array - local alive_count = entity_index.alive_count - if alive_count ~= #dense_array then - alive_count += 1 - entity_index.alive_count = alive_count - local id = dense_array[alive_count] - return id - end - - local id = entity_index.max_id + 1 - entity_index.max_id = id - alive_count += 1 - entity_index.alive_count = alive_count - dense_array[alive_count] = id - entity_index.sparse_array[id] = { dense = alive_count } :: Record - - return id -end - --- ECS_PAIR_FIRST, gets the relationship target / obj / HIGH bits -local function ecs_pair_first(world, e) - return entity_index_get_alive(world.entity_index, ECS_ENTITY_T_LO(e)) -end - --- ECS_PAIR_SECOND gets the relationship / pred / LOW bits -local function ecs_pair_second(world, e) - return entity_index_get_alive(world.entity_index, ECS_ENTITY_T_HI(e)) -end - -local function query_match(query, archetype: Archetype) - local records = archetype.records - local with = query.filter_with - - for _, id in with do - if not records[id] then - return false - end - end - - local without = query.filter_without - if without then - for _, id in without do - if records[id] then - return false - end - end - end - - return true -end - -local function find_observers(world: World, event, component): { Observer }? - local cache = world.observable[event] - if not cache then - return nil - end - return cache[component] :: any -end - -local function archetype_move(entity_index: EntityIndex, to: Archetype, dst_row: i24, from: Archetype, src_row: i24) - local src_columns = from.columns - local dst_columns = to.columns - local dst_entities = to.entities - local src_entities = from.entities - - local last = #src_entities - local id_types = from.types - local records = to.records - - for i, column in src_columns do - if column == NULL_ARRAY then - continue - end - -- Retrieves the new column index from the source archetype's record from each component - -- We have to do this because the columns are tightly packed and indexes may not correspond to each other. - local tr = records[id_types[i]] - - -- Sometimes target column may not exist, e.g. when you remove a component. - if tr then - dst_columns[tr][dst_row] = column[src_row] - end - - -- If the entity is the last row in the archetype then swapping it would be meaningless. - if src_row ~= last then - -- Swap rempves columns to ensure there are no holes in the archetype. - column[src_row] = column[last] - end - column[last] = nil - end - - local moved = #src_entities - - -- Move the entity from the source to the destination archetype. - -- Because we have swapped columns we now have to update the records - -- corresponding to the entities' rows that were swapped. - local e1 = src_entities[src_row] - local e2 = src_entities[moved] - - if src_row ~= moved then - src_entities[src_row] = e2 - end - - src_entities[moved] = nil :: any - dst_entities[dst_row] = e1 - - local sparse_array = entity_index.sparse_array - - local record1 = sparse_array[ECS_ENTITY_T_LO(e1)] - local record2 = sparse_array[ECS_ENTITY_T_LO(e2)] - record1.row = dst_row - record2.row = src_row -end - -local function archetype_append(entity: number, archetype: Archetype): number - local entities = archetype.entities - local length = #entities + 1 - entities[length] = entity - return length -end - -local function new_entity(entity: i53, record: Record, archetype: Archetype): Record - local row = archetype_append(entity, archetype) - record.archetype = archetype - record.row = row - return record -end - -local function entity_move(entity_index: EntityIndex, entity: i53, record: Record, to: Archetype) - local sourceRow = record.row - local from = record.archetype - local dst_row = archetype_append(entity, to) - archetype_move(entity_index, to, dst_row, from, sourceRow) - record.archetype = to - record.row = dst_row -end - -local function hash(arr: { number }): string - return table.concat(arr, "_") -end - -local function fetch(id, records: { number }, columns: { Column }, row: number): any - local tr = records[id] - - if not tr then - return nil - end - - return columns[tr][row] -end - -local function world_get(world: World, entity: i53, a: i53, b: i53?, c: i53?, d: i53?, e: i53?): ...any - local record = entity_index_try_get_fast(world.entity_index, entity) - if not record then - return nil - end - - local archetype = record.archetype - if not archetype then - return nil - end - - local records = archetype.records - local columns = archetype.columns - local row = record.row - - local va = fetch(a, records, columns, row) - - if not b then - return va - elseif not c then - return va, fetch(b, records, columns, row) - elseif not d then - return va, fetch(b, records, columns, row), fetch(c, records, columns, row) - elseif not e then - return va, fetch(b, records, columns, row), fetch(c, records, columns, row), fetch(d, records, columns, row) - else - error("args exceeded") - end -end - -local function world_get_one_inline(world: World, entity: i53, id: i53): any - local record = entity_index_try_get_fast(world.entity_index, entity) - if not record then - return nil - end - - local archetype = record.archetype - if not archetype then - return nil - end - - local tr = archetype.records[id] - if not tr then - return nil - end - return archetype.columns[tr][record.row] -end - -local function world_has_one_inline(world: World, entity: number, id: i53): boolean - local record = entity_index_try_get_fast(world.entity_index, entity) - if not record then - return false - end - - local archetype = record.archetype - if not archetype then - return false - end - - local records = archetype.records - - return records[id] ~= nil -end - -local function world_has(world: World, entity: number, ...: i53): boolean - local record = entity_index_try_get_fast(world.entity_index, entity) - if not record then - return false - end - - local archetype = record.archetype - if not archetype then - return false - end - - local records = archetype.records - - for i = 1, select("#", ...) do - if not records[select(i, ...)] then - return false - end - end - - return true -end - -local function world_target(world: World, entity: i53, relation: i24, index: number?): i24? - local nth = index or 0 - local record = entity_index_try_get_fast(world.entity_index, entity) - if not record then - return nil - end - - local archetype = record.archetype - if not archetype then - return nil - end - - local idr = world.component_index[ECS_PAIR(relation, EcsWildcard)] - if not idr then - return nil - end - - local archetype_id = archetype.id - local count = idr.counts[archetype.id] - if not count then - return nil - end - - if nth >= count then - nth = nth + count + 1 - end - - local tr = idr.columns[archetype_id] - - nth = archetype.types[nth + tr] - - if not nth then - return nil - end - - return ecs_pair_second(world, nth) -end - -local function ECS_ID_IS_WILDCARD(e: i53): boolean - local first = ECS_ENTITY_T_HI(e) - local second = ECS_ENTITY_T_LO(e) - return first == EcsWildcard or second == EcsWildcard -end - -local function id_record_ensure(world: World, id: number): IdRecord - local component_index = world.component_index - local idr: IdRecord = component_index[id] - - if not idr then - local flags = ECS_ID_MASK - local relation = id - local is_pair = ECS_IS_PAIR(id) - if is_pair then - relation = ecs_pair_first(world, id) - end - - local cleanup_policy = world_target(world, relation, EcsOnDelete, 0) - local cleanup_policy_target = world_target(world, relation, EcsOnDeleteTarget, 0) - - local has_delete = false - - if cleanup_policy == EcsDelete or cleanup_policy_target == EcsDelete then - has_delete = true - end - - local on_add, on_set, on_remove = world_get(world, relation, EcsOnAdd, EcsOnSet, EcsOnRemove) - - local is_tag = not world_has_one_inline(world, relation, EcsComponent) - - if is_tag and is_pair then - is_tag = not world_has_one_inline(world, ecs_pair_second(world, id), EcsComponent) - end - - flags = bit32.bor( - flags, - if on_add then ECS_ID_HAS_ON_ADD else 0, - if on_remove then ECS_ID_HAS_ON_REMOVE else 0, - if on_set then ECS_ID_HAS_ON_SET else 0, - if has_delete then ECS_ID_DELETE else 0, - if is_tag then ECS_ID_IS_TAG else 0 - ) - - idr = { - size = 0, - columns = {}, - counts = {}, - flags = flags, - hooks = { - on_add = on_add, - on_set = on_set, - on_remove = on_remove, - }, - } - - component_index[id] = idr - end - - return idr -end - -local function archetype_append_to_records( - idr: IdRecord, - archetype: Archetype, - id: number, - index: number -) - local archetype_id = archetype.id - local archetype_records = archetype.records - local archetype_counts = archetype.counts - local idr_columns = idr.columns - local idr_counts = idr.counts - local tr = idr_columns[archetype_id] - if not tr then - idr_columns[archetype_id] = index - idr_counts[archetype_id] = 1 - - archetype_records[id] = index - archetype_counts[id] = 1 - else - local max_count = idr_counts[archetype_id] + 1 - idr_counts[archetype_id] = max_count - archetype_counts[id] = max_count - end -end - -local function archetype_create(world: World, id_types: { i24 }, ty, prev: i53?): Archetype - local archetype_id = (world.max_archetype_id :: number) + 1 - world.max_archetype_id = archetype_id - - local length = #id_types - local columns = (table.create(length) :: any) :: { Column } - - local records: { number } = {} - local counts: {number} = {} - - local archetype: Archetype = { - columns = columns, - entities = {}, - id = archetype_id, - records = records, - counts = counts, - type = ty, - types = id_types, - - add = {}, - remove = {}, - refs = {} :: GraphEdge, - } - - for i, componentId in id_types do - local idr = id_record_ensure(world, componentId) - archetype_append_to_records(idr, archetype, componentId, i) - - if ECS_IS_PAIR(componentId) then - local relation = ecs_pair_first(world, componentId) - local object = ecs_pair_second(world, componentId) - - local r = ECS_PAIR(relation, EcsWildcard) - local idr_r = id_record_ensure(world, r) - archetype_append_to_records(idr_r, archetype, r, i) - - local t = ECS_PAIR(EcsWildcard, object) - local idr_t = id_record_ensure(world, t) - archetype_append_to_records(idr_t, archetype, t, i) - end - - if bit32.band(idr.flags, ECS_ID_IS_TAG) == 0 then - columns[i] = {} - else - columns[i] = NULL_ARRAY - end - end - - for id in records do - local observer_list = find_observers(world, EcsOnArchetypeCreate, id) - if not observer_list then - continue - end - for _, observer in observer_list do - if query_match(observer.query, archetype) then - observer.callback(archetype) - end - end - end - - world.archetype_index[ty] = archetype - world.archetypes[archetype_id] = archetype - - return archetype -end - -local function world_entity(world: World): i53 - return entity_index_new_id(world.entity_index) -end - -local function world_parent(world: World, entity: i53) - return world_target(world, entity, EcsChildOf, 0) -end - -local function archetype_ensure(world: World, id_types): Archetype - if #id_types < 1 then - return world.ROOT_ARCHETYPE - end - - local ty = hash(id_types) - local archetype = world.archetype_index[ty] - if archetype then - return archetype - end - - return archetype_create(world, id_types, ty) -end - -local function find_insert(id_types: { i53 }, toAdd: i53): number - for i, id in id_types do - if id == toAdd then - return -1 - end - if id > toAdd then - return i - end - end - return #id_types + 1 -end - -local function find_archetype_with(world: World, node: Archetype, id: i53): Archetype - local id_types = node.types - -- Component IDs are added incrementally, so inserting and sorting - -- them each time would be expensive. Instead this insertion sort can find the insertion - -- point in the types array. - - local dst = table.clone(node.types) :: { i53 } - local at = find_insert(id_types, id) - if at == -1 then - -- If it finds a duplicate, it just means it is the same archetype so it can return it - -- directly instead of needing to hash types for a lookup to the archetype. - return node - end - table.insert(dst, at, id) - - return archetype_ensure(world, dst) -end - -local function find_archetype_without(world: World, node: Archetype, id: i53): Archetype - local id_types = node.types - local at = table.find(id_types, id) - if at == nil then - return node - end - - local dst = table.clone(id_types) - table.remove(dst, at) - - return archetype_ensure(world, dst) -end - -local function archetype_init_edge(archetype: Archetype, edge: GraphEdge, id: i53, to: Archetype) - edge.from = archetype - edge.to = to - edge.id = id -end - -local function archetype_ensure_edge(world, edges: GraphEdges, id): GraphEdge - local edge = edges[id] - if not edge then - edge = {} :: GraphEdge - edges[id] = edge - end - - return edge -end - -local function init_edge_for_add(world, archetype: Archetype, edge: GraphEdge, id, to: Archetype) - archetype_init_edge(archetype, edge, id, to) - archetype_ensure_edge(world, archetype.add, id) - if archetype ~= to then - local to_refs = to.refs - local next_edge = to_refs.next - - to_refs.next = edge - edge.prev = to_refs - edge.next = next_edge - - if next_edge then - next_edge.prev = edge - end - end -end - -local function init_edge_for_remove(world: World, archetype: Archetype, edge: GraphEdge, id: number, to: Archetype) - archetype_init_edge(archetype, edge, id, to) - archetype_ensure_edge(world, archetype.remove, id) - if archetype ~= to then - local to_refs = to.refs - local prev_edge = to_refs.prev - - to_refs.prev = edge - edge.next = to_refs - edge.prev = prev_edge - - if prev_edge then - prev_edge.next = edge - end - end -end - -local function create_edge_for_add(world: World, node: Archetype, edge: GraphEdge, id: i53): Archetype - local to = find_archetype_with(world, node, id) - init_edge_for_add(world, node, edge, id, to) - return to -end - -local function create_edge_for_remove(world: World, node: Archetype, edge: GraphEdge, id: i53): Archetype - local to = find_archetype_without(world, node, id) - init_edge_for_remove(world, node, edge, id, to) - return to -end - -local function archetype_traverse_add(world: World, id: i53, from: Archetype): Archetype - from = from or world.ROOT_ARCHETYPE - local edge = archetype_ensure_edge(world, from.add, id) - - local to = edge.to - if not to then - to = create_edge_for_add(world, from, edge, id) - end - - return to :: Archetype -end - -local function archetype_traverse_remove(world: World, id: i53, from: Archetype): Archetype - from = from or world.ROOT_ARCHETYPE - - local edge = archetype_ensure_edge(world, from.remove, id) - - local to = edge.to - if not to then - to = create_edge_for_remove(world, from, edge, id) - end - - return to :: Archetype -end - -local function world_add(world: World, entity: i53, id: i53): () - local entity_index = world.entity_index - local record = entity_index_try_get_fast(entity_index, entity) - if not record then - return - end - - local from = record.archetype - local to = archetype_traverse_add(world, id, from) - if from == to then - return - end - if from then - entity_move(entity_index, entity, record, to) - else - if #to.types > 0 then - new_entity(entity, record, to) - end - end - - local idr = world.component_index[id] - local on_add = idr.hooks.on_add - - if on_add then - on_add(entity) - end -end - -local function world_set(world: World, entity: i53, id: i53, data: unknown): () - local entity_index = world.entity_index - local record = entity_index_try_get_fast(entity_index, entity) - if not record then - return - end - - local from: Archetype = record.archetype - local to: Archetype = archetype_traverse_add(world, id, from) - local idr = world.component_index[id] - local idr_hooks = idr.hooks - - if from == to then - -- If the archetypes are the same it can avoid moving the entity - -- and just set the data directly. - local tr = to.records[id] - local column = from.columns[tr] - column[record.row] = data - local on_set = idr_hooks.on_set - if on_set then - on_set(entity, data) - end - - return - end - - if from then - -- If there was a previous archetype, then the entity needs to move the archetype - entity_move(entity_index, entity, record, to) - else - if #to.types > 0 then - -- When there is no previous archetype it should create the archetype - new_entity(entity, record, to) - end - end - - local on_add = idr_hooks.on_add - if on_add then - on_add(entity) - end - - local tr = to.records[id] - local column = to.columns[tr] - - column[record.row] = data - - local on_set = idr_hooks.on_set - if on_set then - on_set(entity, data) - end -end - -local function world_component(world: World): i53 - local id = (world.max_component_id :: number) + 1 - if id > HI_COMPONENT_ID then - -- IDs are partitioned into ranges because component IDs are not nominal, - -- so it needs to error when IDs intersect into the entity range. - error("Too many components, consider using world:entity() instead to create components.") - end - world.max_component_id = id - - return id -end - -local function world_remove(world: World, entity: i53, id: i53) - local entity_index = world.entity_index - local record = entity_index_try_get_fast(entity_index, entity) - if not record then - return - end - local from = record.archetype - - if not from then - return - end - local to = archetype_traverse_remove(world, id, from) - - if from ~= to then - local idr = world.component_index[id] - local on_remove = idr.hooks.on_remove - if on_remove then - on_remove(entity) - end - - entity_move(entity_index, entity, record, to) - end -end - -local function archetype_fast_delete_last(columns: { Column }, column_count: number, types: { i53 }, entity: i53) - for i, column in columns do - if column ~= NULL_ARRAY then - column[column_count] = nil - end - end -end - -local function archetype_fast_delete(columns: { Column }, column_count: number, row, types, entity) - for i, column in columns do - if column ~= NULL_ARRAY then - column[row] = column[column_count] - column[column_count] = nil - end - end -end - -local function archetype_delete(world: World, archetype: Archetype, row: number, destruct: boolean?) - local entity_index = world.entity_index - local component_index = world.component_index - local columns = archetype.columns - local id_types = archetype.types - local entities = archetype.entities - local column_count = #entities - local last = #entities - local move = entities[last] - -- We assume first that the entity is the last in the archetype - local delete = move - - if row ~= last then - local record_to_move = entity_index_try_get_any(entity_index, move) - if record_to_move then - record_to_move.row = row - end - - delete = entities[row] - entities[row] = move - end - - for _, id in id_types do - local idr = component_index[id] - local on_remove = idr.hooks.on_remove - if on_remove then - on_remove(delete) - end - end - - entities[last] = nil :: any - - if row == last then - archetype_fast_delete_last(columns, column_count, id_types, delete) - else - archetype_fast_delete(columns, column_count, row, id_types, delete) - end -end - -local function world_clear(world: World, entity: i53) - --TODO: use sparse_get (stashed) - local record = entity_index_try_get(world.entity_index, entity) - if not record then - return - end - - local archetype = record.archetype - local row = record.row - - if archetype then - -- In the future should have a destruct mode for - -- deleting archetypes themselves. Maybe requires recycling - archetype_delete(world, archetype, row) - end - - record.archetype = nil :: any - record.row = nil :: any -end - -local function archetype_disconnect_edge(edge: GraphEdge) - local edge_next = edge.next - local edge_prev = edge.prev - if edge_next then - edge_next.prev = edge_prev - end - if edge_prev then - edge_prev.next = edge_next - end -end - -local function archetype_remove_edge(edges: Map, id: i53, edge: GraphEdge) - archetype_disconnect_edge(edge) - edges[id] = nil :: any -end - -local function archetype_clear_edges(archetype: Archetype) - local add: GraphEdges = archetype.add - local remove: GraphEdges = archetype.remove - local node_refs = archetype.refs - for id, edge in add do - archetype_disconnect_edge(edge) - add[id] = nil :: any - end - for id, edge in remove do - archetype_disconnect_edge(edge) - remove[id] = nil :: any - end - - local cur = node_refs.next - while cur do - local edge = cur :: GraphEdge - local next_edge = edge.next - archetype_remove_edge(edge.from.add, edge.id, edge) - cur = next_edge - end - - cur = node_refs.prev - while cur do - local edge: GraphEdge = cur - local next_edge = edge.prev - archetype_remove_edge(edge.from.remove, edge.id, edge) - cur = next_edge - end - - node_refs.next = nil - node_refs.prev = nil -end - -local function archetype_destroy(world: World, archetype: Archetype) - if archetype == world.ROOT_ARCHETYPE then - return - end - - local component_index = world.component_index - archetype_clear_edges(archetype) - local archetype_id = archetype.id - world.archetypes[archetype_id] = nil :: any - world.archetype_index[archetype.type] = nil :: any - local records = archetype.records - - for id in records do - local observer_list = find_observers(world, EcsOnArchetypeDelete, id) - if not observer_list then - continue - end - for _, observer in observer_list do - if query_match(observer.query, archetype) then - observer.callback(archetype) - end - end - end - - for id in records do - local idr = component_index[id] - idr.columns[archetype_id] = nil :: any - idr.counts[archetype_id] = nil - idr.size -= 1 - records[id] = nil :: any - if idr.size == 0 then - component_index[id] = nil :: any - end - end -end - -local function world_cleanup(world: World) - local archetypes = world.archetypes - - for _, archetype in archetypes do - if #archetype.entities == 0 then - archetype_destroy(world, archetype) - end - end - - local new_archetypes = table.create(#archetypes) :: { Archetype } - local new_archetype_map = {} - - for index, archetype in archetypes do - new_archetypes[index] = archetype - new_archetype_map[archetype.type] = archetype - end - - world.archetypes = new_archetypes - world.archetype_index = new_archetype_map -end - -local world_delete: (world: World, entity: i53, destruct: boolean?) -> () -do - function world_delete(world: World, entity: i53, destruct: boolean?) - local entity_index = world.entity_index - local record = entity_index_try_get(entity_index, entity) - if not record then - return - end - - local archetype = record.archetype - local row = record.row - - if archetype then - -- In the future should have a destruct mode for - -- deleting archetypes themselves. Maybe requires recycling - archetype_delete(world, archetype, row, destruct) - end - - local delete = entity - local component_index = world.component_index - local archetypes: Archetypes = world.archetypes - local tgt = ECS_PAIR(EcsWildcard, delete) - local idr_t = component_index[tgt] - local idr = component_index[delete] - - if idr then - local flags = idr.flags - if bit32.band(flags, ECS_ID_DELETE) ~= 0 then - for archetype_id in idr.columns do - local idr_archetype = archetypes[archetype_id] - - local entities = idr_archetype.entities - local n = #entities - for i = n, 1, -1 do - world_delete(world, entities[i]) - end - - archetype_destroy(world, idr_archetype) - end - else - for archetype_id in idr.columns do - local idr_archetype = archetypes[archetype_id] - local entities = idr_archetype.entities - local n = #entities - for i = n, 1, -1 do - world_remove(world, entities[i], delete) - end - - archetype_destroy(world, idr_archetype) - end - end - end - - local sparse_array = entity_index.sparse_array - local dense_array = entity_index.dense_array - - if idr_t then - for archetype_id in idr_t.columns do - local children = {} - local idr_t_archetype = archetypes[archetype_id] - - local idr_t_types = idr_t_archetype.types - - for _, child in idr_t_archetype.entities do - table.insert(children, child) - end - - local n = #children - - for _, id in idr_t_types do - if not ECS_IS_PAIR(id) then - continue - end - local object = ecs_pair_second(world, id) - if object == delete then - local id_record = component_index[id] - local flags = id_record.flags - local flags_delete_mask: number = bit32.band(flags, ECS_ID_DELETE) - if flags_delete_mask ~= 0 then - for i = n, 1, -1 do - world_delete(world, children[i]) - end - break - else - local on_remove = id_record.hooks.on_remove - local to = archetype_traverse_remove(world, id, idr_t_archetype) - local empty = #to.types == 0 - for i = n, 1, -1 do - local child = children[i] - if on_remove then - on_remove(child) - end - local r = sparse_array[ECS_ENTITY_T_LO(child)] - if not empty then - entity_move(entity_index, child, r, to) - end - end - end - end - end - - archetype_destroy(world, idr_t_archetype) - end - end - - local index_of_deleted_entity = record.dense - local index_of_last_alive_entity = entity_index.alive_count - entity_index.alive_count = index_of_last_alive_entity - 1 - - local last_alive_entity = dense_array[index_of_last_alive_entity] - local r_swap = entity_index_try_get_any(entity_index, last_alive_entity) :: Record - r_swap.dense = index_of_deleted_entity - record.archetype = nil :: any - record.row = nil :: any - record.dense = index_of_last_alive_entity - - dense_array[index_of_deleted_entity] = last_alive_entity - dense_array[index_of_last_alive_entity] = ECS_GENERATION_INC(entity) - end -end - -local function world_contains(world: World, entity): boolean - return entity_index_is_alive(world.entity_index, entity) -end - -local function NOOP() end - -local function ARM(query, ...) - return query -end - -local EMPTY_LIST = {} -local EMPTY_QUERY = { - __iter = function() - return NOOP - end, - iter = function() - return NOOP - end, - with = ARM, - without = ARM, - archetypes = function() - return EMPTY_LIST - end, -} - -setmetatable(EMPTY_QUERY, EMPTY_QUERY) - -type QueryInner = { - compatible_archetypes: { Archetype }, - ids: { i53 }, - filter_with: { i53 }, - filter_without: { i53 }, - next: () -> (number, ...any), - world: World, -} - -local function query_iter_init(query: QueryInner): () -> (number, ...any) - local world_query_iter_next - - local compatible_archetypes = query.compatible_archetypes - local lastArchetype = 1 - local archetype = compatible_archetypes[1] - if not archetype then - return NOOP :: () -> (number, ...any) - end - local columns = archetype.columns - local entities = archetype.entities - local i = #entities - local records = archetype.records - - local ids = query.ids - local A, B, C, D, E, F, G, H, I = unpack(ids) - local a: Column, b: Column, c: Column, d: Column - local e: Column, f: Column, g: Column, h: Column - - if not B then - a = columns[records[A]] - elseif not C then - a = columns[records[A]] - b = columns[records[B]] - elseif not D then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - elseif not E then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - elseif not F then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - elseif not G then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - elseif not H then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - elseif not I then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - h = columns[records[H]] - end - - if not B then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - end - - local row = i - i -= 1 - - return entity, a[row] - end - elseif not C then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row] - end - elseif not D then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row] - end - elseif not E then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row] - end - elseif not F then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row] - end - elseif not G then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row], f[row] - end - elseif not H then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row], f[row], g[row] - end - elseif not I then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - h = columns[records[H]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row], f[row], g[row], h[row] - end - else - local output = {} - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - end - - local row = i - i -= 1 - - for j, id in ids do - output[j] = columns[records[id]][row] - end - - return entity, unpack(output) - end - end - - query.next = world_query_iter_next - return world_query_iter_next -end - -local function query_iter(query): () -> (number, ...any) - local query_next = query.next - if not query_next then - query_next = query_iter_init(query) - end - return query_next -end - -local function query_without(query: QueryInner, ...: i53) - local without = { ... } - query.filter_without = without - local compatible_archetypes = query.compatible_archetypes - for i = #compatible_archetypes, 1, -1 do - local archetype = compatible_archetypes[i] - local records = archetype.records - local matches = true - - for _, id in without do - if records[id] then - matches = false - break - end - end - - if matches then - continue - end - - local last = #compatible_archetypes - if last ~= i then - compatible_archetypes[i] = compatible_archetypes[last] - end - compatible_archetypes[last] = nil :: any - end - - return query :: any -end - -local function query_with(query: QueryInner, ...: i53) - local compatible_archetypes = query.compatible_archetypes - local with = { ... } - query.filter_with = with - - for i = #compatible_archetypes, 1, -1 do - local archetype = compatible_archetypes[i] - local records = archetype.records - local matches = true - - for _, id in with do - if not records[id] then - matches = false - break - end - end - - if matches then - continue - end - - local last = #compatible_archetypes - if last ~= i then - compatible_archetypes[i] = compatible_archetypes[last] - end - compatible_archetypes[last] = nil :: any - end - - return query :: any -end - --- Meant for directly iterating over archetypes to minimize --- function call overhead. Should not be used unless iterating over --- hundreds of thousands of entities in bulk. -local function query_archetypes(query) - return query.compatible_archetypes -end - -local function query_cached(query: QueryInner) - local with = query.filter_with - local ids = query.ids - if with then - table.move(ids, 1, #ids, #with + 1, with) - else - query.filter_with = ids - end - - local compatible_archetypes = query.compatible_archetypes - local lastArchetype = 1 - - local A, B, C, D, E, F, G, H, I = unpack(ids) - local a: Column, b: Column, c: Column, d: Column - local e: Column, f: Column, g: Column, h: Column - - local world_query_iter_next - local columns: { Column } - local entities: { number } - local i: number - local archetype: Archetype - local records: { number } - local archetypes = query.compatible_archetypes - - local world = query.world :: { observable: Observable } - -- Only need one observer for EcsArchetypeCreate and EcsArchetypeDelete respectively - -- because the event will be emitted for all components of that Archetype. - local observable = world.observable :: Observable - local on_create_action = observable[EcsOnArchetypeCreate] - if not on_create_action then - on_create_action = {} - observable[EcsOnArchetypeCreate] = on_create_action - end - local query_cache_on_create = on_create_action[A] - if not query_cache_on_create then - query_cache_on_create = {} - on_create_action[A] = query_cache_on_create - end - - local on_delete_action = observable[EcsOnArchetypeDelete] - if not on_delete_action then - on_delete_action = {} - observable[EcsOnArchetypeDelete] = on_delete_action - end - local query_cache_on_delete = on_delete_action[A] - if not query_cache_on_delete then - query_cache_on_delete = {} - on_delete_action[A] = query_cache_on_delete - end - - local function on_create_callback(archetype) - table.insert(archetypes, archetype) - end - - local function on_delete_callback(archetype) - local i = table.find(archetypes, archetype) :: number - local n = #archetypes - archetypes[i] = archetypes[n] - archetypes[n] = nil - end - - local observer_for_create = { query = query, callback = on_create_callback } - local observer_for_delete = { query = query, callback = on_delete_callback } - - table.insert(query_cache_on_create, observer_for_create) - table.insert(query_cache_on_delete, observer_for_delete) - - local function cached_query_iter() - lastArchetype = 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return NOOP - end - entities = archetype.entities - i = #entities - records = archetype.records - columns = archetype.columns - if not B then - a = columns[records[A]] - elseif not C then - a = columns[records[A]] - b = columns[records[B]] - elseif not D then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - elseif not E then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - elseif not F then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - elseif not G then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - elseif not H then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - elseif not I then - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - h = columns[records[H]] - end - - return world_query_iter_next - end - - if not B then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - end - - local row = i - i -= 1 - - return entity, a[row] - end - elseif not C then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row] - end - elseif not D then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row] - end - elseif not E then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row] - end - elseif not F then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row] - end - elseif not G then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row], f[row] - end - elseif not H then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row], f[row], g[row] - end - elseif not I then - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - a = columns[records[A]] - b = columns[records[B]] - c = columns[records[C]] - d = columns[records[D]] - e = columns[records[E]] - f = columns[records[F]] - g = columns[records[G]] - h = columns[records[H]] - end - - local row = i - i -= 1 - - return entity, a[row], b[row], c[row], d[row], e[row], f[row], g[row], h[row] - end - else - local queryOutput = {} - function world_query_iter_next(): any - local entity = entities[i] - while entity == nil do - lastArchetype += 1 - archetype = compatible_archetypes[lastArchetype] - if not archetype then - return nil - end - - entities = archetype.entities - i = #entities - if i == 0 then - continue - end - entity = entities[i] - columns = archetype.columns - records = archetype.records - end - - local row = i - i -= 1 - - if not F then - return entity, a[row], b[row], c[row], d[row], e[row] - elseif not G then - return entity, a[row], b[row], c[row], d[row], e[row], f[row] - elseif not H then - return entity, a[row], b[row], c[row], d[row], e[row], f[row], g[row] - elseif not I then - return entity, a[row], b[row], c[row], d[row], e[row], f[row], g[row], h[row] - end - - for j, id in ids do - queryOutput[j] = columns[records[id]][row] - end - - return entity, unpack(queryOutput) - end - end - - local cached_query = query :: any - cached_query.archetypes = query_archetypes - cached_query.__iter = cached_query_iter - cached_query.iter = cached_query_iter - setmetatable(cached_query, cached_query) - return cached_query -end - -local Query = {} -Query.__index = Query -Query.__iter = query_iter -Query.iter = query_iter_init -Query.without = query_without -Query.with = query_with -Query.archetypes = query_archetypes -Query.cached = query_cached - -local function world_query(world: World, ...) - local compatible_archetypes = {} - local length = 0 - - local ids = { ... } - - local archetypes = world.archetypes - - local idr: IdRecord? - local component_index = world.component_index - - local q = setmetatable({ - ids = ids, - compatible_archetypes = compatible_archetypes, - world = world, - }, Query) - - for _, id in ids do - local map = component_index[id] - if not map then - return q - end - - if idr == nil or map.size < idr.size then - idr = map - end - end - - if not idr then - return q - end - - for archetype_id in idr.columns do - local compatibleArchetype = archetypes[archetype_id] - if #compatibleArchetype.entities == 0 then - continue - end - local records = compatibleArchetype.records - - local skip = false - - for i, id in ids do - local tr = records[id] - if not tr then - skip = true - break - end - end - - if skip then - continue - end - - length += 1 - compatible_archetypes[length] = compatibleArchetype - end - - return q -end - -local function world_each(world: World, id): () -> () - local idr = world.component_index[id] - if not idr then - return NOOP - end - - local idr_columns = idr.columns - local archetypes = world.archetypes - local archetype_id = next(idr_columns, nil) :: number - local archetype = archetypes[archetype_id] - if not archetype then - return NOOP - end - - local entities = archetype.entities - local row = #entities - - return function(): any - local entity = entities[row] - while not entity do - archetype_id = next(idr_columns, archetype_id) :: number - if not archetype_id then - return - end - archetype = archetypes[archetype_id] - entities = archetype.entities - row = #entities - entity = entities[row] - end - row -= 1 - return entity - end -end - -local function world_children(world, parent) - return world_each(world, ECS_PAIR(EcsChildOf, parent)) -end - -local World = {} -World.__index = World - -World.entity = world_entity -World.query = world_query -World.remove = world_remove -World.clear = world_clear -World.delete = world_delete -World.component = world_component -World.add = world_add -World.set = world_set -World.get = world_get -World.has = world_has -World.target = world_target -World.parent = world_parent -World.contains = world_contains -World.cleanup = world_cleanup -World.each = world_each -World.children = world_children - -if _G.__JECS_DEBUG then - local function dbg_info(n: number): any - return debug.info(n, "s") - end - local function throw(msg: string) - local s = 1 - local root = dbg_info(1) - repeat - s += 1 - until dbg_info(s) ~= root - if warn then - error(msg, s) - else - print(`[jecs] error: {msg}\n`) - end - end - - local function ASSERT(v: T, msg: string) - if v then - return - end - throw(msg) - end - - local function get_name(world, id) - return world_get_one_inline(world, id, EcsName) - end - - local function bname(world: World, id): string - local name: string - if ECS_IS_PAIR(id) then - local first = get_name(world, ecs_pair_first(world, id)) - local second = get_name(world, ecs_pair_second(world, id)) - name = `pair({first}, {second})` - else - return get_name(world, id) - end - if name then - return name - else - return `${id}` - end - end - - local function ID_IS_TAG(world: World, id) - if ECS_IS_PAIR(id) then - id = ecs_pair_first(world, id) - end - return not world_has_one_inline(world, id, EcsComponent) - end - - World.query = function(world: World, ...) - ASSERT((...), "Requires at least a single component") - return world_query(world, ...) - end - - World.set = function(world: World, entity: i53, id: i53, value: any): () - local is_tag = ID_IS_TAG(world, id) - if is_tag and value == nil then - local _1 = bname(world, entity) - local _2 = bname(world, id) - local why = "cannot set component value to nil" - throw(why) - return - elseif value ~= nil and is_tag then - local _1 = bname(world, entity) - local _2 = bname(world, id) - local why = `cannot set a component value because {_2} is a tag` - why ..= `\n[jecs] note: consider using "world:add({_1}, {_2})" instead` - throw(why) - return - end - - world_set(world, entity, id, value) - end - - World.add = function(world: World, entity: i53, id: i53, value: any) - if value ~= nil then - local _1 = bname(world, entity) - local _2 = bname(world, id) - throw("You provided a value when none was expected. " .. `Did you mean to use "world:add({_1}, {_2})"`) - end - - world_add(world, entity, id) - end - - World.get = function(world: World, entity: i53, ...) - local length = select("#", ...) - ASSERT(length < 5, "world:get does not support more than 4 components") - local _1 - for i = 1, length do - local id = select(i, ...) - local id_is_tag = not world_has(world, id, EcsComponent) - if id_is_tag then - local name = get_name(world, id) - if not _1 then - _1 = get_name(world, entity) - end - throw( - `cannot get (#{i}) component {name} value because it is a tag.` - .. `\n[jecs] note: If this was intentional, use "world:has({_1}, {name}) instead"` - ) - end - end - - return world_get(world, entity, ...) - end -end - -function World.new() - local entity_index: EntityIndex = { - dense_array = {} :: { [i24]: i53 }, - sparse_array = {} :: { [i53]: Record }, - alive_count = 0, - max_id = 0, - } - local self = setmetatable({ - archetype_index = {} :: { [string]: Archetype }, - archetypes = {} :: Archetypes, - component_index = {} :: ComponentIndex, - entity_index = entity_index, - ROOT_ARCHETYPE = (nil :: any) :: Archetype, - - max_archetype_id = 0, - max_component_id = 0, - - observable = {} :: Observable, - }, World) :: any - - self.ROOT_ARCHETYPE = archetype_create(self, {}, "") - - for i = 1, HI_COMPONENT_ID do - local e = entity_index_new_id(entity_index) - world_add(self, e, EcsComponent) - end - - for i = HI_COMPONENT_ID + 1, EcsRest do - -- Initialize built-in components - entity_index_new_id(entity_index) - end - - world_add(self, EcsName, EcsComponent) - world_add(self, EcsOnSet, EcsComponent) - world_add(self, EcsOnAdd, EcsComponent) - world_add(self, EcsOnRemove, EcsComponent) - world_add(self, EcsWildcard, EcsComponent) - world_add(self, EcsRest, EcsComponent) - - world_set(self, EcsOnAdd, EcsName, "jecs.OnAdd") - world_set(self, EcsOnRemove, EcsName, "jecs.OnRemove") - world_set(self, EcsOnSet, EcsName, "jecs.OnSet") - world_set(self, EcsWildcard, EcsName, "jecs.Wildcard") - world_set(self, EcsChildOf, EcsName, "jecs.ChildOf") - world_set(self, EcsComponent, EcsName, "jecs.Component") - world_set(self, EcsOnDelete, EcsName, "jecs.OnDelete") - world_set(self, EcsOnDeleteTarget, EcsName, "jecs.OnDeleteTarget") - world_set(self, EcsDelete, EcsName, "jecs.Delete") - world_set(self, EcsRemove, EcsName, "jecs.Remove") - world_set(self, EcsName, EcsName, "jecs.Name") - world_set(self, EcsRest, EcsRest, "jecs.Rest") - - world_add(self, EcsChildOf, ECS_PAIR(EcsOnDeleteTarget, EcsDelete)) - - return self -end - -export type Entity = {__T: T} - -export type Id = - | Entity - | Pair, Entity> - | Pair> - | Pair, Entity> - -export type Pair = number & { - __P: P, - __O: O, -} - -type Item = (self: Query) -> (Entity, T...) - -type Iter = (query: Query) -> () -> (Entity, T...) - - -export type Query = typeof(setmetatable({}, { - __iter = (nil :: any) :: Iter, -})) & { - iter: Iter, - with: (self: Query, ...Id) -> Query, - without: (self: Query, ...Id) -> Query, - archetypes: (self: Query) -> { Archetype }, - cached: (self: Query) -> Query, -} - -export type Observer = { - callback: (archetype: Archetype) -> (), - query: QueryInner, -} - -type Observable = { - [i53]: { - [i53]: { - { Observer } - } - } -} - -export type World = { - archetype_index: { [string]: Archetype }, - archetypes: Archetypes, - component_index: ComponentIndex, - entity_index: EntityIndex, - ROOT_ARCHETYPE: Archetype, - - max_component_id: number, - max_archetype_id: number, - - observable: any, - - --- Creates a new entity - entity: (self: World) -> Entity, - --- Creates a new entity located in the first 256 ids. - --- These should be used for static components for fast access. - component: (self: World) -> Entity, - --- Gets the target of an relationship. For example, when a user calls - --- `world:target(id, ChildOf(parent), 0)`, you will obtain the parent entity. - target: (self: World, id: Entity, relation: Entity, index: number?) -> Entity?, - --- Deletes an entity and all it's related components and relationships. - delete: (self: World, id: Entity) -> (), - - --- Adds a component to the entity with no value - add: (self: World, id: Entity, component: Id) -> (), - --- Assigns a value to a component on the given entity - set: (self: World, id: Entity, component: Id, data: U) -> (), - - cleanup: (self: World) -> (), - -- Clears an entity from the world - clear: (self: World, id: Entity) -> (), - --- Removes a component from the given entity - remove: (self: World, id: Entity, component: Id) -> (), - --- Retrieves the value of up to 4 components. These values may be nil. - get: ((self: World, id: Entity, Id) -> A?) - & ((self: World, id: Entity, Id, Id) -> (A?, B?)) - & ((self: World, id: Entity, Id, Id, Id) -> (A?, B?, C?)) - & (self: World, id: Entity, Id, Id, Id, Id) -> (A?, B?, C?, D?), - - --- Returns whether the entity has the ID. - has: ((self: World, entity: Entity, ...Id) -> boolean) - & ((self: World, entity: Entity, Id, Id) -> boolean) - & ((self: World, entity: Entity, Id, Id, Id) -> boolean) - & ((self: World, entity: Entity, Id, Id, Id, Id) -> boolean) - & ((self: World, entity: Entity, Id, Id, Id, Id, Id) -> boolean) - & ((self: World, entity: Entity, Id, Id, Id, Id, Id, Id) -> boolean) - & ((self: World, entity: Entity, Id, Id, Id, Id, Id, Id, Id) -> boolean) - & ((self: World, entity: Entity, Id, Id, Id, Id, Id, Id, Id, ...unknown) -> boolean), - - --- Get parent (target of ChildOf relationship) for entity. If there is no ChildOf relationship pair, it will return nil. - parent: (self: World, entity: Entity) -> Entity, - - --- Checks if the world contains the given entity - contains: (self: World, entity: Entity) -> boolean, - - each: (self: World, id: Id) -> () -> Entity, - - children: (self: World, id: Id) -> () -> Entity, - - --- Searches the world for entities that match a given query - query: ((World, Id) -> Query) - & ((World, Id, Id) -> Query) - & ((World, Id, Id, Id) -> Query) - & ((World, Id, Id, Id, Id) -> Query) - & ((World, Id, Id, Id, Id, Id) -> Query) - & ((World, Id, Id, Id, Id, Id, Id) -> Query) - & ((World, Id, Id, Id, Id, Id, Id, Id) -> Query) - & ((World, Id, Id, Id, Id, Id, Id, Id, Id, ...Id) -> Query) -} --- type function ecs_id_t(entity) --- local ty = entity:components()[2] --- local __T = ty:readproperty(types.singleton("__T")) --- if not __T then --- return ty:readproperty(types.singleton("__jecs_pair_value")) --- end --- return __T --- end - --- type function ecs_pair_t(first, second) --- if ecs_id_t(first):is("nil") then --- return second --- else --- return first --- end --- end - -return { - World = World :: { new: () -> World }, - - OnAdd = EcsOnAdd :: Entity<(entity: Entity) -> ()>, - OnRemove = EcsOnRemove :: Entity<(entity: Entity) -> ()>, - OnSet = EcsOnSet :: Entity<(entity: Entity, data: any) -> ()>, - ChildOf = EcsChildOf :: Entity, - Component = EcsComponent :: Entity, - Wildcard = EcsWildcard :: Entity, - w = EcsWildcard :: Entity, - OnDelete = EcsOnDelete :: Entity, - OnDeleteTarget = EcsOnDeleteTarget :: Entity, - Delete = EcsDelete :: Entity, - Remove = EcsRemove :: Entity, - Name = EcsName :: Entity, - Rest = EcsRest :: Entity, - - pair = ECS_PAIR :: (first: P, second: O) -> Pair, - - -- Inwards facing API for testing - ECS_ID = ECS_ENTITY_T_LO, - ECS_GENERATION_INC = ECS_GENERATION_INC, - ECS_GENERATION = ECS_GENERATION, - ECS_ID_IS_WILDCARD = ECS_ID_IS_WILDCARD, - - IS_PAIR = ECS_IS_PAIR, - pair_first = ecs_pair_first, - pair_second = ecs_pair_second, - entity_index_get_alive = entity_index_get_alive, - - archetype_append_to_records = archetype_append_to_records, - id_record_ensure = id_record_ensure, - archetype_create = archetype_create, - archetype_ensure = archetype_ensure, - find_insert = find_insert, - find_archetype_with = find_archetype_with, - find_archetype_without = find_archetype_without, - archetype_init_edge = archetype_init_edge, - archetype_ensure_edge = archetype_ensure_edge, - init_edge_for_add = init_edge_for_add, - init_edge_for_remove = init_edge_for_remove, - create_edge_for_add = create_edge_for_add, - create_edge_for_remove = create_edge_for_remove, - archetype_traverse_add = archetype_traverse_add, - archetype_traverse_remove = archetype_traverse_remove, - - entity_move = entity_move, - - entity_index_try_get = entity_index_try_get, - entity_index_try_get_any = entity_index_try_get_any, - entity_index_try_get_fast = entity_index_try_get_fast, - entity_index_is_alive = entity_index_is_alive, - entity_index_new_id = entity_index_new_id, - - query_iter = query_iter, - query_iter_init = query_iter_init, - query_with = query_with, - query_without = query_without, - query_archetypes = query_archetypes, - query_match = query_match, - - find_observers = find_observers, -} diff --git a/jecs/pesde.toml b/jecs/pesde.toml deleted file mode 100644 index 909f63c..0000000 --- a/jecs/pesde.toml +++ /dev/null @@ -1,13 +0,0 @@ -authors = ["jecs authors"] -includes = ["init.luau", "pesde.toml", "README.md", "CHANGELOG.md", "LICENSE", ".luaurc"] -license = "MIT" -name = "marked/jecs_nightly" -repository = "https://git.devmarked.win/jecs-nightly" -version = "0.5.5-nightly.20250302T042604Z" - -[indices] -default = "https://github.com/daimond113/pesde-index" - -[target] -environment = "luau" -lib = "jecs.luau" diff --git a/jecs/wally.toml b/jecs/wally.toml deleted file mode 100644 index 0f8c345..0000000 --- a/jecs/wally.toml +++ /dev/null @@ -1,8 +0,0 @@ -[package] -exclude = ["**"] -include = ["default.project.json", "jecs.luau", "wally.toml", "README.md", "CHANGELOG.md", "LICENSE"] -license = "MIT" -name = "mark-marks/jecs-nightly" -realm = "shared" -registry = "https://github.com/UpliftGames/wally-index" -version = "0.5.5-nightly.20250302T042604Z" diff --git a/src/init.luau b/src/init.luau index edf368a..ef84caf 100644 --- a/src/init.luau +++ b/src/init.luau @@ -1,16 +1,80 @@ --!strict +local fs = require("@lune/fs") +local process = require("@lune/process") + +local frkcli = require("./util/frkcli") local release = require("./release") local sync = require("./sync") +local test = require("./test") -local res = sync("jecs") -if not res.ok then - print(`Syncing failed, aborting.`) - return +local cli = frkcli.new_subcommands("nightly_cli", "A CLI to sync & release jecs nightly") + +local command_sync = cli:add_subcommand("sync", "Synchronize files from the jecs repository") +command_sync:add_positional( + "to", + { help = "Directory to synchronize the files in, defaults to current working directory", default = process.cwd } +) + +local command_release = cli:add_subcommand("release", "Publish a release from the given directory") +command_release:add_positional( + "from", + { help = "Directory to release from, defaults to current working directory", default = process.cwd } +) +command_release:add_flag("dry", { help = "Dry run the publishes" }) +command_release:add_option( + "pesde-scope", + { help = "The pesde scope to publish under (eg. marked/jecs_nightly)", default = "nil" } +) +command_release:add_option( + "wally-scope", + { help = "The wally scope to publish under (eg. mark-marks/jecs-nightly)", default = "nil" } +) + +local command_test = cli:add_subcommand("test", "Run unit tests on the stored jecs source") +command_test:add_positional("in", { + help = "Directory with the jecs source to run tests in, defaults to current working directory", + default = process.cwd, +}) + +local parsed, err = cli:parse(process.args) +if err ~= nil then + error(err) end +assert(parsed ~= nil) -if res.val == false then - print(`No changes made. Aborting.`) - return +local values = parsed.result.values +local flags = parsed.result.flags +if parsed.command == "sync" then + sync(values.to) +elseif parsed.command == "release" then + local scopes = {} + if values["pesde-scope"] ~= "nil" then + scopes.pesde = values["pesde-scope"] + end + if values["wally-scope"] ~= "nil" then + scopes.wally = values["wally-scope"] + end + + local metadata = fs.metadata(values.to) + if not metadata.exists or metadata.kind ~= "dir" then + error(`The path {values.to} doesn't exist or isn't a valid directory.`) + end + + local fpath = process.cwd + if values.to ~= process.cwd then + fpath ..= values.to + end + + if flags.dry then + release(fpath, scopes, true) + else + release(fpath, scopes, false) + end +elseif parsed.command == "test" then + local fpath = process.cwd + if values.to ~= process.cwd then + fpath ..= values["in"] + end + + test(fpath) end - -release("jecs", { pesde = "marked/jecs_nightly", wally = "mark-marks/jecs-nightly" }, false) diff --git a/src/release.luau b/src/release.luau index 37408bc..7fa6190 100644 --- a/src/release.luau +++ b/src/release.luau @@ -1,20 +1,14 @@ --!strict -local datetime = require("@lune/datetime") local fs = require("@lune/fs") -local net = require("@lune/net") local process = require("@lune/process") local serde = require("@lune/serde") local stdio = require("@lune/stdio") local progress_bar = require("./util/progress") local result = require("./util/result") +local shared = require("./shared") local types = require("./types") --- Returns an ISO 8601 date (YYYYmmddThhmmssZ) -local function iso_date_light(now: datetime.DateTime): string - return now:formatUniversalTime("%Y%m%dT%H%M%SZ") -end - local function make_pesde_manifest(version: string, scope: string): types.PesdeManifest return { name = scope, @@ -63,32 +57,17 @@ local function make_wally_manifest(version: string, scope: string): types.WallyM } end -local function round_to(n: number, places: number) - local x = 10 ^ (places or 0) - return math.round(n * x) / x -end - ---- Fetches the given file raw from the jecs github -local function fetch_raw(file: string): result.Identity - local res = net.request(`https://raw.githubusercontent.com/Ukendio/jecs/refs/heads/main/{file}`) - if not res.ok then - return result(false, `Not ok: {res.statusMessage} @ {res.statusCode}`) - end - return result(true, res.body) -end - -local function release(origin: string, scopes: { wally: string, pesde: string }, dry: boolean?): result.Identity +local function release(origin: string, scopes: { wally: string?, pesde: string? }, dry: boolean?): result.Identity local begin = os.clock() - local now = datetime.now() local progress = progress_bar .new() :withStage("init", "Initializing") - :withStage("pull", "Pull latest version") + :withStage("version", "Pulling version from build metadata") :withStage("prepare", "Preparing manifests") :withStage("release (pesde)", "Releasing on pesde") :withStage("release (wally)", "Releasing on wally") - progress:start() + progress:start() -- init if dry == nil then dry = true @@ -100,79 +79,96 @@ local function release(origin: string, scopes: { wally: string, pesde: string }, return result(false, `{origin} is not a valid directory which exists.`) end - progress:nextStage() + progress:nextStage() -- version - local wally_contents = fetch_raw("wally.toml") - if not wally_contents.ok then - progress:stop() - stdio.ewrite(`๐Ÿ”ฅ Couldn't get the jecs wally manifest:\n{wally_contents.err}\n`) - return result(false, `Couldn't get the jecs wally manifest:\n{wally_contents.err}`) + local version + do + local encoded_metadata = fs.readFile(`{origin}/build.txt`) + local metadata: { version: string } = serde.decode("toml", encoded_metadata) + version = metadata.version end - local parsed: types.WallyManifest = serde.decode("toml", wally_contents.val) - local version = `{parsed.package.version}-nightly.{iso_date_light(now)}` + progress:nextStage() -- prepare - progress:nextStage() - - do + if scopes.pesde then local manifest = make_pesde_manifest(version, scopes.pesde) local encoded = serde.encode("toml", manifest) fs.writeFile(`{origin}/pesde.toml`, encoded) end - do + if scopes.wally then local manifest = make_wally_manifest(version, scopes.wally) local encoded = serde.encode("toml", manifest) fs.writeFile(`{origin}/wally.toml`, encoded) end - progress:nextStage() + progress:nextStage() -- release (pesde) - local cwd = process.cwd .. origin + local cwd = origin --process.cwd .. origin local res_pesde - if not dry then - res_pesde = process.spawn("pesde", { "publish", "-y" }, { cwd = cwd }) - else - res_pesde = process.spawn("pesde", { "publish", "-d", "-y" }, { cwd = cwd }) + if scopes.pesde then + if not dry then + res_pesde = process.spawn("pesde", { "publish", "-y" }, { cwd = cwd }) + else + res_pesde = process.spawn("pesde", { "publish", "-d", "-y" }, { cwd = cwd }) + end + + if not res_pesde.ok then + progress:stop() + print(`-- Pesde error:\nstdout\n{res_pesde.stdout}\nstderr\n{res_pesde.stderr}`) + return result(false, `Pesde error:\nstdout\n{res_pesde.stdout}\nstderr\n{res_pesde.stderr}`) + end end - if not res_pesde.ok then - progress:stop() - print(`-- Pesde error:\nstdout\n{res_pesde.stdout}\nstderr\n{res_pesde.stderr}`) - return result(false, `Pesde error:\nstdout\n{res_pesde.stdout}\nstderr\n{res_pesde.stderr}`) - end - - progress:nextStage() + progress:nextStage() -- release (wally) local res_wally - if not dry then - res_wally = process.spawn("wally", { "publish" }, { cwd = cwd }) - else - res_wally = process.spawn("wally", { "package", "--output", "wally.tar.gz" }, { cwd = cwd }) + if scopes.wally then + if not dry then + res_wally = process.spawn("wally", { "publish" }, { cwd = cwd }) + else + res_wally = process.spawn("wally", { "package", "--output", "wally.tar.gz" }, { cwd = cwd }) + end + + if not res_wally.ok then + progress:stop() + print(`-- Wally error:\nstdout\n{res_wally.stdout}\nstderr\n{res_wally.stderr}`) + return result(false, `Wally error:\nstdout\n{res_wally.stdout}\nstderr\n{res_wally.stderr}`) + end end - if not res_wally.ok then - progress:stop() - print(`-- Wally error:\nstdout\n{res_wally.stdout}\nstderr\n{res_wally.stderr}`) - return result(false, `Wally error:\nstdout\n{res_wally.stdout}\nstderr\n{res_wally.stderr}`) + progress:stop() -- finish + + if res_pesde then + print(`-- Pesde out:\nstdout\n{res_pesde.stdout}\nstderr\n{res_pesde.stderr}`) end - progress:stop() + if res_wally then + print(`-- Wally out:\nstdout\n{res_wally.stdout}\nstderr\n{res_wally.stderr}`) + end - print(`-- Pesde out:\nstdout\n{res_pesde.stdout}\nstderr\n{res_pesde.stderr}`) - print(`-- Wally out:\nstdout\n{res_wally.stdout}\nstderr\n{res_wally.stderr}`) - - local took = round_to((os.clock() - begin) * 1_000, 2) + local took = shared.round_to((os.clock() - begin) * 1_000, 2) if not dry then print(`๐Ÿš€ Published packages {stdio.style("dim")}(took {took}ms){stdio.style("reset")}`) else print(`๐Ÿ“ฆ Packaged packages {stdio.style("dim")}(took {took}ms){stdio.style("reset")}`) end - print({ - pesde = `{scopes.pesde}@{version}`, - wally = `{scopes.wally}@{version}`, - }) + + if scopes.pesde and scopes.wally then + print({ + pesde = `{scopes.pesde}@{version}`, + wally = `{scopes.wally}@{version}`, + }) + elseif scopes.pesde then + print({ + pesde = `{scopes.pesde}@{version}`, + }) + elseif scopes.wally then + print({ + wally = `{scopes.wally}@{version}`, + }) + end return result(true, nil) end diff --git a/src/shared.luau b/src/shared.luau new file mode 100644 index 0000000..5194e9a --- /dev/null +++ b/src/shared.luau @@ -0,0 +1,47 @@ +--!strict +local datetime = require("@lune/datetime") +local fs = require("@lune/fs") +local net = require("@lune/net") + +local result = require("./util/result") + +--- Fetches the given file raw from the jecs github +local function fetch_raw(file: string): result.Identity + local res = net.request(`https://raw.githubusercontent.com/Ukendio/jecs/refs/heads/main/{file}`) + if not res.ok then + return result(false, `Not ok: {res.statusMessage} @ {res.statusCode}`) + end + return result(true, res.body) +end + +local function save_if_diff(filepath: string, contents: string): result.Identity + if not fs.metadata(filepath).exists then + fs.writeFile(filepath, contents) + return result(true, nil) + end + + local existing = fs.readFile(filepath) + if existing == contents then + return result(false, "Contents are the same.") + end + + fs.writeFile(filepath, contents) + return result(true, nil) +end + +local function round_to(n: number, places: number) + local x = 10 ^ (places or 0) + return math.round(n * x) / x +end + +-- Returns an ISO 8601 date (YYYYmmddThhmmssZ) +local function iso_date_light(date: datetime.DateTime): string + return date:formatUniversalTime("%Y%m%dT%H%M%SZ") +end + +return { + fetch_raw = fetch_raw, + save_if_diff = save_if_diff, + round_to = round_to, + iso_date_light = iso_date_light, +} diff --git a/src/sync.luau b/src/sync.luau index 1fd4252..2407cb4 100644 --- a/src/sync.luau +++ b/src/sync.luau @@ -1,42 +1,17 @@ --!strict +local datetime = require("@lune/datetime") local fs = require("@lune/fs") -local net = require("@lune/net") +local serde = require("@lune/serde") local stdio = require("@lune/stdio") local progress_bar = require("./util/progress") local result = require("./util/result") - ---- Fetches the given file raw from the jecs github -local function fetch_raw(file: string): result.Identity - local res = net.request(`https://raw.githubusercontent.com/Ukendio/jecs/refs/heads/main/{file}`) - if not res.ok then - return result(false, `Not ok: {res.statusMessage} @ {res.statusCode}`) - end - return result(true, res.body) -end - -local function save_if_diff(filepath: string, contents: string): result.Identity - if not fs.metadata(filepath).exists then - fs.writeFile(filepath, contents) - return result(true, nil) - end - - local existing = fs.readFile(filepath) - if existing == contents then - return result(false, "Contents are the same.") - end - - fs.writeFile(filepath, contents) - return result(true, nil) -end - -local function round_to(n: number, places: number) - local x = 10 ^ (places or 0) - return math.round(n * x) / x -end +local shared = require("./shared") +local types = require("./types") --- Synchronizes the required files from the jecs main branch. local function sync(to: string): result.Identity + local now = datetime.now() local begin = os.clock() local progress = progress_bar @@ -44,13 +19,15 @@ local function sync(to: string): result.Identity :withStage("init", "Initializing") :withStage("fetch", "Fetching latest files") :withStage("save", "Saving files") - progress:start() + :withStage("version", "Pulling latest jecs version") + :withStage("metadata", "Writing build metadata") + progress:start() -- init if not fs.metadata(to).exists then fs.writeDir(to) end - progress:nextStage() + progress:nextStage() -- fetch local includes = { "jecs.luau", @@ -58,13 +35,12 @@ local function sync(to: string): result.Identity "CHANGELOG.md", "LICENSE", ".luaurc", - "default.project.json", } local sources = {} for _, file in includes do - local contents = fetch_raw(file) + local contents = shared.fetch_raw(file) if not contents.ok then progress:stop() stdio.ewrite(`๐Ÿ”ฅ Couldn't get the latest source for {file}:\n{contents.err}\n`) @@ -74,12 +50,12 @@ local function sync(to: string): result.Identity sources[file] = contents.val end - progress:nextStage() + progress:nextStage() -- save local sources_modified = {} local any_changed = false for file, contents in sources do - local res = save_if_diff(`{to}/{file}`, contents) + local res = shared.save_if_diff(`{to}/{file}`, contents) if res.ok then any_changed = true table.insert(sources_modified, file) @@ -88,7 +64,7 @@ local function sync(to: string): result.Identity if not any_changed then progress:stop() - local took = round_to((os.clock() - begin) * 1_000, 2) + local took = shared.round_to((os.clock() - begin) * 1_000, 2) print( `๐Ÿ•› Finished synchronizing, no changes since latest source {stdio.style("dim")}(took {took}ms){stdio.style( "reset" @@ -97,9 +73,39 @@ local function sync(to: string): result.Identity return result(true, false) end - progress:stop() + local project_json = { + name = "jecs-nightly", + tree = { + ["$path"] = "jecs.luau", + }, + } + local encoded_project_json = serde.encode("json", project_json) + fs.writeFile(`{to}/default.project.json`, encoded_project_json) - local took = round_to((os.clock() - begin) * 1_000, 2) + progress:nextStage() -- version + + local wally_contents = shared.fetch_raw("wally.toml") + if not wally_contents.ok then + progress:stop() + stdio.ewrite(`๐Ÿ”ฅ Couldn't get the jecs wally manifest:\n{wally_contents.err}\n`) + return result(false, `Couldn't get the jecs wally manifest:\n{wally_contents.err}`) + end + + local parsed: types.WallyManifest = serde.decode("toml", wally_contents.val) + + progress:nextStage() -- metadata + + local version = `{parsed.package.version}-nightly.{shared.iso_date_light(now)}` + local metadata = { + version = version, + modified = sources_modified, + } + local encoded_metadata = serde.encode("toml", metadata) + fs.writeFile(`{to}/build.txt`, encoded_metadata) + + progress:stop() -- finish + + local took = shared.round_to((os.clock() - begin) * 1_000, 2) print(`๐Ÿชจ Finished synchronizing {stdio.style("dim")}(took {took}ms){stdio.style("reset")}`) print(`Changed files:`) print(sources_modified) diff --git a/src/test.luau b/src/test.luau new file mode 100644 index 0000000..c12686d --- /dev/null +++ b/src/test.luau @@ -0,0 +1,105 @@ +--!strict +local datetime = require("@lune/datetime") +local fs = require("@lune/fs") +local process = require("@lune/process") +local serde = require("@lune/serde") +local stdio = require("@lune/stdio") + +local progress_bar = require("./util/progress") +local result = require("./util/result") +local shared = require("./shared") + +local function test(origin: string): result.Identity + local now = datetime.now() + local begin = os.clock() + + local progress = progress_bar + .new() + :withStage("init", "Initializing") + :withStage("fetch", "Fetching latest tests") + :withStage("test", "Running tests") + :withStage("cleanup", "Cleaning up") + :withStage("metadata", "Writing test metadata") + progress:start() -- init + + if not fs.metadata(origin).exists then + progress:stop() + stdio.ewrite(`๐Ÿ”ฅ {origin} is not a valid directory which exists.\n`) + return result(false, `{origin} is not a valid directory which exists.`) + end + + if not fs.metadata(`{origin}/jecs.luau`).exists then + progress:stop() + stdio.ewrite(`๐Ÿ”ฅ {origin}/jecs.luau is not a vald file which exists.\n`) + return result(false, `{origin}/jecs.luau is not a vald file which exists.`) + end + + if fs.metadata(`{origin}/test`).exists then + fs.removeDir(`{origin}/test`) + end + fs.writeDir(`{origin}/test`) + + progress:nextStage() -- fetch + + do + local contents = shared.fetch_raw("test/testkit.luau") + if not contents.ok then + progress:stop() + stdio.ewrite(`๐Ÿ”ฅ Couldn't get the latest source for test/testkit.luau:\n{contents.err}\n`) + return result(false, "Couldn't get the latest source for test/testkit.luau.") + end + fs.writeFile(`{origin}/test/testkit.luau`, contents.val) + end + do + local contents = shared.fetch_raw("test/tests.luau") + if not contents.ok then + progress:stop() + stdio.ewrite(`๐Ÿ”ฅ Couldn't get the latest source for test/tests.luau:\n{contents.err}\n`) + return result(false, "Couldn't get the latest source for test/tests.luau.") + end + fs.writeFile(`{origin}/test/tests.luau`, contents.val) + end + + progress:nextStage() -- test + + local test_result = process.spawn("luau", { "test/tests.luau" }, { cwd = origin }) + if not test_result.ok then + progress:stop() + stdio.ewrite(`๐Ÿ”ฅ Tests failed to run:\n{test_result.stderr}\n`) + return result(false, `Tests failed to run.`) + end + + local passed = true + if not string.find(test_result.stdout, "0 fails") then + passed = false + end + + progress:nextStage() -- cleanup + + fs.removeDir(`{origin}/test`) + + progress:nextStage() -- metadata + + local metadata = { + timestamp = shared.iso_date_light(now), + passed = passed, + } + local encoded_metadata = serde.encode("toml", metadata) + fs.writeFile(`{origin}/test.txt`, encoded_metadata) + fs.writeFile(`{origin}/test_fulllog.txt`, test_result.stdout) + + progress:stop() -- finish + + print(test_result.stdout) + + local passed_txt = if passed + then `{stdio.style("bold")}{stdio.color("green")}passed{stdio.color("reset")}{stdio.style("reset")}` + else `{stdio.style("bold")}{stdio.color("red")}failed{stdio.color("reset")}{stdio.style("reset")}` + + local took = shared.round_to((os.clock() - begin) * 1_000, 2) + print(`๐Ÿงช Finished testing, {passed_txt} {stdio.style("dim")}(took {took}ms){stdio.style("reset")}`) + + return result(true, true) +end + +return test diff --git a/src/util/frkcli.luau b/src/util/frkcli.luau new file mode 100644 index 0000000..d306791 --- /dev/null +++ b/src/util/frkcli.luau @@ -0,0 +1,586 @@ +--!nolint LocalShadow +-- https://raw.githubusercontent.com/itsfrank/frkcli/refs/heads/main/src/frkcli.luau + +local process = require("@lune/process") + +local M = {} + +type ArgKind = "POSITIONAL" | "FLAG" | "OPTION" +type ArgOptions = { + help: string?, + aliases: { string }?, + default: string?, -- if this is not nil, the arg will be optional +} + +-- this is what is stored, we guarante non nullity when args are added so this types makes Luau feel better +type ArgOptionsSafe = { + help: string, + aliases: { string }, + default: string?, +} + +type ArgData = { + name: string, + kind: ArgKind, + options: ArgOptionsSafe, +} + +type ParseResult = { + values: { [string]: string }, + flags: { [string]: boolean }, + fwd_args: { string }, -- all args after `--` +} + +type SubcommandParseResult = { + command: string, + result: ParseResult, +} + +local DEFAULT_OPTIONS: ArgOptionsSafe = { + help = "", + default = nil, + aliases = {}, +} + +local DEFAULT_helpset = { + ["-h"] = true, + ["--help"] = true, +} + +local function validate_key_or_error(key: string) + if (key:sub(1, 2) == "--" or key:sub(1, 1) == "-") and not key:find(" ") then + return + end + error(`arg key {key} is invalid. Keys must start with either '-' or '--' and may not contain spaces`) +end + +local function validate_subcommand_name_or_error(name: string) + if name:sub(1, 2) == "--" or name:sub(1, 1) == "-" or name:find(" ") then + error(`subcommand name '{name}' is invalid. Name must not start with '-' or '--' and may not contain spaces`) + end +end + +type HelpSection = { title: string, lines: { { string } | string }? } +local function make_help(sections: { HelpSection }, indent: number?): string + local function align_cols(rows: { { string } }, sep: string?): { { string } } + local sep = if sep == nil then " " else sep + local max_col_lengths: { number } = {} + for _, row in rows do + for i, s in row do + if max_col_lengths[i] == nil or s:len() > max_col_lengths[i] then + max_col_lengths[i] = s:len() + end + end + end + local aligned_rows: { { string } } = {} + for _, row in rows do + local line: { string } = {} + for i, col in row do + table.insert(line, col) + if i < #row then + local spacing = sep + local diff = max_col_lengths[i] - col:len() + if diff > 0 then + spacing = string.rep(" ", diff) .. sep + end + table.insert(line, spacing) + end + end + table.insert(aligned_rows, line) + end + return aligned_rows + end + + local function append_list(dest: { T }, src: { T }) + for _, v in src do + table.insert(dest, v) + end + end + + local ind = " " + if indent ~= nil then + ind = string.rep(" ", indent) + end + + local help_lines: { { string } } = {} + + for _, s in sections do + table.insert(help_lines, { s.title }) + if s.lines ~= nil then + local section_lines: { { string } } = {} + for _, l in s.lines do + if typeof(l) == "string" then + table.insert(section_lines, { ind, l }) + else + table.insert(section_lines, { ind, table.unpack(l) }) + end + end + append_list(help_lines, align_cols(section_lines)) + end + table.insert(help_lines, {}) + end + + -- remove last empty line + table.remove(help_lines) + + local help_text = "" + for _, line in help_lines do + help_text ..= table.concat(line, " ") .. "\n" + end + return help_text +end +-- subcommand cli, must be first positional arg +function M.new_subcommands(name, description: string?, helpkeys: { string }?) + -- args that trigger print help + abort + local helpset: { [string]: boolean } = DEFAULT_helpset + if helpkeys ~= nil then + helpset = {} + for _, k in helpkeys do + helpset[k] = true + end + end + + local cli = {} + cli.name = name + cli.description = description + cli._helpkeys = helpkeys + cli._helpset = helpset + cli._subcommands = {} + + function cli:add_subcommand(name: string, description: string?) + validate_subcommand_name_or_error(name) + local sc = M.new(name, description, cli._helpkeys) + cli._subcommands[name] = sc + return sc + end + + function cli:parse(args: { string }): (SubcommandParseResult?, string?) + if #args == 0 then + return nil, "insufficient arguments, required at least 1, got 0" + end + local command = args[1] + assert(command ~= nil) + + -- asking for help? + if cli._helpset[command] ~= nil then + print(cli:help()) + process.exit(0) + end + + local subcommand = cli._subcommands[command] + if subcommand == nil then + return nil, `'{command} is not a valid subcommand` + end + + local sub_args = { table.unpack(args, 2) } + local res, err = subcommand:parse(sub_args) + if err ~= nil then + return nil, err + end + assert(res ~= nil) + + return { + command = command, + result = res, + }, nil + end + + function cli:help(indent: number?): string + local help_sections: { HelpSection } = {} + + table.insert(help_sections, { title = `usage: {cli.name} ` }) + + if cli.description ~= nil then + table.insert(help_sections, { + title = "description:", + lines = { { cli.description } }, + }) + end + + local subcommand_lines = {} + for _, c in cli._subcommands do + local desc = if c.description then c.description else "" + table.insert(subcommand_lines, { c.name, desc }) + end + + table.insert(help_sections, { + title = "commands:", + lines = subcommand_lines, + }) + + return make_help(help_sections, indent) + end + + return cli +end + +function M.new(name: string, description: string?, helpkeys: { string }?) + -- args that trigger print help + abort + local helpset: { [string]: boolean } = DEFAULT_helpset + if helpkeys ~= nil then + helpset = {} + for _, k in helpkeys do + helpset[k] = true + end + end + + -- I'm fairly certain this is not the right pattern for making objects, but it results in single definition + great LSP so I'm happy + local cli = {} + cli.name = name + cli.description = description + cli._helpset = helpset + cli._positionals = {} :: { ArgData } + cli._flags = {} :: { ArgData } + cli._options = {} :: { ArgData } + cli._argdata_set = {} :: { [string]: ArgData } + cli._required_list = {} :: { string } + cli._lookup = {} :: { [string]: ArgData } + cli._required_positional_count = 0 + cli._default_result = { values = {}, flags = {} } :: ParseResult + + local function add_arg_lookups(keys: { string }, arg_data: ArgData) + for _, k in keys do + if cli._helpset[k] ~= nil then + error(`key '{k}' is already used as a help key.`) + end + validate_key_or_error(k) + if cli._lookup[k] ~= nil then + error(`key '{k}' already exists.`) + end + cli._lookup[k] = arg_data + end + end + + local function add_name_error_on_duplicate(name: string, data: ArgData) + if cli._argdata_set[name] ~= nil then + error(`arg with name {name} already exists`) + end + cli._argdata_set[name] = data + end + + local function make_safe_options(options: ArgOptions?): ArgOptionsSafe + if options == nil then + return DEFAULT_OPTIONS + end + assert(options) + + options.aliases = if options.aliases == nil then {} else options.aliases + + return options :: ArgOptionsSafe + end + + -- return is guaranteed to start with at least one '-', if none, adds '--' + local function to_lookup_name(name: string): string + if name:sub(1, 1) == "-" then + return name + end + return "--" .. name + end + + -- return is guaranted to not start with '-' + local function to_result_name(name: string): string + while name:sub(1, 1) == "-" do + name = name:sub(2, -1) + end + return name + end + + function cli:add_positional(name: string, options: ArgOptions?) + if name:sub(1, 1) == "-" then + error(`invalid arg name '{name}', positional cannot start with dashes`) + end + + local options = make_safe_options(options) + + local arg_data: ArgData = { + name = name, + kind = "POSITIONAL", + options = options, + } + add_name_error_on_duplicate(name, arg_data) + if options.default == nil then + -- check positional required ordering, can't have a required pos after an optional one + local last = cli._positionals[#cli._positionals] + if last ~= nil and last.options.default ~= nil then + error( + `{name} is required, but {last.name} is optional. Cannot have required positional after optional positional` + ) + end + + table.insert(cli._required_list, name) + else + cli._default_result.values[name] = options.default + end + table.insert(cli._positionals, arg_data) + end + + function cli:add_flag(name: string, options: ArgOptions?) + if options and options.default ~= nil then + -- these are called constants :) + error(`flag {name} has non nil default, default value is not supported for flags`) + end + + local options = make_safe_options(options) + + local lookup_name = to_lookup_name(name) + name = to_result_name(name) + for i, v in options.aliases do + options.aliases[i] = to_lookup_name(v) + end + + local arg_data: ArgData = { + name = name, + kind = "FLAG", + options = options, + } + add_name_error_on_duplicate(name, arg_data) + add_arg_lookups({ lookup_name, table.unpack(options.aliases) }, arg_data) + + if options.default ~= nil then + error(`flag {name} has non nil default value, defaults are not supported for flag args`) + end + + cli._default_result.flags[name] = false + table.insert(cli._flags, arg_data) + end + + function cli:add_option(name: string, options: ArgOptions?) + local options = make_safe_options(options) + + local lookup_name = to_lookup_name(name) + name = to_result_name(name) + for i, v in options.aliases do + options.aliases[i] = to_lookup_name(v) + end + + local arg_data: ArgData = { + name = name, + kind = "OPTION", + options = options, + } + add_name_error_on_duplicate(name, arg_data) + add_arg_lookups({ lookup_name, table.unpack(options.aliases) }, arg_data) + if options.default == nil then + table.insert(cli._required_list, name) + else + if options.default == nil then + error(`optional arg {name} must have a default value`) + end + assert(options.default ~= nil) + cli._default_result.values[name] = options.default + end + table.insert(cli._options, arg_data) + end + + -- return: data, err - where data is a table, and err is a string + -- errors early, first encountered error ends the parse + -- if a help key is found will print help text and exit the process + function cli:parse(args: { string }): (ParseResult?, string?) + local parsed: ParseResult = { values = {}, flags = {}, fwd_args = {} } :: ParseResult + + local positional_idx = 1 + local skip_next = false -- used for options + for i, arg in args do + -- asking for help? + if cli._helpset[arg] ~= nil then + print(cli:help()) + process.exit(0) + end + + if skip_next then + skip_next = false + continue + end + + -- rest of args should be fwd args + if arg == "--" then + parsed.fwd_args = { table.unpack(args, i + 1) } + break + end + + local has_dash = arg:sub(1, 1) == "-" + local arg_data: ArgData? = nil + local value: string? = nil + + if not has_dash then -- positional + if positional_idx > #cli._positionals then + return nil, `too many positional arguments, expected {#cli._positionals}.` + end + arg_data = cli._positionals[positional_idx] + positional_idx += 1 + value = arg + else -- flag or option + arg_data = cli._lookup[arg] + end + + if arg_data == nil then + return nil, `unknown flag or option: '{arg}'` + end + assert(arg_data ~= nil) + + -- handle flags early + if arg_data.kind == "FLAG" then + parsed.flags[arg_data.name] = true + continue + end + + -- get option value + if arg_data.kind == "OPTION" then + if i < #args then + value = args[i + 1] + skip_next = true + end + end + + -- resolve positionals and options + if value == nil then + return nil, `no value provided for option '{arg}'` + end + assert(value ~= nil) + + parsed.values[arg_data.name] = value + end + + -- check all required options are provided + for _, name in cli._required_list do + if parsed.values[name] == nil and parsed.flags[name] == nil then + local kind = cli._argdata_set[name].kind:lower() + return nil, `required {kind} arg '{name}' was not found` + end + end + + -- apply defaults for missing optional + for k, v in cli._default_result.flags do + if parsed.flags[k] == nil then + parsed.flags[k] = v + end + end + for k, v in cli._default_result.values do + if parsed.values[k] == nil then + parsed.values[k] = v + end + end + + return parsed, nil + end + + function cli:help(indent: number?): string + local function align_cols(rows: { { string } }, sep: string?): { { string } } + local sep = if sep == nil then " " else sep + local max_col_lengths: { number } = {} + for _, row in rows do + for i, s in row do + if max_col_lengths[i] == nil or s:len() > max_col_lengths[i] then + max_col_lengths[i] = s:len() + end + end + end + local aligned_rows: { { string } } = {} + for _, row in rows do + local line: { string } = {} + for i, col in row do + table.insert(line, col) + if i < #row then + local spacing = sep + local diff = max_col_lengths[i] - col:len() + if diff > 0 then + spacing = string.rep(" ", diff) .. sep + end + table.insert(line, spacing) + end + end + table.insert(aligned_rows, line) + end + return aligned_rows + end + + local help_sections: { HelpSection } = {} + + -- usage + local usage = `usage: {cli.name}` + if #cli._flags > 0 or #cli._options > 0 then + usage ..= " [options]" + end + for _, arg in cli._positionals do + if arg.options.default == nil then + usage ..= ` <{arg.name}>` + else + usage ..= ` [{arg.name}]` + end + end + + table.insert(help_sections, { title = usage }) + + -- description + if cli.description ~= nil then + table.insert(help_sections, { title = "description:", lines = { cli.description } }) + end + + local function make_arg_line(arg: ArgData): { string } + local keys = arg.name + if arg.kind ~= "POSITIONAL" then + keys = `--{arg.name}` + for _, a in arg.options.aliases do + keys ..= `, {a}` + end + end + + local reqdef = "" + -- optional & default + if arg.kind ~= "FLAG" then + if arg.options.default == nil then + reqdef ..= "[required]" + else + assert(arg.options.default) + reqdef ..= `[default: '{arg.options.default}']` + end + end + + local help = "" + if arg.options.help then + if help ~= "" then + help ..= " " + end + help ..= arg.options.help + end + + return { keys, reqdef, help } + end + + -- positionals + if #cli._positionals > 0 then + local positional_lines: { { string } } = {} + for _, arg in cli._positionals do + table.insert(positional_lines, make_arg_line(arg)) + end + table.insert(help_sections, { title = "positional arguments:", lines = positional_lines }) + end + + -- flags + if #cli._flags > 0 then + local flag_lines: { { string } } = {} + for _, arg in cli._flags do + table.insert(flag_lines, make_arg_line(arg)) + end + table.insert(help_sections, { title = "flags:", lines = flag_lines }) + end + + -- options + if #cli._options > 0 then + local option_lines: { { string } } = {} + for _, arg in cli._options do + table.insert(option_lines, make_arg_line(arg)) + end + table.insert(help_sections, { title = "options:", lines = option_lines }) + end + + return make_help(help_sections, indent) + end + + return cli +end + +export type Cli = typeof(M.new("")) +export type CliSubcommands = typeof(M.new_subcommands("")) + +return M