diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 0000000..718572b --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,7 @@ +version: 2 + +updates: + - package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "weekly" diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..c91cf3c --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,24 @@ +name: CI + +on: + push: + branches: + - main + pull_request: + +jobs: + check: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + with: + ref: ${{ github.event.pull_request.head.ref }} + + - uses: creyD/prettier_action@v4.3 + with: + prettier_options: --write . + + - uses: EndBug/add-and-commit@v9 + with: + message: "chore: lint" + default_author: github_actions diff --git a/.github/workflows/deploy.yml b/.github/workflows/deploy.yml new file mode 100644 index 0000000..ffeffcd --- /dev/null +++ b/.github/workflows/deploy.yml @@ -0,0 +1,34 @@ +name: Deploy + +on: + push: + branches: + - main + workflow_dispatch: + +concurrency: + group: pages + cancel-in-progress: false + +jobs: + deploy: + environment: + name: github-pages + url: ${{ steps.deployment.outputs.page_url }} + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - run: | + mkdir -p out + cp -r social/dwn/protocols out/protocols + cp -r social/dwn/schemas out/schemas + - uses: s0/git-publish-subdir-action@develop + env: + BRANCH: gh-pages + COMMIT_EMAIL: 41898282+github-actions[bot]@users.noreply.github.com + COMMIT_NAME: github-actions[bot] + FOLDER: out + GITHUB_TOKEN: ${{ github.token }} + REPO: self + SQUASH_HISTORY: true + TARGET_DIR: v0 diff --git a/CNAME b/CNAME new file mode 100644 index 0000000..4ef3a35 --- /dev/null +++ b/CNAME @@ -0,0 +1 @@ +wired-protocol.org diff --git a/LICENSE-APACHE b/LICENSE-APACHE new file mode 100644 index 0000000..d9a10c0 --- /dev/null +++ b/LICENSE-APACHE @@ -0,0 +1,176 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS diff --git a/LICENSE-MIT b/LICENSE-MIT new file mode 100644 index 0000000..3188970 --- /dev/null +++ b/LICENSE-MIT @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2023 UNAVI + +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/README.md b/README.md new file mode 100644 index 0000000..bb599b9 --- /dev/null +++ b/README.md @@ -0,0 +1,23 @@ +# The Wired + +The Wired is an open protocol for the metaverse. +It is a collection of open standards focused on interactive 3D environments, +self-sovereign identity, and seamless interoperability. + +## Protocol + +- [Social](./social) +- [Spatial](./spatial) + +## Motivation + +Today's VR ecosystem is fragmented by proprietary platforms that confine users and developers to isolated, controlled environments. +Each platform enforces its own set of rules and restrictions, often under leadership +that makes arbitrary decisions with no recourse for the people who live within the virtual spaces they govern. + +With VR our digital rights matter more than ever. +The Wired acknowledges this importance by fostering a decentralized environment where anyone +can contribute and shape the future of the metaverse. + +By embracing open source principles, The Wired doesn't just safeguard against abuse; +it unleashes a wave of creativity and collaboration not possible within walled gardens. diff --git a/_config.yml b/_config.yml new file mode 100644 index 0000000..10f50fb --- /dev/null +++ b/_config.yml @@ -0,0 +1,3 @@ +remote_theme: pages-themes/cayman@v0.2.0 +plugins: + - jekyll-remote-theme diff --git a/social/README.md b/social/README.md new file mode 100644 index 0000000..e9066a9 --- /dev/null +++ b/social/README.md @@ -0,0 +1,49 @@ +# Social Protocol + +The Wired's social protocol provides an application-agnostic self-hostable base layer for user identity and data storage. + +## Decentralized Identifiers (DIDs) + +[Decentralized Identifiers](https://en.wikipedia.org/wiki/Decentralized_identifier) (DIDs) provide a +globally unique namespace for both users and content within The Wired. + +DIDs are a generic format for addressing content that can be extended to support any protocol. +For example, [`did:web`](https://w3c-ccg.github.io/did-method-web/) can be used to address content using a traditional web URL, +while [`did:ipid`](https://did-ipid.github.io/ipid-did-method/) can be used to address content over [IPFS](https://docs.ipfs.tech/). + +A user's DID resolves to a document containing information such as their name, +cryptographic keys that can be used to verify their identity, servers to contact them at, and anything else they want to store. + +### DID Host + +A DID host is a server that remotely hosts a DID and provides convenient access to it. +For example, the server may use a method like `did:web` and host your DID document at a web domain. + +You can log in to the server with a typical login method - such as a username and password or OAuth connection - then receive +cryptographic keys which can be used to verify your ownership of the hosted DID to other parties. +This allows you to use DIDs with the convienance of traditional app logins. + +Running your own DID host server is a great first step towards self-sovereignity within The Wired. + +## Decentralized Web Nodes (DWNs) + +[Decentralized Web Nodes](https://identity.foundation/decentralized-web-node/spec/) (DWNs) are the data layer of The Wired. +DWNs build upon DIDs to store user data and provide an API for others to interact with it. +These interactions could be a simple read to view a document hosted by the user, or more complex writes to add +comments or send encrypted messages. + +DWNs are built using [CRDT](https://en.wikipedia.org/wiki/Conflict-free_replicated_data_type)s and will sync their data +with other DWNs. +This allows you, for example, to make use of a public cloud-hosted DWN, while at the same time running your own +local DWN to keep a backup of your data. + +## Moderation + +**The Wired is decentralized.** + +There is no central registry of users - a user cannot be "banned" from The Wired. +Same with any content within The Wired - files can be hosted and shared by anyone. +That's just the internet. + +However specific services (websites, worlds, APIs) can and WILL moderate as necessary - +but like the Web, users will always have the option of moving to a different service they are not banned from (or running their own). diff --git a/social/dwn/protocols/world-host.json b/social/dwn/protocols/world-host.json new file mode 100644 index 0000000..74c82f3 --- /dev/null +++ b/social/dwn/protocols/world-host.json @@ -0,0 +1,52 @@ +{ + "protocol": "https://wired-protocol.org/v0/protocols/world-host.json", + "published": true, + "types": { + "instance": { + "schema": "https://wired-protocol.org/v0/schemas/instance.json", + "dataFormat": ["application/json"] + }, + "instance-info": { + "schema": "https://wired-protocol.org/v0/schemas/instance-info.json", + "dataFormat": ["application/json"] + }, + "connect-url": { + "dataFormat": ["text/plain"] + } + }, + "structure": { + "connect-url": { + "$actions": [ + { + "who": "anyone", + "can": ["read"] + }, + { + "who": "recipient", + "can": ["write"] + } + ] + }, + "instance": { + "$actions": [ + { + "who": "anyone", + "can": ["read", "write"] + } + ], + "instance-info": { + "$actions": [ + { + "who": "anyone", + "can": ["read"] + }, + { + "who": "recipient", + "of": "instance", + "can": ["write"] + } + ] + } + } + } +} diff --git a/social/dwn/schemas/home.json b/social/dwn/schemas/home.json new file mode 100644 index 0000000..b412a23 --- /dev/null +++ b/social/dwn/schemas/home.json @@ -0,0 +1,20 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/home.json", + "type": "object", + "properties": { + "world": { + "type": "object", + "properties": { + "did": { + "type": "string" + }, + "record_id": { + "type": "string" + } + }, + "required": ["did", "record_id"] + } + }, + "required": ["world"] +} diff --git a/social/dwn/schemas/instance-info.json b/social/dwn/schemas/instance-info.json new file mode 100644 index 0000000..98f1eff --- /dev/null +++ b/social/dwn/schemas/instance-info.json @@ -0,0 +1,16 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/instance-info.json", + "type": "object", + "properties": { + "numPlayers": { + "type": "number" + }, + "maxPlayers": { + "type": "number" + }, + "extras": { + "type": "object" + } + } +} diff --git a/social/dwn/schemas/instance.json b/social/dwn/schemas/instance.json new file mode 100644 index 0000000..c4654eb --- /dev/null +++ b/social/dwn/schemas/instance.json @@ -0,0 +1,20 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/instance.json", + "type": "object", + "properties": { + "world": { + "type": "object", + "properties": { + "did": { + "type": "string" + }, + "record_id": { + "type": "string" + } + }, + "required": ["did", "record_id"] + } + }, + "required": ["world"] +} diff --git a/social/dwn/schemas/world.json b/social/dwn/schemas/world.json new file mode 100644 index 0000000..fd8fd5e --- /dev/null +++ b/social/dwn/schemas/world.json @@ -0,0 +1,26 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/world.json", + "type": "object", + "properties": { + "name": { + "type": "string" + }, + "description": { + "type": "string" + }, + "tags": { + "type": "array", + "items": { + "type": "string" + } + }, + "host": { + "type": "string", + "description": "DID of the world host." + }, + "extras": { + "type": "object" + } + } +} diff --git a/spatial/README.md b/spatial/README.md new file mode 100644 index 0000000..87f0929 --- /dev/null +++ b/spatial/README.md @@ -0,0 +1,41 @@ +# Spatial Protocol + +The Wired's spatial protocol is concerned with running interactive user-created content in a dynamic 3D multiplayer environment. + +## Scenes (glXF) + +The Wired uses the [glXF](https://github.com/KhronosGroup/glXF) file format for describing scenes. +glXF is a variant of [glTF](https://github.com/KhronosGroup/glTF), focused on the composition of multiple glTF (or other glXF) assets. +glTF is a well supported format for 3D models, and its extensible nature makes it a great fit for The Wired. + +## Scripts (WASM) + +The Wired uses [WebAssembly](https://webassembly.org/) (WASM) as a cross-platform compilation target and sandboxed execution environment for user scripting. +Scripts use the [component model](https://github.com/WebAssembly/component-model) to interact with a set of +[WIT](https://github.com/WebAssembly/component-model/blob/main/design/mvp/WIT.md) interfaces defined by The Wired. +Host environments can then implement these interfaces, allowing scripts to interact with the scene in a controlled manner. + +### Input + +TODO. + +The Wired will handle user input through a system inspired by Stardust's [SUIS](https://docs.rs/stardust-xr-fusion/latest/stardust_xr_fusion/input/index.html). + +## Worlds + +Worlds are created at a user's [DWN](../social/#decentralized-web-nodes-dwns), following to the [world schema](../social/dwn/schemas/world.json). +This countains metadata about the world such as a name and description, as well as the world scene. + +### Instances + +To join a world you must join an **instance** of the world. +An instance is a multiplayer room running on a host server. +This server acts as a relay for communication between players within the world. +Instances are created at the host server's DWN. + +Additionally the world host DWN acts as a location for world discovery, where you can, +for example, query for active instances to join. + +## Networking + +Networking within an instance follows a [Cap'n Proto](https://capnproto.org/) schema over [WebTransport](https://developer.mozilla.org/en-US/docs/Web/API/WebTransport). diff --git a/spatial/capnp/datagram.capnp b/spatial/capnp/datagram.capnp new file mode 100644 index 0000000..9608fe1 --- /dev/null +++ b/spatial/capnp/datagram.capnp @@ -0,0 +1,25 @@ +@0x875d2f2c381d4863; + +struct Vec3 { + x @0 :Float32; + y @1 :Float32; + z @2 :Float32; +} + +struct Vec4 { + x @0 :Float32; + y @1 :Float32; + z @2 :Float32; + w @3 :Float32; +} + +struct PublishTransform { + translation @0 :Vec3; + rotation @1 :Vec4; +} + +struct ReceiveTransform { + playerId @0 :UInt16; + translation @1 :Vec3; + rotation @2 :Vec4; +} diff --git a/spatial/capnp/world_server.capnp b/spatial/capnp/world_server.capnp new file mode 100644 index 0000000..2c7e4dc --- /dev/null +++ b/spatial/capnp/world_server.capnp @@ -0,0 +1,39 @@ +@0xae09eca2c525e50a; + +struct Option(T) { + union { + some @0 :T; + none @1 :Void; + } +} + +struct Result(T) { + union { + result @0 :T; + error @1 :Text; + } +} + +struct Success { + union { + success @0 :Void; + error @1 :Text; + } +} + +interface WorldServer { + join @0 (recordId :Text) -> (success :Success); + leave @1 (recordId :Text); + + # Tickrate of the server, in seconds. + # Sending updates faster than this will be detrimental. + tickrate @2 () -> (tickrate :Float32); + + player @3 (id :UInt16) -> (player :Option(Player)); + players @4 () -> (players :List(Player)); + + struct Player { + id @0 :UInt16; + did @1 :Text; + } +} diff --git a/spatial/wit/wired-input/deps/wired-math b/spatial/wit/wired-input/deps/wired-math new file mode 120000 index 0000000..967f496 --- /dev/null +++ b/spatial/wit/wired-input/deps/wired-math @@ -0,0 +1 @@ +../../wired-math/ \ No newline at end of file diff --git a/spatial/wit/wired-input/types.wit b/spatial/wit/wired-input/types.wit new file mode 100644 index 0000000..4414325 --- /dev/null +++ b/spatial/wit/wired-input/types.wit @@ -0,0 +1,66 @@ +interface types { + use wired:math/types.{vec3, quat}; + + enum hand-side { + left, + right, + } + + record joint { + translation: vec3, + rotation: quat, + radius: f32, + } + + record finger { + tip: joint, + distal: joint, + proximal: joint, + metacarpal: joint, + } + + // Hand tracking data. + record hand { + side: hand-side, + + thumb: finger, + index: finger, + middle: finger, + ring: finger, + little: finger, + + palm: joint, + wrist: joint, + elbow: option, + } + + // A line with an origin and a direction. + record ray { + origin: vec3, + orientation: quat, + } + + // A single point of interaction, such as the tip of a stylus. + record tip { + origin: vec3, + orientation: quat, + radius: f32, + } + + variant input-type { + hand(hand), + ray(ray), + tip(tip), + } + + record input-event { + // Unique id for the event. + id: u64, + // Spatial input data. + input: input-type, + // Distance from the input method to the handler. + distance: f32, + // How many handlers received the event before this one. + order: u32, + } +} diff --git a/spatial/wit/wired-input/world.wit b/spatial/wit/wired-input/world.wit new file mode 100644 index 0000000..313bd2d --- /dev/null +++ b/spatial/wit/wired-input/world.wit @@ -0,0 +1,18 @@ +package wired:input; + +world host { + import handler; + import types; +} + +interface handler { + use types.{input-event}; + + resource input-handler { + constructor(); + + // Handle the next recieved input event. + // Events only last for one tick. + handle-input: func() -> option; + } +} diff --git a/spatial/wit/wired-log/world.wit b/spatial/wit/wired-log/world.wit new file mode 100644 index 0000000..64b87e1 --- /dev/null +++ b/spatial/wit/wired-log/world.wit @@ -0,0 +1,16 @@ +package wired:log; + +world host { + import api; +} + +interface api { + enum log-level { + debug, + info, + warn, + error + } + + log: func(level: log-level, message: string); +} diff --git a/spatial/wit/wired-math/world.wit b/spatial/wit/wired-math/world.wit new file mode 100644 index 0000000..dc9c79d --- /dev/null +++ b/spatial/wit/wired-math/world.wit @@ -0,0 +1,22 @@ +package wired:math; + +interface types { + record transform { + rotation: quat, + scale: vec3, + translation: vec3, + } + + record vec3 { + x: f32, + y: f32, + z: f32, + } + + record quat { + x: f32, + y: f32, + z: f32, + w: f32, + } +} diff --git a/spatial/wit/wired-physics/deps/wired-math b/spatial/wit/wired-physics/deps/wired-math new file mode 120000 index 0000000..967f496 --- /dev/null +++ b/spatial/wit/wired-physics/deps/wired-math @@ -0,0 +1 @@ +../../wired-math/ \ No newline at end of file diff --git a/spatial/wit/wired-physics/world.wit b/spatial/wit/wired-physics/world.wit new file mode 100644 index 0000000..62c4ca6 --- /dev/null +++ b/spatial/wit/wired-physics/world.wit @@ -0,0 +1,41 @@ +package wired:physics; + +world host { + import types; +} + +interface types { + use wired:math/types.{vec3}; + + resource collider { + constructor(shape: shape); + + density: func() -> f32; + set-density: func(value: f32); + } + + variant shape { + cuboid(vec3), + sphere(sphere), + } + + record sphere { + radius: f32 + } + + resource rigid-body { + constructor(rigid-body-type: rigid-body-type); + + angvel: func() -> vec3; + set-angvel: func(value: vec3); + + linvel: func() -> vec3; + set-linvel: func(value: vec3); + } + + enum rigid-body-type { + dynamic, + fixed, + kinematic, + } +} diff --git a/spatial/wit/wired-player/deps/wired-input b/spatial/wit/wired-player/deps/wired-input new file mode 120000 index 0000000..6160bce --- /dev/null +++ b/spatial/wit/wired-player/deps/wired-input @@ -0,0 +1 @@ +../../wired-input/ \ No newline at end of file diff --git a/spatial/wit/wired-player/deps/wired-math b/spatial/wit/wired-player/deps/wired-math new file mode 120000 index 0000000..967f496 --- /dev/null +++ b/spatial/wit/wired-player/deps/wired-math @@ -0,0 +1 @@ +../../wired-math/ \ No newline at end of file diff --git a/spatial/wit/wired-player/deps/wired-physics b/spatial/wit/wired-player/deps/wired-physics new file mode 120000 index 0000000..8caa253 --- /dev/null +++ b/spatial/wit/wired-player/deps/wired-physics @@ -0,0 +1 @@ +../../wired-physics/ \ No newline at end of file diff --git a/spatial/wit/wired-player/deps/wired-scene b/spatial/wit/wired-player/deps/wired-scene new file mode 120000 index 0000000..2b292a8 --- /dev/null +++ b/spatial/wit/wired-player/deps/wired-scene @@ -0,0 +1 @@ +../../wired-scene/ \ No newline at end of file diff --git a/spatial/wit/wired-player/world.wit b/spatial/wit/wired-player/world.wit new file mode 100644 index 0000000..cb7ac2b --- /dev/null +++ b/spatial/wit/wired-player/world.wit @@ -0,0 +1,38 @@ +package wired:player; + +world host { + import api; +} + +interface api { + use wired:scene/node.{node}; + + record skeleton { + head: node, + spine: node, + hips: node, + + left-upper-arm: node, + left-lower-arm: node, + left-hand: node, + + right-upper-arm: node, + right-lower-arm: node, + right-hand: node, + + left-upper-leg: node, + left-lower-leg: node, + left-foot: node, + + right-upper-leg: node, + right-lower-leg: node, + right-foot: node, + } + + resource player { + skeleton: func() -> skeleton; + } + + list-players: func() -> list; + local-player: func() -> player; +} diff --git a/spatial/wit/wired-prelude/deps/wired-input b/spatial/wit/wired-prelude/deps/wired-input new file mode 120000 index 0000000..1f55dd4 --- /dev/null +++ b/spatial/wit/wired-prelude/deps/wired-input @@ -0,0 +1 @@ +../../wired-input \ No newline at end of file diff --git a/spatial/wit/wired-prelude/deps/wired-log b/spatial/wit/wired-prelude/deps/wired-log new file mode 120000 index 0000000..056c845 --- /dev/null +++ b/spatial/wit/wired-prelude/deps/wired-log @@ -0,0 +1 @@ +../../wired-log \ No newline at end of file diff --git a/spatial/wit/wired-prelude/deps/wired-math b/spatial/wit/wired-prelude/deps/wired-math new file mode 120000 index 0000000..33c05fe --- /dev/null +++ b/spatial/wit/wired-prelude/deps/wired-math @@ -0,0 +1 @@ +../../wired-math \ No newline at end of file diff --git a/spatial/wit/wired-prelude/deps/wired-physics b/spatial/wit/wired-prelude/deps/wired-physics new file mode 120000 index 0000000..b981cad --- /dev/null +++ b/spatial/wit/wired-prelude/deps/wired-physics @@ -0,0 +1 @@ +../../wired-physics \ No newline at end of file diff --git a/spatial/wit/wired-prelude/deps/wired-player b/spatial/wit/wired-prelude/deps/wired-player new file mode 120000 index 0000000..d965478 --- /dev/null +++ b/spatial/wit/wired-prelude/deps/wired-player @@ -0,0 +1 @@ +../../wired-player/ \ No newline at end of file diff --git a/spatial/wit/wired-prelude/deps/wired-scene b/spatial/wit/wired-prelude/deps/wired-scene new file mode 120000 index 0000000..2b292a8 --- /dev/null +++ b/spatial/wit/wired-prelude/deps/wired-scene @@ -0,0 +1 @@ +../../wired-scene/ \ No newline at end of file diff --git a/spatial/wit/wired-prelude/world.wit b/spatial/wit/wired-prelude/world.wit new file mode 100644 index 0000000..8b771ce --- /dev/null +++ b/spatial/wit/wired-prelude/world.wit @@ -0,0 +1,16 @@ +package wired:prelude; + +// Prelude to be `include`d in your world for ease of use. +world imports { + import wired:input/handler; + import wired:log/api; + import wired:math/types; + import wired:physics/types; + import wired:player/api; + include wired:scene/prelude; +} + +world script { + include imports; + include wired:script/script; +} diff --git a/spatial/wit/wired-scene/deps/wired-input b/spatial/wit/wired-scene/deps/wired-input new file mode 120000 index 0000000..6160bce --- /dev/null +++ b/spatial/wit/wired-scene/deps/wired-input @@ -0,0 +1 @@ +../../wired-input/ \ No newline at end of file diff --git a/spatial/wit/wired-scene/deps/wired-math b/spatial/wit/wired-scene/deps/wired-math new file mode 120000 index 0000000..967f496 --- /dev/null +++ b/spatial/wit/wired-scene/deps/wired-math @@ -0,0 +1 @@ +../../wired-math/ \ No newline at end of file diff --git a/spatial/wit/wired-scene/deps/wired-physics b/spatial/wit/wired-scene/deps/wired-physics new file mode 120000 index 0000000..8caa253 --- /dev/null +++ b/spatial/wit/wired-scene/deps/wired-physics @@ -0,0 +1 @@ +../../wired-physics/ \ No newline at end of file diff --git a/spatial/wit/wired-scene/gltf.wit b/spatial/wit/wired-scene/gltf.wit new file mode 100644 index 0000000..cb58201 --- /dev/null +++ b/spatial/wit/wired-scene/gltf.wit @@ -0,0 +1,138 @@ +interface gltf { + use material.{material}; + use mesh.{mesh}; + use node.{node}; + use scene.{scene}; + + // A glTF document. + // Can be saved or loaded independently of the rest of the world. + resource gltf { + constructor(); + + // The currently loaded scene. + active-scene: func() -> option; + set-active-scene: func(value: option>); + + // The default active scene, used when an asset is first loaded. + // If not explicitly set, the first scene will be used. + default-scene: func() -> option; + set-default-scene: func(value: borrow); + + list-materials: func() -> list; + add-material: func(value: borrow); + remove-material: func(value: borrow); + + list-meshes: func() -> list; + add-mesh: func(value: borrow); + remove-mesh: func(value: borrow); + + list-nodes: func() -> list; + add-node: func(value: borrow); + remove-node: func(value: borrow); + + list-scenes: func() -> list; + add-scene: func(value: borrow); + remove-scene: func(value: borrow); + } +} + +interface material { + record color { + r: f32, + g: f32, + b: f32, + a: f32, + } + + resource material { + constructor(); + + id: func() -> u32; + + name: func() -> string; + set-name: func(value: string); + + color: func() -> color; + set-color: func(value: color); + } +} + +interface mesh { + use material.{material}; + + resource primitive { + id: func() -> u32; + + material: func() -> option; + set-material: func(value: option>); + + set-indices: func(value: list); + set-normals: func(value: list); + set-positions: func(value: list); + set-uvs: func(value: list); + } + + resource mesh { + constructor(); + + id: func() -> u32; + + name: func() -> string; + set-name: func(value: string); + + list-primitives: func() -> list; + create-primitive: func() -> primitive; + remove-primitive: func(value: primitive); + } +} + +interface node { + use mesh.{mesh}; + use wired:input/handler.{input-handler}; + use wired:math/types.{transform}; + use wired:physics/types.{collider, rigid-body}; + + resource node { + constructor(); + + id: func() -> u32; + + name: func() -> string; + set-name: func(value: string); + + children: func() -> list; + add-child: func(value: borrow); + remove-child: func(value: borrow); + + parent: func() -> option; + + transform: func() -> transform; + set-transform: func(value: transform); + + mesh: func() -> option; + set-mesh: func(value: option>); + + collider: func() -> option; + set-collider: func(value: option>); + + rigid-body: func() -> option; + set-rigid-body: func(value: option>); + + input-handler: func() -> option; + set-input-handler: func(value: option>); + } +} + +interface scene { + use node.{node}; + + resource scene { + constructor(); + + id: func() -> u32; + + nodes: func() -> list; + add-node: func(value: borrow); + remove-node: func(value: borrow); + } +} diff --git a/spatial/wit/wired-scene/glxf.wit b/spatial/wit/wired-scene/glxf.wit new file mode 100644 index 0000000..791ff3d --- /dev/null +++ b/spatial/wit/wired-scene/glxf.wit @@ -0,0 +1,65 @@ +interface glxf { + use gltf.{gltf}; + use node.{node}; + + // Returns the root glXF that the script is attached to. + get-root: func() -> glxf; + + // A glXF document. + // Used to compose multiple independent glTF assets together. + resource glxf { + constructor(); + + list-assets: func() -> list; + add-asset: func(value: asset); + remove-asset: func(value: asset); + + list-nodes: func() -> list; + add-node: func(value: borrow); + remove-node: func(value: borrow); + } + + variant asset { + gltf(asset-gltf), + glxf(asset-glxf), + } + + resource asset-gltf { + constructor(document: borrow); + + document: func() -> gltf; + + list-nodes: func() -> list; + add-node: func(value: borrow); + remove-node: func(value: borrow); + } + + resource asset-glxf { + constructor(document: borrow); + + document: func() -> glxf; + + list-nodes: func() -> list; + add-node: func(value: borrow); + remove-node: func(value: borrow); + } + + resource glxf-node { + constructor(); + + id: func() -> u32; + + name: func() -> string; + set-name: func(value: string); + + parent: func() -> option; + + children: func() -> children; + set-children: func(value: children); + } + + variant children { + asset(asset), + nodes(list), + } +} diff --git a/spatial/wit/wired-scene/world.wit b/spatial/wit/wired-scene/world.wit new file mode 100644 index 0000000..3e2097b --- /dev/null +++ b/spatial/wit/wired-scene/world.wit @@ -0,0 +1,10 @@ +package wired:scene; + +world prelude { + import gltf; + import glxf; + import material; + import mesh; + import node; + import scene; +} diff --git a/spatial/wit/wired-script/world.wit b/spatial/wit/wired-script/world.wit new file mode 100644 index 0000000..4f83037 --- /dev/null +++ b/spatial/wit/wired-script/world.wit @@ -0,0 +1,13 @@ +package wired:script; + +world script { + export types; +} + +interface types { + resource script { + constructor(); + // Called every tick. + update: func(delta: f32); + } +} diff --git a/v0/protocols/world-host.json b/v0/protocols/world-host.json new file mode 100644 index 0000000..74c82f3 --- /dev/null +++ b/v0/protocols/world-host.json @@ -0,0 +1,52 @@ +{ + "protocol": "https://wired-protocol.org/v0/protocols/world-host.json", + "published": true, + "types": { + "instance": { + "schema": "https://wired-protocol.org/v0/schemas/instance.json", + "dataFormat": ["application/json"] + }, + "instance-info": { + "schema": "https://wired-protocol.org/v0/schemas/instance-info.json", + "dataFormat": ["application/json"] + }, + "connect-url": { + "dataFormat": ["text/plain"] + } + }, + "structure": { + "connect-url": { + "$actions": [ + { + "who": "anyone", + "can": ["read"] + }, + { + "who": "recipient", + "can": ["write"] + } + ] + }, + "instance": { + "$actions": [ + { + "who": "anyone", + "can": ["read", "write"] + } + ], + "instance-info": { + "$actions": [ + { + "who": "anyone", + "can": ["read"] + }, + { + "who": "recipient", + "of": "instance", + "can": ["write"] + } + ] + } + } + } +} diff --git a/v0/schemas/home.json b/v0/schemas/home.json new file mode 100644 index 0000000..b412a23 --- /dev/null +++ b/v0/schemas/home.json @@ -0,0 +1,20 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/home.json", + "type": "object", + "properties": { + "world": { + "type": "object", + "properties": { + "did": { + "type": "string" + }, + "record_id": { + "type": "string" + } + }, + "required": ["did", "record_id"] + } + }, + "required": ["world"] +} diff --git a/v0/schemas/instance-info.json b/v0/schemas/instance-info.json new file mode 100644 index 0000000..98f1eff --- /dev/null +++ b/v0/schemas/instance-info.json @@ -0,0 +1,16 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/instance-info.json", + "type": "object", + "properties": { + "numPlayers": { + "type": "number" + }, + "maxPlayers": { + "type": "number" + }, + "extras": { + "type": "object" + } + } +} diff --git a/v0/schemas/instance.json b/v0/schemas/instance.json new file mode 100644 index 0000000..c4654eb --- /dev/null +++ b/v0/schemas/instance.json @@ -0,0 +1,20 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/instance.json", + "type": "object", + "properties": { + "world": { + "type": "object", + "properties": { + "did": { + "type": "string" + }, + "record_id": { + "type": "string" + } + }, + "required": ["did", "record_id"] + } + }, + "required": ["world"] +} diff --git a/v0/schemas/world.json b/v0/schemas/world.json new file mode 100644 index 0000000..fd8fd5e --- /dev/null +++ b/v0/schemas/world.json @@ -0,0 +1,26 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://wired-protocol.org/v0/schemas/world.json", + "type": "object", + "properties": { + "name": { + "type": "string" + }, + "description": { + "type": "string" + }, + "tags": { + "type": "array", + "items": { + "type": "string" + } + }, + "host": { + "type": "string", + "description": "DID of the world host." + }, + "extras": { + "type": "object" + } + } +}