mirror of
https://github.com/rojo-rbx/rojo.git
synced 2026-04-20 20:55:50 +00:00
Compare commits
52 Commits
v7.0.0-rc.
...
project-cr
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b96a236333 | ||
|
|
79b57b3359 | ||
|
|
c7aeffe586 | ||
|
|
79c02f2457 | ||
|
|
b9ed68fa9e | ||
|
|
6c6d6c9c8d | ||
|
|
e169d7be68 | ||
|
|
192fd7d4dd | ||
|
|
1f1193e857 | ||
|
|
0a412ade88 | ||
|
|
3cef2fe9aa | ||
|
|
18e53f06fe | ||
|
|
eaac539087 | ||
|
|
57005c4fd5 | ||
|
|
ea58999a2a | ||
|
|
a5a69fd9fc | ||
|
|
f1d0f1c1c9 | ||
|
|
83492d7495 | ||
|
|
10abc2254a | ||
|
|
5d5536a95e | ||
|
|
fe81e55925 | ||
|
|
654690d73e | ||
|
|
256aba4bc1 | ||
|
|
49f8845105 | ||
|
|
12370846b4 | ||
|
|
07637dfe96 | ||
|
|
f389a4a1db | ||
|
|
af077c796c | ||
|
|
1c319f2fa8 | ||
|
|
e8afa03f7b | ||
|
|
9b22545842 | ||
|
|
adc733d25c | ||
|
|
6896257647 | ||
|
|
1d9845a6cb | ||
|
|
8461339e9a | ||
|
|
9904d94e4c | ||
|
|
da25c80d0b | ||
|
|
5fa63733fd | ||
|
|
8b54bf0ba1 | ||
|
|
173dc12cb3 | ||
|
|
e136529ff0 | ||
|
|
75542dacb3 | ||
|
|
07abfbde43 | ||
|
|
96112fe118 | ||
|
|
9d0b313261 | ||
|
|
277ddfa9be | ||
|
|
5d88bdb256 | ||
|
|
8d29b43155 | ||
|
|
cc071a6415 | ||
|
|
8954def25c | ||
|
|
d484098781 | ||
|
|
9f06cbf3a0 |
1
.github/FUNDING.yml
vendored
Normal file
1
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
patreon: lpghatguy
|
||||||
42
.github/workflows/ci.yml
vendored
42
.github/workflows/ci.yml
vendored
@@ -11,29 +11,49 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
|
name: Build and Test
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
rust_version: [stable, "1.46.0"]
|
rust_version: [stable, 1.55.0]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v1
|
- uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: true
|
submodules: true
|
||||||
|
|
||||||
- name: Setup Rust toolchain
|
- name: Install Rust
|
||||||
run: rustup default ${{ matrix.rust_version }}
|
uses: actions-rs/toolchain@v1
|
||||||
|
with:
|
||||||
|
toolchain: ${{ matrix.rust_version }}
|
||||||
|
override: true
|
||||||
|
profile: minimal
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: cargo build --locked --verbose
|
run: cargo build --locked --verbose
|
||||||
|
|
||||||
- name: Run tests
|
- name: Test
|
||||||
run: cargo test --locked --verbose
|
run: cargo test --locked --verbose
|
||||||
|
|
||||||
- name: Rustfmt and Clippy
|
lint:
|
||||||
run: |
|
name: Rustfmt and Clippy
|
||||||
cargo fmt -- --check
|
runs-on: ubuntu-latest
|
||||||
cargo clippy
|
|
||||||
if: matrix.rust_version == 'stable'
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
submodules: true
|
||||||
|
|
||||||
|
- name: Install Rust
|
||||||
|
uses: actions-rs/toolchain@v1
|
||||||
|
with:
|
||||||
|
toolchain: stable
|
||||||
|
override: true
|
||||||
|
components: rustfmt, clippy
|
||||||
|
|
||||||
|
- name: Rustfmt
|
||||||
|
run: cargo fmt -- --check
|
||||||
|
|
||||||
|
- name: Clippy
|
||||||
|
run: cargo clippy
|
||||||
199
.github/workflows/release.yml
vendored
199
.github/workflows/release.yml
vendored
@@ -2,65 +2,152 @@ name: Release
|
|||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
tags: ["*"]
|
tags: ["v*"]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
windows:
|
create-release:
|
||||||
runs-on: windows-latest
|
name: Create Release
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v1
|
|
||||||
with:
|
|
||||||
submodules: true
|
|
||||||
|
|
||||||
- name: Build release binary
|
|
||||||
run: cargo build --verbose --locked --release
|
|
||||||
|
|
||||||
- name: Upload artifacts
|
|
||||||
uses: actions/upload-artifact@v1
|
|
||||||
with:
|
|
||||||
name: rojo-win64
|
|
||||||
path: target/release/rojo.exe
|
|
||||||
|
|
||||||
macos:
|
|
||||||
runs-on: macos-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v1
|
|
||||||
with:
|
|
||||||
submodules: true
|
|
||||||
|
|
||||||
- name: Install Rust
|
|
||||||
run: curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y
|
|
||||||
|
|
||||||
- name: Build release binary
|
|
||||||
run: |
|
|
||||||
source $HOME/.cargo/env
|
|
||||||
cargo build --verbose --locked --release
|
|
||||||
env:
|
|
||||||
OPENSSL_STATIC: 1
|
|
||||||
|
|
||||||
- name: Upload artifacts
|
|
||||||
uses: actions/upload-artifact@v1
|
|
||||||
with:
|
|
||||||
name: rojo-macos
|
|
||||||
path: target/release/rojo
|
|
||||||
|
|
||||||
linux:
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v1
|
- name: Create Release
|
||||||
with:
|
id: create_release
|
||||||
submodules: true
|
uses: actions/create-release@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
tag_name: ${{ github.ref }}
|
||||||
|
release_name: ${{ github.ref }}
|
||||||
|
draft: true
|
||||||
|
prerelease: false
|
||||||
|
|
||||||
- name: Build
|
build-plugin:
|
||||||
run: cargo build --locked --verbose --release
|
needs: ["create-release"]
|
||||||
env:
|
name: Build Roblox Studio Plugin
|
||||||
OPENSSL_STATIC: 1
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
submodules: true
|
||||||
|
|
||||||
- name: Upload artifacts
|
- name: Setup Foreman
|
||||||
uses: actions/upload-artifact@v1
|
uses: Roblox/setup-foreman@v1
|
||||||
with:
|
with:
|
||||||
name: rojo-linux
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
path: target/release/rojo
|
|
||||||
|
- name: Build Plugin
|
||||||
|
run: rojo build plugin --output Rojo.rbxm
|
||||||
|
|
||||||
|
- name: Upload Plugin to Release
|
||||||
|
uses: actions/upload-release-asset@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
upload_url: ${{ needs.create-release.outputs.upload_url }}
|
||||||
|
asset_path: Rojo.rbxm
|
||||||
|
asset_name: Rojo.rbxm
|
||||||
|
asset_content_type: application/octet-stream
|
||||||
|
|
||||||
|
- name: Upload Plugin to Artifacts
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Rojo.rbxm
|
||||||
|
path: Rojo.rbxm
|
||||||
|
|
||||||
|
build:
|
||||||
|
needs: ["create-release"]
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
# https://doc.rust-lang.org/rustc/platform-support.html
|
||||||
|
#
|
||||||
|
# FIXME: After the Rojo VS Code extension updates, add architecture
|
||||||
|
# names to each of these releases. We'll rename win64 to windows and add
|
||||||
|
# -x86_64 to each release.
|
||||||
|
include:
|
||||||
|
- host: linux
|
||||||
|
os: ubuntu-latest
|
||||||
|
target: x86_64-unknown-linux-gnu
|
||||||
|
label: linux
|
||||||
|
|
||||||
|
- host: windows
|
||||||
|
os: windows-latest
|
||||||
|
target: x86_64-pc-windows-msvc
|
||||||
|
label: win64
|
||||||
|
|
||||||
|
- host: macos
|
||||||
|
os: macos-latest
|
||||||
|
target: x86_64-apple-darwin
|
||||||
|
label: macos
|
||||||
|
|
||||||
|
- host: macos
|
||||||
|
os: macos-latest
|
||||||
|
target: aarch64-apple-darwin
|
||||||
|
label: macos-aarch64
|
||||||
|
|
||||||
|
name: Build (${{ matrix.target }})
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
env:
|
||||||
|
BIN: rojo
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
submodules: true
|
||||||
|
|
||||||
|
- name: Get Version from Tag
|
||||||
|
shell: bash
|
||||||
|
# https://github.community/t/how-to-get-just-the-tag-name/16241/7#M1027
|
||||||
|
run: |
|
||||||
|
echo "PROJECT_VERSION=${GITHUB_REF#refs/tags/v}" >> $GITHUB_ENV
|
||||||
|
echo "Version is: ${{ env.PROJECT_VERSION }}"
|
||||||
|
|
||||||
|
- name: Install Rust
|
||||||
|
uses: actions-rs/toolchain@v1
|
||||||
|
with:
|
||||||
|
toolchain: stable
|
||||||
|
target: ${{ matrix.target }}
|
||||||
|
override: true
|
||||||
|
profile: minimal
|
||||||
|
|
||||||
|
- name: Build Release
|
||||||
|
run: cargo build --release --locked --verbose
|
||||||
|
env:
|
||||||
|
# Build into a known directory so we can find our build artifact more
|
||||||
|
# easily.
|
||||||
|
CARGO_TARGET_DIR: output
|
||||||
|
|
||||||
|
# On platforms that use OpenSSL, ensure it is statically linked to
|
||||||
|
# make binaries more portable.
|
||||||
|
OPENSSL_STATIC: 1
|
||||||
|
|
||||||
|
- name: Create Release Archive
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
mkdir staging
|
||||||
|
|
||||||
|
if [ "${{ matrix.host }}" = "windows" ]; then
|
||||||
|
cp "output/release/$BIN.exe" staging/
|
||||||
|
cd staging
|
||||||
|
7z a ../release.zip *
|
||||||
|
else
|
||||||
|
cp "output/release/$BIN" staging/
|
||||||
|
cd staging
|
||||||
|
zip ../release.zip *
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Upload Archive to Release
|
||||||
|
uses: actions/upload-release-asset@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
upload_url: ${{ needs.create-release.outputs.upload_url }}
|
||||||
|
asset_path: release.zip
|
||||||
|
asset_name: ${{ env.BIN }}-${{ env.PROJECT_VERSION }}-${{ matrix.label }}.zip
|
||||||
|
asset_content_type: application/octet-stream
|
||||||
|
|
||||||
|
- name: Upload Archive to Artifacts
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: ${{ env.BIN }}-${{ env.PROJECT_VERSION }}-${{ matrix.label }}.zip
|
||||||
|
path: release.zip
|
||||||
64
CHANGELOG.md
64
CHANGELOG.md
@@ -1,8 +1,62 @@
|
|||||||
# Rojo Changelog
|
# Rojo Changelog
|
||||||
|
|
||||||
## Unreleased Changes
|
## Unreleased Changes
|
||||||
|
* Switched from structopt to clap for command line argument parsing.
|
||||||
|
|
||||||
## [7.0.0-rc.1][7.0.0-rc.1] (August 23, 2021)
|
## [7.1.1] - May 26, 2022
|
||||||
|
* Fixed sourcemap command not stripping paths correctly ([#544])
|
||||||
|
* Fixed Studio plugin settings not saving correctly.
|
||||||
|
|
||||||
|
[#544]: https://github.com/rojo-rbx/rojo/pull/544
|
||||||
|
[#545]: https://github.com/rojo-rbx/rojo/pull/545
|
||||||
|
[7.1.1]: https://github.com/rojo-rbx/rojo/releases/tag/v7.1.1
|
||||||
|
|
||||||
|
## [7.1.0] - May 22, 2022
|
||||||
|
* Added support for specifying an address to be used by default in project files. ([#507])
|
||||||
|
* Added support for optional paths in project files. ([#472])
|
||||||
|
* Added support for the new Open Cloud API when uploading. ([#504])
|
||||||
|
* Added `sourcemap` command for generating sourcemaps to feed into other tools. ([#530])
|
||||||
|
* Added PluginActions for connecting/disconnecting a session ([#537])
|
||||||
|
* Added changing toolbar icon to indicate state ([#538])
|
||||||
|
|
||||||
|
[#472]: https://github.com/rojo-rbx/rojo/pull/472
|
||||||
|
[#504]: https://github.com/rojo-rbx/rojo/pull/504
|
||||||
|
[#507]: https://github.com/rojo-rbx/rojo/pull/507
|
||||||
|
[#530]: https://github.com/rojo-rbx/rojo/pull/530
|
||||||
|
[#537]: https://github.com/rojo-rbx/rojo/pull/537
|
||||||
|
[#538]: https://github.com/rojo-rbx/rojo/pull/538
|
||||||
|
[7.1.0]: https://github.com/rojo-rbx/rojo/releases/tag/v7.1.0
|
||||||
|
|
||||||
|
## [7.0.0] - December 10, 2021
|
||||||
|
* Fixed Rojo's interactions with properties enabled by FFlags that are not yet enabled. ([#493])
|
||||||
|
* Improved output in Roblox Studio plugin when bad property data is encountered.
|
||||||
|
* Reintroduced support for CFrame shorthand syntax in Rojo project and `.meta.json` files, matching Rojo 6. ([#430])
|
||||||
|
* Connection settings are now remembered when reconnecting in Roblox Studio. ([#500])
|
||||||
|
* Updated reflection database to Roblox v503.
|
||||||
|
|
||||||
|
[#430]: https://github.com/rojo-rbx/rojo/issues/430
|
||||||
|
[#493]: https://github.com/rojo-rbx/rojo/pull/493
|
||||||
|
[#500]: https://github.com/rojo-rbx/rojo/pull/500
|
||||||
|
[7.0.0]: https://github.com/rojo-rbx/rojo/releases/tag/v7.0.0
|
||||||
|
|
||||||
|
## [7.0.0-rc.3] - October 19, 2021
|
||||||
|
This is the last release candidate for Rojo 7. In an effort to get Rojo 7 out the door, we'll be freezing features from here on out, something we should've done a couple months ago.
|
||||||
|
|
||||||
|
Expect to see Rojo 7 stable soon!
|
||||||
|
|
||||||
|
* Added support for writing `Tags` in project files, model files, and meta files. ([#484])
|
||||||
|
* Adjusted Studio plugin colors to match Roblox Studio palette. ([#482])
|
||||||
|
* Improved experimental two-way sync feature by batching changes. ([#478])
|
||||||
|
|
||||||
|
[#482]: https://github.com/rojo-rbx/rojo/pull/482
|
||||||
|
[#484]: https://github.com/rojo-rbx/rojo/pull/484
|
||||||
|
[#478]: https://github.com/rojo-rbx/rojo/pull/478
|
||||||
|
[7.0.0-rc.3]: https://github.com/rojo-rbx/rojo/releases/tag/v7.0.0-rc.3
|
||||||
|
|
||||||
|
## 7.0.0-rc.2 - October 19, 2021
|
||||||
|
(Botched release due to Git mishap, oops!)
|
||||||
|
|
||||||
|
## [7.0.0-rc.1] - August 23, 2021
|
||||||
In Rojo 6 and previous Rojo 7 alphas, an explicit Vector3 property would be written like this:
|
In Rojo 6 and previous Rojo 7 alphas, an explicit Vector3 property would be written like this:
|
||||||
|
|
||||||
```json
|
```json
|
||||||
@@ -45,9 +99,9 @@ The shorthand property format that most users use is not impacted. For reference
|
|||||||
* Added the `fmt-project` subcommand for formatting Rojo project files.
|
* Added the `fmt-project` subcommand for formatting Rojo project files.
|
||||||
* Improved error output for many subcommands.
|
* Improved error output for many subcommands.
|
||||||
* Updated to stable versions of rbx-dom libraries.
|
* Updated to stable versions of rbx-dom libraries.
|
||||||
* Updated async infrastructure, which should fix a handful of bugs. ([#459][#459])
|
* Updated async infrastructure, which should fix a handful of bugs. ([#459])
|
||||||
* Fixed syncing refs in the Roblox Studio plugin ([#462][#462], [#466][#466])
|
* Fixed syncing refs in the Roblox Studio plugin ([#462], [#466])
|
||||||
* Added support for long paths on Windows. ([#464][#464])
|
* Added support for long paths on Windows. ([#464])
|
||||||
|
|
||||||
[#459]: https://github.com/rojo-rbx/rojo/pull/459
|
[#459]: https://github.com/rojo-rbx/rojo/pull/459
|
||||||
[#462]: https://github.com/rojo-rbx/rojo/pull/462
|
[#462]: https://github.com/rojo-rbx/rojo/pull/462
|
||||||
@@ -58,7 +112,7 @@ The shorthand property format that most users use is not impacted. For reference
|
|||||||
## [7.0.0-alpha.4][7.0.0-alpha.4] (May 5, 2021)
|
## [7.0.0-alpha.4][7.0.0-alpha.4] (May 5, 2021)
|
||||||
* Added the `gameId` and `placeId` optional properties to project files.
|
* Added the `gameId` and `placeId` optional properties to project files.
|
||||||
* When connecting from the Rojo Roblox Studio plugin, Rojo will set the game and place ID of the current place to these values, if set.
|
* When connecting from the Rojo Roblox Studio plugin, Rojo will set the game and place ID of the current place to these values, if set.
|
||||||
* This is equivalent to running `game:SetUniverseId(...)` and `game:SetPlaceId(...)` from the command bar in Studio.
|
* This is equivalent to running `game:SetUniverseId(...)` and `game:SetPlaceId(...)` from the command bar in Studio.
|
||||||
* Added "EXPERIMENTAL!" label to two-way sync toggle in Rojo's Roblox Studio plugin.
|
* Added "EXPERIMENTAL!" label to two-way sync toggle in Rojo's Roblox Studio plugin.
|
||||||
* Fixed `Name` and `Parent` properties being allowed in Rojo projects. ([#413][pr-413])
|
* Fixed `Name` and `Parent` properties being allowed in Rojo projects. ([#413][pr-413])
|
||||||
* Fixed "Open Scripts Externally" feature crashing Studio. ([#369][issue-369])
|
* Fixed "Open Scripts Externally" feature crashing Studio. ([#369][issue-369])
|
||||||
|
|||||||
@@ -29,25 +29,29 @@ Sometimes there's something that Rojo doesn't do that it probably should.
|
|||||||
|
|
||||||
Please file issues and we'll try to help figure out what the best way forward is.
|
Please file issues and we'll try to help figure out what the best way forward is.
|
||||||
|
|
||||||
|
## Local Development Gotchas
|
||||||
|
|
||||||
|
If your build fails with "Error: failed to open file `D:\code\rojo\plugin\modules\roact\src`" you need to update your Git submodules.
|
||||||
|
Run the command and try building again: `git submodule update --init --recursive`.
|
||||||
|
|
||||||
## Pushing a Rojo Release
|
## Pushing a Rojo Release
|
||||||
The Rojo release process is pretty manual right now. If you need to do it, here's how:
|
The Rojo release process is pretty manual right now. If you need to do it, here's how:
|
||||||
|
|
||||||
1. Bump server version in [`Cargo.toml`](Cargo.toml)
|
1. Bump server version in [`Cargo.toml`](Cargo.toml)
|
||||||
2. Bump plugin version in [`plugin/src/Config.lua`](plugin/src/Config.lua)
|
2. Bump plugin version in [`plugin/src/Config.lua`](plugin/src/Config.lua)
|
||||||
3. Run `cargo test` to update `Cargo.lock` and double-check tests
|
3. Run `cargo test` to update `Cargo.lock` and run tests
|
||||||
4. Update [`CHANGELOG.md`](CHANGELOG.md)
|
4. Update [`CHANGELOG.md`](CHANGELOG.md)
|
||||||
5. Commit!
|
5. Commit!
|
||||||
* `git add . && git commit -m "Release vX.Y.Z"`
|
* `git add . && git commit -m "Release vX.Y.Z"`
|
||||||
6. Tag the commit with the version from `Cargo.toml` prepended with a v, like `v0.4.13`
|
6. Tag the commit
|
||||||
|
* `git tag vX.Y.Z`
|
||||||
7. Publish the CLI
|
7. Publish the CLI
|
||||||
* `cargo publish`
|
* `cargo publish`
|
||||||
8. Publish the Plugin
|
8. Publish the Plugin
|
||||||
* `rojo publish plugin --asset_id 6415005344`
|
* `cargo run -- upload plugin --asset_id 6415005344`
|
||||||
* `rojo build plugin -o Rojo.rbxm`
|
|
||||||
9. Push commits and tags
|
9. Push commits and tags
|
||||||
* `git push && git push --tags`
|
* `git push && git push --tags`
|
||||||
10. Copy GitHub release content from previous release
|
10. Copy GitHub release content from previous release
|
||||||
* Update the leading text with a summary about the release
|
* Update the leading text with a summary about the release
|
||||||
* Paste the changelog notes (as-is!) from [`CHANGELOG.md`](CHANGELOG.md)
|
* Paste the changelog notes (as-is!) from [`CHANGELOG.md`](CHANGELOG.md)
|
||||||
* Write a small summary of each major feature
|
* Write a small summary of each major feature
|
||||||
* Attach release artifacts from GitHub Actions for each platform
|
|
||||||
1748
Cargo.lock
generated
1748
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
96
Cargo.toml
96
Cargo.toml
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "rojo"
|
name = "rojo"
|
||||||
version = "7.0.0-rc.1"
|
version = "7.1.1"
|
||||||
authors = ["Lucien Greathouse <me@lpghatguy.com>"]
|
authors = ["Lucien Greathouse <me@lpghatguy.com>"]
|
||||||
description = "Enables professional-grade development tools for Roblox developers"
|
description = "Enables professional-grade development tools for Roblox developers"
|
||||||
license = "MPL-2.0"
|
license = "MPL-2.0"
|
||||||
@@ -28,25 +28,19 @@ default = []
|
|||||||
dev_live_assets = []
|
dev_live_assets = []
|
||||||
|
|
||||||
[workspace]
|
[workspace]
|
||||||
members = [
|
members = ["crates/*"]
|
||||||
"rojo-insta-ext",
|
|
||||||
"memofs",
|
|
||||||
]
|
|
||||||
|
|
||||||
[lib]
|
[lib]
|
||||||
name = "librojo"
|
name = "librojo"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[[bin]]
|
|
||||||
name = "rojo"
|
|
||||||
path = "src/bin.rs"
|
|
||||||
|
|
||||||
[[bench]]
|
[[bench]]
|
||||||
name = "build"
|
name = "build"
|
||||||
harness = false
|
harness = false
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
memofs = { version = "0.2.0", path = "memofs" }
|
rojo-project = { path = "crates/rojo-project" }
|
||||||
|
memofs = { version = "0.2.0", path = "crates/memofs" }
|
||||||
|
|
||||||
# These dependencies can be uncommented when working on rbx-dom simultaneously
|
# These dependencies can be uncommented when working on rbx-dom simultaneously
|
||||||
# rbx_binary = { path = "../rbx-dom/rbx_binary" }
|
# rbx_binary = { path = "../rbx-dom/rbx_binary" }
|
||||||
@@ -55,62 +49,58 @@ memofs = { version = "0.2.0", path = "memofs" }
|
|||||||
# rbx_reflection_database = { path = "../rbx-dom/rbx_reflection_database" }
|
# rbx_reflection_database = { path = "../rbx-dom/rbx_reflection_database" }
|
||||||
# rbx_xml = { path = "../rbx-dom/rbx_xml" }
|
# rbx_xml = { path = "../rbx-dom/rbx_xml" }
|
||||||
|
|
||||||
rbx_binary = "0.6.1"
|
rbx_binary = "0.6.4"
|
||||||
rbx_dom_weak = "2.1.0"
|
rbx_dom_weak = "2.3.0"
|
||||||
rbx_reflection = "4.1.0"
|
rbx_reflection = "4.2.0"
|
||||||
rbx_reflection_database = "0.2.1"
|
rbx_reflection_database = "0.2.2"
|
||||||
rbx_xml = "0.12.1"
|
rbx_xml = "0.12.3"
|
||||||
|
|
||||||
anyhow = "1.0.27"
|
anyhow = "1.0.44"
|
||||||
backtrace = "0.3"
|
backtrace = "0.3.61"
|
||||||
bincode = "1.2.1"
|
bincode = "1.3.3"
|
||||||
crossbeam-channel = "0.5.1"
|
crossbeam-channel = "0.5.1"
|
||||||
csv = "1.1.1"
|
csv = "1.1.6"
|
||||||
env_logger = "0.9.0"
|
env_logger = "0.9.0"
|
||||||
fs-err = "2.2.0"
|
fs-err = "2.6.0"
|
||||||
futures = "0.3.16"
|
futures = "0.3.17"
|
||||||
globset = "0.4.4"
|
globset = "0.4.8"
|
||||||
humantime = "2.1.0"
|
humantime = "2.1.0"
|
||||||
hyper = { version = "0.14.11", features = ["server", "tcp", "http1"] }
|
hyper = { version = "0.14.13", features = ["server", "tcp", "http1"] }
|
||||||
jod-thread = "0.1.0"
|
jod-thread = "0.1.2"
|
||||||
lazy_static = "1.4.0"
|
log = "0.4.14"
|
||||||
log = "0.4.8"
|
maplit = "1.0.2"
|
||||||
maplit = "1.0.1"
|
notify = "4.0.17"
|
||||||
notify = "4.0.14"
|
|
||||||
opener = "0.5.0"
|
opener = "0.5.0"
|
||||||
regex = "1.3.1"
|
reqwest = { version = "0.11.10", features = ["blocking", "json"] }
|
||||||
reqwest = "0.9.20"
|
|
||||||
ritz = "0.1.0"
|
ritz = "0.1.0"
|
||||||
rlua = "0.17.0"
|
|
||||||
roblox_install = "1.0.0"
|
roblox_install = "1.0.0"
|
||||||
serde = { version = "1.0", features = ["derive", "rc"] }
|
serde = { version = "1.0.130", features = ["derive", "rc"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0.68"
|
||||||
structopt = "0.3.5"
|
termcolor = "1.1.2"
|
||||||
termcolor = "1.0.5"
|
thiserror = "1.0.30"
|
||||||
thiserror = "1.0.11"
|
tokio = { version = "1.12.0", features = ["rt", "rt-multi-thread"] }
|
||||||
tokio = { version = "1.9.0", features = ["rt", "rt-multi-thread"] }
|
uuid = { version = "1.0.0", features = ["v4", "serde"] }
|
||||||
uuid = { version = "0.8.1", features = ["v4", "serde"] }
|
clap = { version = "3.1.18", features = ["derive"] }
|
||||||
|
|
||||||
[target.'cfg(windows)'.dependencies]
|
[target.'cfg(windows)'.dependencies]
|
||||||
winreg = "0.9.0"
|
winreg = "0.10.1"
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
memofs = { version = "0.2.0", path = "memofs" }
|
memofs = { version = "0.2.0", path = "crates/memofs" }
|
||||||
|
|
||||||
embed-resource = "1.6"
|
embed-resource = "1.6.4"
|
||||||
anyhow = "1.0.27"
|
anyhow = "1.0.44"
|
||||||
bincode = "1.2.1"
|
bincode = "1.3.3"
|
||||||
fs-err = "2.3.0"
|
fs-err = "2.6.0"
|
||||||
maplit = "1.0.1"
|
maplit = "1.0.2"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
rojo-insta-ext = { path = "rojo-insta-ext" }
|
rojo-insta-ext = { path = "crates/rojo-insta-ext" }
|
||||||
|
|
||||||
criterion = "0.3"
|
criterion = "0.3.5"
|
||||||
insta = { version = "1.3.0", features = ["redactions"] }
|
insta = { version = "1.8.0", features = ["redactions"] }
|
||||||
lazy_static = "1.2"
|
|
||||||
paste = "1.0.5"
|
paste = "1.0.5"
|
||||||
pretty_assertions = "0.7.2"
|
pretty_assertions = "1.2.1"
|
||||||
serde_yaml = "0.8.9"
|
serde_yaml = "0.8.21"
|
||||||
tempfile = "3.0"
|
tempfile = "3.2.0"
|
||||||
walkdir = "2.1"
|
walkdir = "2.3.2"
|
||||||
|
|||||||
BIN
assets/icon-link-32.png
Normal file
BIN
assets/icon-link-32.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.3 KiB |
BIN
assets/icon-warn-32.png
Normal file
BIN
assets/icon-warn-32.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.2 KiB |
@@ -1,5 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
watchexec -c -w plugin "sh -c './bin/install-dev-plugin.sh'"
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
DIR="$( mktemp -d )"
|
|
||||||
PLUGIN_FILE="$DIR/Rojo.rbxm"
|
|
||||||
TESTEZ_FILE="$DIR/TestEZ.rbxm"
|
|
||||||
|
|
||||||
rojo build plugin -o "$PLUGIN_FILE"
|
|
||||||
rojo build plugin/testez.project.json -o "$TESTEZ_FILE"
|
|
||||||
remodel bin/mark-plugin-as-dev.lua "$PLUGIN_FILE" "$TESTEZ_FILE" 2>/dev/null
|
|
||||||
|
|
||||||
cp "$PLUGIN_FILE" "$LOCALAPPDATA/Roblox/Plugins/Rojo.rbxm"
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
rojo build plugin -o "$LOCALAPPDATA/Roblox/Plugins/Rojo.rbxm"
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
local pluginPath, testezPath = ...
|
|
||||||
|
|
||||||
local plugin = remodel.readModelFile(pluginPath)[1]
|
|
||||||
local testez = remodel.readModelFile(testezPath)[1]
|
|
||||||
|
|
||||||
local marker = Instance.new("Folder")
|
|
||||||
marker.Name = "ROJO_DEV_BUILD"
|
|
||||||
marker.Parent = plugin
|
|
||||||
|
|
||||||
testez.Parent = plugin
|
|
||||||
|
|
||||||
remodel.writeModelFile(plugin, pluginPath)
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
local pluginPath, placePath = ...
|
|
||||||
|
|
||||||
local plugin = remodel.readModelFile(pluginPath)[1]
|
|
||||||
local place = remodel.readPlaceFile(placePath)
|
|
||||||
|
|
||||||
plugin.Parent = place:GetService("ReplicatedStorage")
|
|
||||||
|
|
||||||
remodel.writePlaceFile(place, placePath)
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
./bin/run-cli-tests.sh
|
|
||||||
./bin/run-plugin-tests.sh
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
cargo test --all --locked
|
|
||||||
cargo fmt -- --check
|
|
||||||
|
|
||||||
touch src/lib.rs # Nudge Rust source to make Clippy actually check things
|
|
||||||
cargo clippy
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
DIR="$( mktemp -d )"
|
|
||||||
PLUGIN_FILE="$DIR/Rojo.rbxmx"
|
|
||||||
PLACE_FILE="$DIR/RojoTestPlace.rbxlx"
|
|
||||||
|
|
||||||
rojo build plugin -o "$PLUGIN_FILE"
|
|
||||||
rojo build plugin/place.project.json -o "$PLACE_FILE"
|
|
||||||
|
|
||||||
remodel bin/put-plugin-in-test-place.lua "$PLUGIN_FILE" "$PLACE_FILE"
|
|
||||||
|
|
||||||
run-in-roblox -s plugin/testBootstrap.server.lua "$PLACE_FILE"
|
|
||||||
|
|
||||||
luacheck plugin/src plugin/log plugin/http
|
|
||||||
16
crates/rojo-project/Cargo.toml
Normal file
16
crates/rojo-project/Cargo.toml
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
[package]
|
||||||
|
name = "rojo-project"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
anyhow = "1.0.57"
|
||||||
|
globset = { version = "0.4.8", features = ["serde1"] }
|
||||||
|
log = "0.4.17"
|
||||||
|
rbx_dom_weak = "2.3.0"
|
||||||
|
rbx_reflection = "4.2.0"
|
||||||
|
rbx_reflection_database = "0.2.4"
|
||||||
|
serde = { version = "1.0.137", features = ["derive"] }
|
||||||
|
serde_json = "1.0.81"
|
||||||
4
crates/rojo-project/README.md
Normal file
4
crates/rojo-project/README.md
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# rojo-project
|
||||||
|
Project file format crate for [Rojo].
|
||||||
|
|
||||||
|
[Rojo]: https://rojo.space
|
||||||
@@ -1,6 +1,3 @@
|
|||||||
//! Wrapper around globset's Glob type that has better serialization
|
|
||||||
//! characteristics by coupling Glob and GlobMatcher into a single type.
|
|
||||||
|
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
use globset::{Glob as InnerGlob, GlobMatcher};
|
use globset::{Glob as InnerGlob, GlobMatcher};
|
||||||
@@ -8,6 +5,8 @@ use serde::{de::Error as _, Deserialize, Deserializer, Serialize, Serializer};
|
|||||||
|
|
||||||
pub use globset::Error;
|
pub use globset::Error;
|
||||||
|
|
||||||
|
/// Wrapper around globset's Glob type that has better serialization
|
||||||
|
/// characteristics by coupling Glob and GlobMatcher into a single type.
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct Glob {
|
pub struct Glob {
|
||||||
inner: InnerGlob,
|
inner: InnerGlob,
|
||||||
7
crates/rojo-project/src/lib.rs
Normal file
7
crates/rojo-project/src/lib.rs
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
pub mod glob;
|
||||||
|
mod path_serializer;
|
||||||
|
mod project;
|
||||||
|
mod resolution;
|
||||||
|
|
||||||
|
pub use project::{OptionalPathNode, PathNode, Project, ProjectNode};
|
||||||
|
pub use resolution::{AmbiguousValue, UnresolvedValue};
|
||||||
21
crates/rojo-project/src/path_serializer.rs
Normal file
21
crates/rojo-project/src/path_serializer.rs
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
//! Path serializer is used to serialize absolute paths in a cross-platform way,
|
||||||
|
//! by replacing all directory separators with /.
|
||||||
|
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
use serde::Serializer;
|
||||||
|
|
||||||
|
pub fn serialize_absolute<S, T>(path: T, serializer: S) -> Result<S::Ok, S::Error>
|
||||||
|
where
|
||||||
|
S: Serializer,
|
||||||
|
T: AsRef<Path>,
|
||||||
|
{
|
||||||
|
let as_str = path
|
||||||
|
.as_ref()
|
||||||
|
.as_os_str()
|
||||||
|
.to_str()
|
||||||
|
.expect("Invalid Unicode in file path, cannot serialize");
|
||||||
|
let replaced = as_str.replace("\\", "/");
|
||||||
|
|
||||||
|
serializer.serialize_str(&replaced)
|
||||||
|
}
|
||||||
363
crates/rojo-project/src/project.rs
Normal file
363
crates/rojo-project/src/project.rs
Normal file
@@ -0,0 +1,363 @@
|
|||||||
|
use std::collections::{BTreeMap, HashMap, HashSet};
|
||||||
|
use std::fs;
|
||||||
|
use std::net::IpAddr;
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::glob::Glob;
|
||||||
|
use crate::resolution::UnresolvedValue;
|
||||||
|
|
||||||
|
static PROJECT_FILENAME: &str = "default.project.json";
|
||||||
|
|
||||||
|
/// Contains all of the configuration for a Rojo-managed project.
|
||||||
|
///
|
||||||
|
/// Rojo project files are stored in `.project.json` files.
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
|
#[serde(deny_unknown_fields, rename_all = "camelCase")]
|
||||||
|
pub struct Project {
|
||||||
|
/// The name of the top-level instance described by the project.
|
||||||
|
pub name: String,
|
||||||
|
|
||||||
|
/// The tree of instances described by this project. Projects always
|
||||||
|
/// describe at least one instance.
|
||||||
|
pub tree: ProjectNode,
|
||||||
|
|
||||||
|
/// If specified, sets the default port that `rojo serve` should use when
|
||||||
|
/// using this project for live sync.
|
||||||
|
///
|
||||||
|
/// Can be overriden with the `--port` flag.
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub serve_port: Option<u16>,
|
||||||
|
|
||||||
|
/// If specified, sets the default IP address that `rojo serve` should use
|
||||||
|
/// when using this project for live sync.
|
||||||
|
///
|
||||||
|
/// Can be overridden with the `--address` flag.
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub serve_address: Option<IpAddr>,
|
||||||
|
|
||||||
|
/// If specified, contains the set of place IDs that this project is
|
||||||
|
/// compatible with when doing live sync.
|
||||||
|
///
|
||||||
|
/// This setting is intended to help prevent syncing a Rojo project into the
|
||||||
|
/// wrong Roblox place.
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub serve_place_ids: Option<HashSet<u64>>,
|
||||||
|
|
||||||
|
/// If specified, sets the current place's place ID when connecting to the
|
||||||
|
/// Rojo server from Roblox Studio.
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub place_id: Option<u64>,
|
||||||
|
|
||||||
|
/// If specified, sets the current place's game ID when connecting to the
|
||||||
|
/// Rojo server from Roblox Studio.
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub game_id: Option<u64>,
|
||||||
|
|
||||||
|
/// A list of globs, relative to the folder the project file is in, that
|
||||||
|
/// match files that should be excluded if Rojo encounters them.
|
||||||
|
#[serde(default, skip_serializing_if = "Vec::is_empty")]
|
||||||
|
pub glob_ignore_paths: Vec<Glob>,
|
||||||
|
|
||||||
|
/// The path to the file that this project came from. Relative paths in the
|
||||||
|
/// project should be considered relative to the parent of this field, also
|
||||||
|
/// given by `Project::folder_location`.
|
||||||
|
#[serde(skip)]
|
||||||
|
pub file_location: PathBuf,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Project {
|
||||||
|
/// Tells whether the given path describes a Rojo project.
|
||||||
|
pub fn is_project_file(path: &Path) -> bool {
|
||||||
|
path.file_name()
|
||||||
|
.and_then(|name| name.to_str())
|
||||||
|
.map(|name| name.ends_with(".project.json"))
|
||||||
|
.unwrap_or(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Loads a project file from a slice and a path that indicates where the
|
||||||
|
/// project should resolve paths relative to.
|
||||||
|
pub fn load_from_slice(contents: &[u8], project_file_location: &Path) -> anyhow::Result<Self> {
|
||||||
|
let mut project: Self = serde_json::from_slice(&contents).with_context(|| {
|
||||||
|
format!(
|
||||||
|
"Error parsing Rojo project at {}",
|
||||||
|
project_file_location.display()
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
|
||||||
|
project.file_location = project_file_location.to_path_buf();
|
||||||
|
project.check_compatibility();
|
||||||
|
Ok(project)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Fuzzy-find a Rojo project and load it.
|
||||||
|
pub fn load_fuzzy(fuzzy_project_location: &Path) -> anyhow::Result<Option<Self>> {
|
||||||
|
if let Some(project_path) = Self::locate(fuzzy_project_location) {
|
||||||
|
let project = Self::load_exact(&project_path)?;
|
||||||
|
|
||||||
|
Ok(Some(project))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Gives the path that all project file paths should resolve relative to.
|
||||||
|
pub fn folder_location(&self) -> &Path {
|
||||||
|
self.file_location.parent().unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempt to locate a project represented by the given path.
|
||||||
|
///
|
||||||
|
/// This will find a project if the path refers to a `.project.json` file,
|
||||||
|
/// or is a folder that contains a `default.project.json` file.
|
||||||
|
fn locate(path: &Path) -> Option<PathBuf> {
|
||||||
|
let meta = fs::metadata(path).ok()?;
|
||||||
|
|
||||||
|
if meta.is_file() {
|
||||||
|
if Project::is_project_file(path) {
|
||||||
|
Some(path.to_path_buf())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let child_path = path.join(PROJECT_FILENAME);
|
||||||
|
let child_meta = fs::metadata(&child_path).ok()?;
|
||||||
|
|
||||||
|
if child_meta.is_file() {
|
||||||
|
Some(child_path)
|
||||||
|
} else {
|
||||||
|
// This is a folder with the same name as a Rojo default project
|
||||||
|
// file.
|
||||||
|
//
|
||||||
|
// That's pretty weird, but we can roll with it.
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn load_exact(project_file_location: &Path) -> anyhow::Result<Self> {
|
||||||
|
let contents = fs::read_to_string(project_file_location)?;
|
||||||
|
|
||||||
|
let mut project: Project = serde_json::from_str(&contents).with_context(|| {
|
||||||
|
format!(
|
||||||
|
"Error parsing Rojo project at {}",
|
||||||
|
project_file_location.display()
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
|
||||||
|
project.file_location = project_file_location.to_path_buf();
|
||||||
|
project.check_compatibility();
|
||||||
|
|
||||||
|
Ok(project)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Checks if there are any compatibility issues with this project file and
|
||||||
|
/// warns the user if there are any.
|
||||||
|
fn check_compatibility(&self) {
|
||||||
|
self.tree.validate_reserved_names();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Default, Serialize, Deserialize)]
|
||||||
|
pub struct OptionalPathNode {
|
||||||
|
#[serde(serialize_with = "crate::path_serializer::serialize_absolute")]
|
||||||
|
pub optional: PathBuf,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl OptionalPathNode {
|
||||||
|
pub fn new(optional: PathBuf) -> Self {
|
||||||
|
OptionalPathNode { optional }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Describes a path that is either optional or required
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
|
#[serde(untagged)]
|
||||||
|
pub enum PathNode {
|
||||||
|
Required(#[serde(serialize_with = "crate::path_serializer::serialize_absolute")] PathBuf),
|
||||||
|
Optional(OptionalPathNode),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PathNode {
|
||||||
|
pub fn path(&self) -> &Path {
|
||||||
|
match self {
|
||||||
|
PathNode::Required(pathbuf) => &pathbuf,
|
||||||
|
PathNode::Optional(OptionalPathNode { optional }) => &optional,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Describes an instance and its descendants in a project.
|
||||||
|
#[derive(Debug, Clone, PartialEq, Default, Serialize, Deserialize)]
|
||||||
|
pub struct ProjectNode {
|
||||||
|
/// If set, defines the ClassName of the described instance.
|
||||||
|
///
|
||||||
|
/// `$className` MUST be set if `$path` is not set.
|
||||||
|
///
|
||||||
|
/// `$className` CANNOT be set if `$path` is set and the instance described
|
||||||
|
/// by that path has a ClassName other than Folder.
|
||||||
|
#[serde(rename = "$className", skip_serializing_if = "Option::is_none")]
|
||||||
|
pub class_name: Option<String>,
|
||||||
|
|
||||||
|
/// Contains all of the children of the described instance.
|
||||||
|
#[serde(flatten)]
|
||||||
|
pub children: BTreeMap<String, ProjectNode>,
|
||||||
|
|
||||||
|
/// The properties that will be assigned to the resulting instance.
|
||||||
|
#[serde(
|
||||||
|
rename = "$properties",
|
||||||
|
default,
|
||||||
|
skip_serializing_if = "HashMap::is_empty"
|
||||||
|
)]
|
||||||
|
pub properties: HashMap<String, UnresolvedValue>,
|
||||||
|
|
||||||
|
/// Defines the behavior when Rojo encounters unknown instances in Roblox
|
||||||
|
/// Studio during live sync. `$ignoreUnknownInstances` should be considered
|
||||||
|
/// a large hammer and used with care.
|
||||||
|
///
|
||||||
|
/// If set to `true`, those instances will be left alone. This may cause
|
||||||
|
/// issues when files that turn into instances are removed while Rojo is not
|
||||||
|
/// running.
|
||||||
|
///
|
||||||
|
/// If set to `false`, Rojo will destroy any instances it does not
|
||||||
|
/// recognize.
|
||||||
|
///
|
||||||
|
/// If unset, its default value depends on other settings:
|
||||||
|
/// - If `$path` is not set, defaults to `true`
|
||||||
|
/// - If `$path` is set, defaults to `false`
|
||||||
|
#[serde(
|
||||||
|
rename = "$ignoreUnknownInstances",
|
||||||
|
skip_serializing_if = "Option::is_none"
|
||||||
|
)]
|
||||||
|
pub ignore_unknown_instances: Option<bool>,
|
||||||
|
|
||||||
|
/// Defines that this instance should come from the given file path. This
|
||||||
|
/// path can point to any file type supported by Rojo, including Lua files
|
||||||
|
/// (`.lua`), Roblox models (`.rbxm`, `.rbxmx`), and localization table
|
||||||
|
/// spreadsheets (`.csv`).
|
||||||
|
#[serde(rename = "$path", skip_serializing_if = "Option::is_none")]
|
||||||
|
pub path: Option<PathNode>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ProjectNode {
|
||||||
|
fn validate_reserved_names(&self) {
|
||||||
|
for (name, child) in &self.children {
|
||||||
|
if name.starts_with('$') {
|
||||||
|
log::warn!(
|
||||||
|
"Keys starting with '$' are reserved by Rojo to ensure forward compatibility."
|
||||||
|
);
|
||||||
|
log::warn!(
|
||||||
|
"This project uses the key '{}', which should be renamed.",
|
||||||
|
name
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
child.validate_reserved_names();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn path_node_required() {
|
||||||
|
let path_node: PathNode = serde_json::from_str(r#""src""#).unwrap();
|
||||||
|
assert_eq!(path_node, PathNode::Required(PathBuf::from("src")));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn path_node_optional() {
|
||||||
|
let path_node: PathNode = serde_json::from_str(r#"{ "optional": "src" }"#).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
path_node,
|
||||||
|
PathNode::Optional(OptionalPathNode::new(PathBuf::from("src")))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn project_node_required() {
|
||||||
|
let project_node: ProjectNode = serde_json::from_str(
|
||||||
|
r#"{
|
||||||
|
"$path": "src"
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
project_node.path,
|
||||||
|
Some(PathNode::Required(PathBuf::from("src")))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn project_node_optional() {
|
||||||
|
let project_node: ProjectNode = serde_json::from_str(
|
||||||
|
r#"{
|
||||||
|
"$path": { "optional": "src" }
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
project_node.path,
|
||||||
|
Some(PathNode::Optional(OptionalPathNode::new(PathBuf::from(
|
||||||
|
"src"
|
||||||
|
))))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn project_node_none() {
|
||||||
|
let project_node: ProjectNode = serde_json::from_str(
|
||||||
|
r#"{
|
||||||
|
"$className": "Folder"
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(project_node.path, None);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn project_node_optional_serialize_absolute() {
|
||||||
|
let project_node: ProjectNode = serde_json::from_str(
|
||||||
|
r#"{
|
||||||
|
"$path": { "optional": "..\\src" }
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let serialized = serde_json::to_string(&project_node).unwrap();
|
||||||
|
assert_eq!(serialized, r#"{"$path":{"optional":"../src"}}"#);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn project_node_optional_serialize_absolute_no_change() {
|
||||||
|
let project_node: ProjectNode = serde_json::from_str(
|
||||||
|
r#"{
|
||||||
|
"$path": { "optional": "../src" }
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let serialized = serde_json::to_string(&project_node).unwrap();
|
||||||
|
assert_eq!(serialized, r#"{"$path":{"optional":"../src"}}"#);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn project_node_optional_serialize_optional() {
|
||||||
|
let project_node: ProjectNode = serde_json::from_str(
|
||||||
|
r#"{
|
||||||
|
"$path": "..\\src"
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let serialized = serde_json::to_string(&project_node).unwrap();
|
||||||
|
assert_eq!(serialized, r#"{"$path":"../src"}"#);
|
||||||
|
}
|
||||||
|
}
|
||||||
294
crates/rojo-project/src/resolution.rs
Normal file
294
crates/rojo-project/src/resolution.rs
Normal file
@@ -0,0 +1,294 @@
|
|||||||
|
use std::borrow::Borrow;
|
||||||
|
|
||||||
|
use anyhow::format_err;
|
||||||
|
use rbx_dom_weak::types::{
|
||||||
|
CFrame, Color3, Content, Enum, Matrix3, Tags, Variant, VariantType, Vector2, Vector3,
|
||||||
|
};
|
||||||
|
use rbx_reflection::{DataType, PropertyDescriptor};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
/// A user-friendly version of `Variant` that supports specifying ambiguous
|
||||||
|
/// values. Ambiguous values need a reflection database to be resolved to a
|
||||||
|
/// usable value.
|
||||||
|
///
|
||||||
|
/// This type is used in Rojo projects and JSON models to make specifying the
|
||||||
|
/// most common types of properties, like strings or vectors, much easier.
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
|
#[serde(untagged)]
|
||||||
|
pub enum UnresolvedValue {
|
||||||
|
FullyQualified(Variant),
|
||||||
|
Ambiguous(AmbiguousValue),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UnresolvedValue {
|
||||||
|
pub fn resolve(self, class_name: &str, prop_name: &str) -> anyhow::Result<Variant> {
|
||||||
|
match self {
|
||||||
|
UnresolvedValue::FullyQualified(full) => Ok(full),
|
||||||
|
UnresolvedValue::Ambiguous(partial) => partial.resolve(class_name, prop_name),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
|
#[serde(untagged)]
|
||||||
|
pub enum AmbiguousValue {
|
||||||
|
Bool(bool),
|
||||||
|
String(String),
|
||||||
|
StringArray(Vec<String>),
|
||||||
|
Number(f64),
|
||||||
|
Array2([f64; 2]),
|
||||||
|
Array3([f64; 3]),
|
||||||
|
Array4([f64; 4]),
|
||||||
|
Array12([f64; 12]),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AmbiguousValue {
|
||||||
|
pub fn resolve(self, class_name: &str, prop_name: &str) -> anyhow::Result<Variant> {
|
||||||
|
let property = find_descriptor(class_name, prop_name)
|
||||||
|
.ok_or_else(|| format_err!("Unknown property {}.{}", class_name, prop_name))?;
|
||||||
|
|
||||||
|
match &property.data_type {
|
||||||
|
DataType::Enum(enum_name) => {
|
||||||
|
let database = rbx_reflection_database::get();
|
||||||
|
|
||||||
|
let enum_descriptor = database.enums.get(enum_name).ok_or_else(|| {
|
||||||
|
format_err!("Unknown enum {}. This is a Rojo bug!", enum_name)
|
||||||
|
})?;
|
||||||
|
|
||||||
|
let error = |what: &str| {
|
||||||
|
let mut all_values = enum_descriptor
|
||||||
|
.items
|
||||||
|
.keys()
|
||||||
|
.map(|value| value.borrow())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
all_values.sort();
|
||||||
|
|
||||||
|
let examples = nonexhaustive_list(&all_values);
|
||||||
|
|
||||||
|
format_err!(
|
||||||
|
"Invalid value for property {}.{}. Got {} but \
|
||||||
|
expected a member of the {} enum such as {}",
|
||||||
|
class_name,
|
||||||
|
prop_name,
|
||||||
|
what,
|
||||||
|
enum_name,
|
||||||
|
examples,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
let value = match self {
|
||||||
|
AmbiguousValue::String(value) => value,
|
||||||
|
unresolved => return Err(error(unresolved.describe())),
|
||||||
|
};
|
||||||
|
|
||||||
|
let resolved = enum_descriptor
|
||||||
|
.items
|
||||||
|
.get(value.as_str())
|
||||||
|
.ok_or_else(|| error(value.as_str()))?;
|
||||||
|
|
||||||
|
Ok(Enum::from_u32(*resolved).into())
|
||||||
|
}
|
||||||
|
DataType::Value(variant_ty) => match (variant_ty, self) {
|
||||||
|
(VariantType::Bool, AmbiguousValue::Bool(value)) => Ok(value.into()),
|
||||||
|
|
||||||
|
(VariantType::Float32, AmbiguousValue::Number(value)) => Ok((value as f32).into()),
|
||||||
|
(VariantType::Float64, AmbiguousValue::Number(value)) => Ok(value.into()),
|
||||||
|
(VariantType::Int32, AmbiguousValue::Number(value)) => Ok((value as i32).into()),
|
||||||
|
(VariantType::Int64, AmbiguousValue::Number(value)) => Ok((value as i64).into()),
|
||||||
|
|
||||||
|
(VariantType::String, AmbiguousValue::String(value)) => Ok(value.into()),
|
||||||
|
(VariantType::Tags, AmbiguousValue::StringArray(value)) => {
|
||||||
|
Ok(Tags::from(value).into())
|
||||||
|
}
|
||||||
|
(VariantType::Content, AmbiguousValue::String(value)) => {
|
||||||
|
Ok(Content::from(value).into())
|
||||||
|
}
|
||||||
|
|
||||||
|
(VariantType::Vector2, AmbiguousValue::Array2(value)) => {
|
||||||
|
Ok(Vector2::new(value[0] as f32, value[1] as f32).into())
|
||||||
|
}
|
||||||
|
|
||||||
|
(VariantType::Vector3, AmbiguousValue::Array3(value)) => {
|
||||||
|
Ok(Vector3::new(value[0] as f32, value[1] as f32, value[2] as f32).into())
|
||||||
|
}
|
||||||
|
|
||||||
|
(VariantType::Color3, AmbiguousValue::Array3(value)) => {
|
||||||
|
Ok(Color3::new(value[0] as f32, value[1] as f32, value[2] as f32).into())
|
||||||
|
}
|
||||||
|
|
||||||
|
(VariantType::CFrame, AmbiguousValue::Array12(value)) => {
|
||||||
|
let value = value.map(|v| v as f32);
|
||||||
|
let pos = Vector3::new(value[0], value[1], value[2]);
|
||||||
|
let orientation = Matrix3::new(
|
||||||
|
Vector3::new(value[3], value[4], value[5]),
|
||||||
|
Vector3::new(value[6], value[7], value[8]),
|
||||||
|
Vector3::new(value[9], value[10], value[11]),
|
||||||
|
);
|
||||||
|
|
||||||
|
Ok(CFrame::new(pos, orientation).into())
|
||||||
|
}
|
||||||
|
|
||||||
|
(_, unresolved) => Err(format_err!(
|
||||||
|
"Wrong type of value for property {}.{}. Expected {:?}, got {}",
|
||||||
|
class_name,
|
||||||
|
prop_name,
|
||||||
|
variant_ty,
|
||||||
|
unresolved.describe(),
|
||||||
|
)),
|
||||||
|
},
|
||||||
|
_ => Err(format_err!(
|
||||||
|
"Unknown data type for property {}.{}",
|
||||||
|
class_name,
|
||||||
|
prop_name
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn describe(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
AmbiguousValue::Bool(_) => "a bool",
|
||||||
|
AmbiguousValue::String(_) => "a string",
|
||||||
|
AmbiguousValue::StringArray(_) => "an array of strings",
|
||||||
|
AmbiguousValue::Number(_) => "a number",
|
||||||
|
AmbiguousValue::Array2(_) => "an array of two numbers",
|
||||||
|
AmbiguousValue::Array3(_) => "an array of three numbers",
|
||||||
|
AmbiguousValue::Array4(_) => "an array of four numbers",
|
||||||
|
AmbiguousValue::Array12(_) => "an array of twelve numbers",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn find_descriptor(
|
||||||
|
class_name: &str,
|
||||||
|
prop_name: &str,
|
||||||
|
) -> Option<&'static PropertyDescriptor<'static>> {
|
||||||
|
let database = rbx_reflection_database::get();
|
||||||
|
let mut current_class_name = class_name;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let class = database.classes.get(current_class_name)?;
|
||||||
|
if let Some(descriptor) = class.properties.get(prop_name) {
|
||||||
|
return Some(descriptor);
|
||||||
|
}
|
||||||
|
|
||||||
|
current_class_name = class.superclass.as_deref()?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Outputs a string containing up to MAX_ITEMS entries from the given list. If
|
||||||
|
/// there are more than MAX_ITEMS items, the number of remaining items will be
|
||||||
|
/// listed.
|
||||||
|
fn nonexhaustive_list(values: &[&str]) -> String {
|
||||||
|
use std::fmt::Write;
|
||||||
|
|
||||||
|
const MAX_ITEMS: usize = 8;
|
||||||
|
|
||||||
|
let mut output = String::new();
|
||||||
|
|
||||||
|
let last_index = values.len() - 1;
|
||||||
|
let main_length = last_index.min(9);
|
||||||
|
|
||||||
|
let main_list = &values[..main_length];
|
||||||
|
for value in main_list {
|
||||||
|
output.push_str(value);
|
||||||
|
output.push_str(", ");
|
||||||
|
}
|
||||||
|
|
||||||
|
if values.len() > MAX_ITEMS {
|
||||||
|
write!(output, "or {} more", values.len() - main_length).unwrap();
|
||||||
|
} else {
|
||||||
|
output.push_str("or ");
|
||||||
|
output.push_str(values[values.len() - 1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
output
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
fn resolve(class: &str, prop: &str, json_value: &str) -> Variant {
|
||||||
|
let unresolved: UnresolvedValue = serde_json::from_str(json_value).unwrap();
|
||||||
|
unresolved.resolve(class, prop).unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bools() {
|
||||||
|
assert_eq!(resolve("BoolValue", "Value", "false"), Variant::Bool(false));
|
||||||
|
|
||||||
|
// Script.Disabled is inherited from BaseScript
|
||||||
|
assert_eq!(resolve("Script", "Disabled", "true"), Variant::Bool(true));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn strings() {
|
||||||
|
// String literals can stay as strings
|
||||||
|
assert_eq!(
|
||||||
|
resolve("StringValue", "Value", "\"Hello!\""),
|
||||||
|
Variant::String("Hello!".into()),
|
||||||
|
);
|
||||||
|
|
||||||
|
// String literals can also turn into Content
|
||||||
|
assert_eq!(
|
||||||
|
resolve("Sky", "MoonTextureId", "\"rbxassetid://12345\""),
|
||||||
|
Variant::Content("rbxassetid://12345".into()),
|
||||||
|
);
|
||||||
|
|
||||||
|
// What about BinaryString values? For forward-compatibility reasons, we
|
||||||
|
// don't support any shorthands for BinaryString.
|
||||||
|
//
|
||||||
|
// assert_eq!(
|
||||||
|
// resolve("Folder", "Tags", "\"a\\u0000b\\u0000c\""),
|
||||||
|
// Variant::BinaryString(b"a\0b\0c".to_vec().into()),
|
||||||
|
// );
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn numbers() {
|
||||||
|
assert_eq!(
|
||||||
|
resolve("Part", "CollisionGroupId", "123"),
|
||||||
|
Variant::Int32(123),
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
resolve("Folder", "SourceAssetId", "532413"),
|
||||||
|
Variant::Int64(532413),
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(resolve("Part", "Transparency", "1"), Variant::Float32(1.0));
|
||||||
|
assert_eq!(resolve("NumberValue", "Value", "1"), Variant::Float64(1.0));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn vectors() {
|
||||||
|
assert_eq!(
|
||||||
|
resolve("ParticleEmitter", "SpreadAngle", "[1, 2]"),
|
||||||
|
Variant::Vector2(Vector2::new(1.0, 2.0)),
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
resolve("Part", "Position", "[4, 5, 6]"),
|
||||||
|
Variant::Vector3(Vector3::new(4.0, 5.0, 6.0)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn colors() {
|
||||||
|
assert_eq!(
|
||||||
|
resolve("Part", "Color", "[1, 1, 1]"),
|
||||||
|
Variant::Color3(Color3::new(1.0, 1.0, 1.0)),
|
||||||
|
);
|
||||||
|
|
||||||
|
// There aren't any user-facing Color3uint8 properties. If there are
|
||||||
|
// some, we should treat them the same in the future.
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn enums() {
|
||||||
|
assert_eq!(
|
||||||
|
resolve("Lighting", "Technology", "\"Voxel\""),
|
||||||
|
Variant::Enum(Enum::from_u32(1)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
[tools]
|
[tools]
|
||||||
rojo = { source = "rojo-rbx/rojo", version = "6.1.0" }
|
rojo = { source = "rojo-rbx/rojo", version = "7.1.1" }
|
||||||
run-in-roblox = { source = "rojo-rbx/run-in-roblox", version = "0.3.0" }
|
run-in-roblox = { source = "rojo-rbx/run-in-roblox", version = "0.3.0" }
|
||||||
|
selene = { source = "Kampfkarren/selene", version = "0.17.0" }
|
||||||
|
|||||||
Submodule plugin/modules/testez updated: 6e9157db3c...25d957d4d5
@@ -388,6 +388,11 @@ types = {
|
|||||||
end,
|
end,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
Tags = {
|
||||||
|
fromPod = identity,
|
||||||
|
toPod = identity,
|
||||||
|
},
|
||||||
|
|
||||||
Vector2 = {
|
Vector2 = {
|
||||||
fromPod = unpackDecoder(Vector2.new),
|
fromPod = unpackDecoder(Vector2.new),
|
||||||
|
|
||||||
|
|||||||
@@ -20,7 +20,21 @@ local function set(container, key, value)
|
|||||||
end
|
end
|
||||||
|
|
||||||
function PropertyDescriptor.fromRaw(data, className, propertyName)
|
function PropertyDescriptor.fromRaw(data, className, propertyName)
|
||||||
|
local key, value = next(data.DataType)
|
||||||
|
|
||||||
return setmetatable({
|
return setmetatable({
|
||||||
|
-- The meanings of the key and value in DataType differ when the type of
|
||||||
|
-- the property is Enum. When the property is of type Enum, the key is
|
||||||
|
-- the name of the type:
|
||||||
|
--
|
||||||
|
-- { Enum = "<name of enum>" }
|
||||||
|
--
|
||||||
|
-- When the property is not of type Enum, the value is the name of the
|
||||||
|
-- type:
|
||||||
|
--
|
||||||
|
-- { Value = "<data type>" }
|
||||||
|
dataType = key == "Enum" and key or value,
|
||||||
|
|
||||||
scriptability = data.Scriptability,
|
scriptability = data.Scriptability,
|
||||||
className = className,
|
className = className,
|
||||||
name = propertyName,
|
name = propertyName,
|
||||||
@@ -77,4 +91,4 @@ function PropertyDescriptor:write(instance, value)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
return PropertyDescriptor
|
return PropertyDescriptor
|
||||||
|
|||||||
@@ -251,6 +251,16 @@
|
|||||||
},
|
},
|
||||||
"ty": "String"
|
"ty": "String"
|
||||||
},
|
},
|
||||||
|
"Tags": {
|
||||||
|
"value": {
|
||||||
|
"Tags": [
|
||||||
|
"foo",
|
||||||
|
"con'fusion?!",
|
||||||
|
"bar"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"ty": "Tags"
|
||||||
|
},
|
||||||
"UDim": {
|
"UDim": {
|
||||||
"value": {
|
"value": {
|
||||||
"UDim": [
|
"UDim": [
|
||||||
|
|||||||
@@ -6,12 +6,10 @@ local CollectionService = game:GetService("CollectionService")
|
|||||||
return {
|
return {
|
||||||
Instance = {
|
Instance = {
|
||||||
Tags = {
|
Tags = {
|
||||||
read = function(instance, key)
|
read = function(instance)
|
||||||
local tagList = CollectionService:GetTags(instance)
|
return true, CollectionService:GetTags(instance)
|
||||||
|
|
||||||
return true, table.concat(tagList, "\0")
|
|
||||||
end,
|
end,
|
||||||
write = function(instance, key, value)
|
write = function(instance, _, value)
|
||||||
local existingTags = CollectionService:GetTags(instance)
|
local existingTags = CollectionService:GetTags(instance)
|
||||||
|
|
||||||
local unseenTags = {}
|
local unseenTags = {}
|
||||||
@@ -19,8 +17,7 @@ return {
|
|||||||
unseenTags[tag] = true
|
unseenTags[tag] = true
|
||||||
end
|
end
|
||||||
|
|
||||||
local tagList = string.split(value, "\0")
|
for _, tag in ipairs(value) do
|
||||||
for _, tag in ipairs(tagList) do
|
|
||||||
unseenTags[tag] = nil
|
unseenTags[tag] = nil
|
||||||
CollectionService:AddTag(instance, tag)
|
CollectionService:AddTag(instance, tag)
|
||||||
end
|
end
|
||||||
@@ -44,4 +41,4 @@ return {
|
|||||||
end,
|
end,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
40
plugin/src/App/Components/Studio/StudioPluginAction.lua
Normal file
40
plugin/src/App/Components/Studio/StudioPluginAction.lua
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
local Rojo = script:FindFirstAncestor("Rojo")
|
||||||
|
local Plugin = Rojo.Plugin
|
||||||
|
|
||||||
|
local Roact = require(Rojo.Roact)
|
||||||
|
|
||||||
|
local Dictionary = require(Plugin.Dictionary)
|
||||||
|
|
||||||
|
local StudioPluginContext = require(script.Parent.StudioPluginContext)
|
||||||
|
|
||||||
|
local e = Roact.createElement
|
||||||
|
|
||||||
|
local StudioPluginAction = Roact.Component:extend("StudioPluginAction")
|
||||||
|
|
||||||
|
function StudioPluginAction:init()
|
||||||
|
self.pluginAction = self.props.plugin:CreatePluginAction(
|
||||||
|
self.props.name, self.props.title, self.props.description, self.props.icon, self.props.bindable
|
||||||
|
)
|
||||||
|
|
||||||
|
self.pluginAction.Triggered:Connect(self.props.onTriggered)
|
||||||
|
end
|
||||||
|
|
||||||
|
function StudioPluginAction:render()
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
function StudioPluginAction:willUnmount()
|
||||||
|
self.pluginAction:Destroy()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function StudioPluginActionWrapper(props)
|
||||||
|
return e(StudioPluginContext.Consumer, {
|
||||||
|
render = function(plugin)
|
||||||
|
return e(StudioPluginAction, Dictionary.merge(props, {
|
||||||
|
plugin = plugin,
|
||||||
|
}))
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
|
return StudioPluginActionWrapper
|
||||||
@@ -44,6 +44,10 @@ function StudioToggleButton:didUpdate(lastProps)
|
|||||||
self.button.Enabled = self.props.enabled
|
self.button.Enabled = self.props.enabled
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if self.props.icon ~= lastProps.icon then
|
||||||
|
self.button.Icon = self.props.icon
|
||||||
|
end
|
||||||
|
|
||||||
if self.props.active ~= lastProps.active then
|
if self.props.active ~= lastProps.active then
|
||||||
self.button:SetActive(self.props.active)
|
self.button:SetActive(self.props.active)
|
||||||
end
|
end
|
||||||
@@ -63,4 +67,4 @@ local function StudioToggleButtonWrapper(props)
|
|||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
return StudioToggleButtonWrapper
|
return StudioToggleButtonWrapper
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ local function AddressEntry(props)
|
|||||||
layoutOrder = props.layoutOrder,
|
layoutOrder = props.layoutOrder,
|
||||||
}, {
|
}, {
|
||||||
Host = e("TextBox", {
|
Host = e("TextBox", {
|
||||||
Text = "",
|
Text = props.host or "",
|
||||||
Font = Enum.Font.Code,
|
Font = Enum.Font.Code,
|
||||||
TextSize = 18,
|
TextSize = 18,
|
||||||
TextColor3 = theme.AddressEntry.TextColor,
|
TextColor3 = theme.AddressEntry.TextColor,
|
||||||
@@ -32,6 +32,7 @@ local function AddressEntry(props)
|
|||||||
TextTransparency = props.transparency,
|
TextTransparency = props.transparency,
|
||||||
PlaceholderText = Config.defaultHost,
|
PlaceholderText = Config.defaultHost,
|
||||||
PlaceholderColor3 = theme.AddressEntry.PlaceholderColor,
|
PlaceholderColor3 = theme.AddressEntry.PlaceholderColor,
|
||||||
|
ClearTextOnFocus = false,
|
||||||
|
|
||||||
Size = UDim2.new(1, -(HOST_OFFSET + DIVIDER_WIDTH + PORT_WIDTH), 1, 0),
|
Size = UDim2.new(1, -(HOST_OFFSET + DIVIDER_WIDTH + PORT_WIDTH), 1, 0),
|
||||||
Position = UDim2.new(0, HOST_OFFSET, 0, 0),
|
Position = UDim2.new(0, HOST_OFFSET, 0, 0),
|
||||||
@@ -39,17 +40,22 @@ local function AddressEntry(props)
|
|||||||
ClipsDescendants = true,
|
ClipsDescendants = true,
|
||||||
BackgroundTransparency = 1,
|
BackgroundTransparency = 1,
|
||||||
|
|
||||||
[Roact.Ref] = props.hostRef,
|
[Roact.Change.Text] = function(object)
|
||||||
|
if props.onHostChange ~= nil then
|
||||||
|
props.onHostChange(object.Text)
|
||||||
|
end
|
||||||
|
end
|
||||||
}),
|
}),
|
||||||
|
|
||||||
Port = e("TextBox", {
|
Port = e("TextBox", {
|
||||||
Text = "",
|
Text = props.port or "",
|
||||||
Font = Enum.Font.Code,
|
Font = Enum.Font.Code,
|
||||||
TextSize = 18,
|
TextSize = 18,
|
||||||
TextColor3 = theme.AddressEntry.TextColor,
|
TextColor3 = theme.AddressEntry.TextColor,
|
||||||
TextTransparency = props.transparency,
|
TextTransparency = props.transparency,
|
||||||
PlaceholderText = Config.defaultPort,
|
PlaceholderText = Config.defaultPort,
|
||||||
PlaceholderColor3 = theme.AddressEntry.PlaceholderColor,
|
PlaceholderColor3 = theme.AddressEntry.PlaceholderColor,
|
||||||
|
ClearTextOnFocus = false,
|
||||||
|
|
||||||
Size = UDim2.new(0, PORT_WIDTH, 1, 0),
|
Size = UDim2.new(0, PORT_WIDTH, 1, 0),
|
||||||
Position = UDim2.new(1, 0, 0, 0),
|
Position = UDim2.new(1, 0, 0, 0),
|
||||||
@@ -58,12 +64,14 @@ local function AddressEntry(props)
|
|||||||
ClipsDescendants = true,
|
ClipsDescendants = true,
|
||||||
BackgroundTransparency = 1,
|
BackgroundTransparency = 1,
|
||||||
|
|
||||||
[Roact.Ref] = props.portRef,
|
|
||||||
|
|
||||||
[Roact.Change.Text] = function(object)
|
[Roact.Change.Text] = function(object)
|
||||||
local text = object.Text
|
local text = object.Text
|
||||||
text = text:gsub("%D", "")
|
text = text:gsub("%D", "")
|
||||||
object.Text = text
|
object.Text = text
|
||||||
|
|
||||||
|
if props.onPortChange ~= nil then
|
||||||
|
props.onPortChange(text)
|
||||||
|
end
|
||||||
end,
|
end,
|
||||||
}, {
|
}, {
|
||||||
Divider = e("Frame", {
|
Divider = e("Frame", {
|
||||||
@@ -80,11 +88,6 @@ end
|
|||||||
|
|
||||||
local NotConnectedPage = Roact.Component:extend("NotConnectedPage")
|
local NotConnectedPage = Roact.Component:extend("NotConnectedPage")
|
||||||
|
|
||||||
function NotConnectedPage:init()
|
|
||||||
self.hostRef = Roact.createRef()
|
|
||||||
self.portRef = Roact.createRef()
|
|
||||||
end
|
|
||||||
|
|
||||||
function NotConnectedPage:render()
|
function NotConnectedPage:render()
|
||||||
return Roact.createFragment({
|
return Roact.createFragment({
|
||||||
Header = e(Header, {
|
Header = e(Header, {
|
||||||
@@ -93,8 +96,10 @@ function NotConnectedPage:render()
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
AddressEntry = e(AddressEntry, {
|
AddressEntry = e(AddressEntry, {
|
||||||
hostRef = self.hostRef,
|
host = self.props.host,
|
||||||
portRef = self.portRef,
|
port = self.props.port,
|
||||||
|
onHostChange = self.props.onHostChange,
|
||||||
|
onPortChange = self.props.onPortChange,
|
||||||
transparency = self.props.transparency,
|
transparency = self.props.transparency,
|
||||||
layoutOrder = 2,
|
layoutOrder = 2,
|
||||||
}),
|
}),
|
||||||
@@ -117,15 +122,7 @@ function NotConnectedPage:render()
|
|||||||
style = "Solid",
|
style = "Solid",
|
||||||
transparency = self.props.transparency,
|
transparency = self.props.transparency,
|
||||||
layoutOrder = 2,
|
layoutOrder = 2,
|
||||||
onClick = function()
|
onClick = self.props.onConnect,
|
||||||
local hostText = self.hostRef.current.Text
|
|
||||||
local portText = self.portRef.current.Text
|
|
||||||
|
|
||||||
self.props.onConnect(
|
|
||||||
#hostText > 0 and hostText or Config.defaultHost,
|
|
||||||
#portText > 0 and portText or Config.defaultPort
|
|
||||||
)
|
|
||||||
end,
|
|
||||||
}),
|
}),
|
||||||
|
|
||||||
Layout = e("UIListLayout", {
|
Layout = e("UIListLayout", {
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ end
|
|||||||
local BRAND_COLOR = hexColor(0xE13835)
|
local BRAND_COLOR = hexColor(0xE13835)
|
||||||
|
|
||||||
local lightTheme = strict("LightTheme", {
|
local lightTheme = strict("LightTheme", {
|
||||||
BackgroundColor = hexColor(0xF0F0F0),
|
BackgroundColor = hexColor(0xFFFFFF),
|
||||||
Button = {
|
Button = {
|
||||||
Solid = {
|
Solid = {
|
||||||
ActionFillColor = hexColor(0xFFFFFF),
|
ActionFillColor = hexColor(0xFFFFFF),
|
||||||
@@ -67,7 +67,7 @@ local lightTheme = strict("LightTheme", {
|
|||||||
BackgroundColor = BRAND_COLOR,
|
BackgroundColor = BRAND_COLOR,
|
||||||
},
|
},
|
||||||
Inactive = {
|
Inactive = {
|
||||||
IconColor = hexColor(0xCACACA),
|
IconColor = hexColor(0xEEEEEE),
|
||||||
BorderColor = hexColor(0xAFAFAF),
|
BorderColor = hexColor(0xAFAFAF),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -77,11 +77,11 @@ local lightTheme = strict("LightTheme", {
|
|||||||
},
|
},
|
||||||
BorderedContainer = {
|
BorderedContainer = {
|
||||||
BorderColor = hexColor(0xCBCBCB),
|
BorderColor = hexColor(0xCBCBCB),
|
||||||
BackgroundColor = hexColor(0xE0E0E0),
|
BackgroundColor = hexColor(0xEEEEEE),
|
||||||
},
|
},
|
||||||
Spinner = {
|
Spinner = {
|
||||||
ForegroundColor = BRAND_COLOR,
|
ForegroundColor = BRAND_COLOR,
|
||||||
BackgroundColor = hexColor(0xE0E0E0),
|
BackgroundColor = hexColor(0xEEEEEE),
|
||||||
},
|
},
|
||||||
ConnectionDetails = {
|
ConnectionDetails = {
|
||||||
ProjectNameColor = hexColor(0x00000),
|
ProjectNameColor = hexColor(0x00000),
|
||||||
@@ -108,7 +108,7 @@ local lightTheme = strict("LightTheme", {
|
|||||||
})
|
})
|
||||||
|
|
||||||
local darkTheme = strict("DarkTheme", {
|
local darkTheme = strict("DarkTheme", {
|
||||||
BackgroundColor = hexColor(0x272727),
|
BackgroundColor = hexColor(0x2E2E2E),
|
||||||
Button = {
|
Button = {
|
||||||
Solid = {
|
Solid = {
|
||||||
ActionFillColor = hexColor(0xFFFFFF),
|
ActionFillColor = hexColor(0xFFFFFF),
|
||||||
@@ -147,15 +147,15 @@ local darkTheme = strict("DarkTheme", {
|
|||||||
},
|
},
|
||||||
AddressEntry = {
|
AddressEntry = {
|
||||||
TextColor = hexColor(0xFFFFFF),
|
TextColor = hexColor(0xFFFFFF),
|
||||||
PlaceholderColor = hexColor(0x717171)
|
PlaceholderColor = hexColor(0x8B8B8B)
|
||||||
},
|
},
|
||||||
BorderedContainer = {
|
BorderedContainer = {
|
||||||
BorderColor = hexColor(0x535353),
|
BorderColor = hexColor(0x535353),
|
||||||
BackgroundColor = hexColor(0x323232),
|
BackgroundColor = hexColor(0x2B2B2B),
|
||||||
},
|
},
|
||||||
Spinner = {
|
Spinner = {
|
||||||
ForegroundColor = BRAND_COLOR,
|
ForegroundColor = BRAND_COLOR,
|
||||||
BackgroundColor = hexColor(0x323232),
|
BackgroundColor = hexColor(0x2B2B2B),
|
||||||
},
|
},
|
||||||
ConnectionDetails = {
|
ConnectionDetails = {
|
||||||
ProjectNameColor = hexColor(0xFFFFFF),
|
ProjectNameColor = hexColor(0xFFFFFF),
|
||||||
@@ -236,4 +236,4 @@ return {
|
|||||||
StudioProvider = StudioProvider,
|
StudioProvider = StudioProvider,
|
||||||
Consumer = Context.Consumer,
|
Consumer = Context.Consumer,
|
||||||
with = with,
|
with = with,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ local Theme = require(script.Theme)
|
|||||||
local PluginSettings = require(script.PluginSettings)
|
local PluginSettings = require(script.PluginSettings)
|
||||||
|
|
||||||
local Page = require(script.Page)
|
local Page = require(script.Page)
|
||||||
|
local StudioPluginAction = require(script.Components.Studio.StudioPluginAction)
|
||||||
local StudioToolbar = require(script.Components.Studio.StudioToolbar)
|
local StudioToolbar = require(script.Components.Studio.StudioToolbar)
|
||||||
local StudioToggleButton = require(script.Components.Studio.StudioToggleButton)
|
local StudioToggleButton = require(script.Components.Studio.StudioToggleButton)
|
||||||
local StudioPluginGui = require(script.Components.Studio.StudioPluginGui)
|
local StudioPluginGui = require(script.Components.Studio.StudioPluginGui)
|
||||||
@@ -37,13 +38,34 @@ local App = Roact.Component:extend("App")
|
|||||||
function App:init()
|
function App:init()
|
||||||
preloadAssets()
|
preloadAssets()
|
||||||
|
|
||||||
|
self.host, self.setHost = Roact.createBinding("")
|
||||||
|
self.port, self.setPort = Roact.createBinding("")
|
||||||
|
|
||||||
self:setState({
|
self:setState({
|
||||||
appStatus = AppStatus.NotConnected,
|
appStatus = AppStatus.NotConnected,
|
||||||
guiEnabled = false,
|
guiEnabled = false,
|
||||||
|
toolbarIcon = Assets.Images.PluginButton,
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
function App:startSession(host, port, sessionOptions)
|
function App:getHostAndPort()
|
||||||
|
local host = self.host:getValue()
|
||||||
|
local port = self.port:getValue()
|
||||||
|
|
||||||
|
local host = if #host > 0 then host else Config.defaultHost
|
||||||
|
local port = if #port > 0 then port else Config.defaultPort
|
||||||
|
|
||||||
|
return host, port
|
||||||
|
end
|
||||||
|
|
||||||
|
function App:startSession()
|
||||||
|
local host, port = self:getHostAndPort()
|
||||||
|
|
||||||
|
local sessionOptions = {
|
||||||
|
openScriptsExternally = self.props.settings:get("openScriptsExternally"),
|
||||||
|
twoWaySync = self.props.settings:get("twoWaySync"),
|
||||||
|
}
|
||||||
|
|
||||||
local baseUrl = ("http://%s:%s"):format(host, port)
|
local baseUrl = ("http://%s:%s"):format(host, port)
|
||||||
local apiContext = ApiContext.new(baseUrl)
|
local apiContext = ApiContext.new(baseUrl)
|
||||||
|
|
||||||
@@ -57,6 +79,7 @@ function App:startSession(host, port, sessionOptions)
|
|||||||
if status == ServeSession.Status.Connecting then
|
if status == ServeSession.Status.Connecting then
|
||||||
self:setState({
|
self:setState({
|
||||||
appStatus = AppStatus.Connecting,
|
appStatus = AppStatus.Connecting,
|
||||||
|
toolbarIcon = Assets.Images.PluginButton,
|
||||||
})
|
})
|
||||||
elseif status == ServeSession.Status.Connected then
|
elseif status == ServeSession.Status.Connected then
|
||||||
local address = ("%s:%s"):format(host, port)
|
local address = ("%s:%s"):format(host, port)
|
||||||
@@ -64,7 +87,10 @@ function App:startSession(host, port, sessionOptions)
|
|||||||
appStatus = AppStatus.Connected,
|
appStatus = AppStatus.Connected,
|
||||||
projectName = details,
|
projectName = details,
|
||||||
address = address,
|
address = address,
|
||||||
|
toolbarIcon = Assets.Images.PluginButtonConnected,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Log.info("Connected to session '{}' at {}", details, address)
|
||||||
elseif status == ServeSession.Status.Disconnected then
|
elseif status == ServeSession.Status.Disconnected then
|
||||||
self.serveSession = nil
|
self.serveSession = nil
|
||||||
|
|
||||||
@@ -76,11 +102,15 @@ function App:startSession(host, port, sessionOptions)
|
|||||||
self:setState({
|
self:setState({
|
||||||
appStatus = AppStatus.Error,
|
appStatus = AppStatus.Error,
|
||||||
errorMessage = tostring(details),
|
errorMessage = tostring(details),
|
||||||
|
toolbarIcon = Assets.Images.PluginButtonWarning,
|
||||||
})
|
})
|
||||||
else
|
else
|
||||||
self:setState({
|
self:setState({
|
||||||
appStatus = AppStatus.NotConnected,
|
appStatus = AppStatus.NotConnected,
|
||||||
|
toolbarIcon = Assets.Images.PluginButton,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Log.info("Disconnected session")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end)
|
end)
|
||||||
@@ -90,6 +120,22 @@ function App:startSession(host, port, sessionOptions)
|
|||||||
self.serveSession = serveSession
|
self.serveSession = serveSession
|
||||||
end
|
end
|
||||||
|
|
||||||
|
function App:endSession()
|
||||||
|
if self.serveSession == nil then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
Log.trace("Disconnecting session")
|
||||||
|
|
||||||
|
self.serveSession:stop()
|
||||||
|
self.serveSession = nil
|
||||||
|
self:setState({
|
||||||
|
appStatus = AppStatus.NotConnected,
|
||||||
|
})
|
||||||
|
|
||||||
|
Log.trace("Session terminated by user")
|
||||||
|
end
|
||||||
|
|
||||||
function App:render()
|
function App:render()
|
||||||
local pluginName = "Rojo " .. Version.display(Config.version)
|
local pluginName = "Rojo " .. Version.display(Config.version)
|
||||||
|
|
||||||
@@ -108,119 +154,160 @@ function App:render()
|
|||||||
value = self.props.plugin,
|
value = self.props.plugin,
|
||||||
}, {
|
}, {
|
||||||
e(Theme.StudioProvider, nil, {
|
e(Theme.StudioProvider, nil, {
|
||||||
e(PluginSettings.StudioProvider, {
|
gui = e(StudioPluginGui, {
|
||||||
plugin = self.props.plugin,
|
id = pluginName,
|
||||||
|
title = pluginName,
|
||||||
|
active = self.state.guiEnabled,
|
||||||
|
|
||||||
|
initDockState = Enum.InitialDockState.Right,
|
||||||
|
initEnabled = false,
|
||||||
|
overridePreviousState = false,
|
||||||
|
floatingSize = Vector2.new(300, 200),
|
||||||
|
minimumSize = Vector2.new(300, 200),
|
||||||
|
|
||||||
|
zIndexBehavior = Enum.ZIndexBehavior.Sibling,
|
||||||
|
|
||||||
|
onInitialState = function(initialState)
|
||||||
|
self:setState({
|
||||||
|
guiEnabled = initialState,
|
||||||
|
})
|
||||||
|
end,
|
||||||
|
|
||||||
|
onClose = function()
|
||||||
|
self:setState({
|
||||||
|
guiEnabled = false,
|
||||||
|
})
|
||||||
|
end,
|
||||||
}, {
|
}, {
|
||||||
gui = e(StudioPluginGui, {
|
NotConnectedPage = createPageElement(AppStatus.NotConnected, {
|
||||||
id = pluginName,
|
host = self.host,
|
||||||
title = pluginName,
|
onHostChange = self.setHost,
|
||||||
active = self.state.guiEnabled,
|
port = self.port,
|
||||||
|
onPortChange = self.setPort,
|
||||||
|
|
||||||
initDockState = Enum.InitialDockState.Right,
|
onConnect = function()
|
||||||
initEnabled = false,
|
self:startSession()
|
||||||
overridePreviousState = false,
|
end,
|
||||||
floatingSize = Vector2.new(300, 200),
|
|
||||||
minimumSize = Vector2.new(300, 200),
|
|
||||||
|
|
||||||
zIndexBehavior = Enum.ZIndexBehavior.Sibling,
|
onNavigateSettings = function()
|
||||||
|
|
||||||
onInitialState = function(initialState)
|
|
||||||
self:setState({
|
self:setState({
|
||||||
guiEnabled = initialState,
|
appStatus = AppStatus.Settings,
|
||||||
})
|
})
|
||||||
end,
|
end,
|
||||||
|
}),
|
||||||
|
|
||||||
|
Connecting = createPageElement(AppStatus.Connecting),
|
||||||
|
|
||||||
|
Connected = createPageElement(AppStatus.Connected, {
|
||||||
|
projectName = self.state.projectName,
|
||||||
|
address = self.state.address,
|
||||||
|
|
||||||
|
onDisconnect = function()
|
||||||
|
self:endSession()
|
||||||
|
end,
|
||||||
|
}),
|
||||||
|
|
||||||
|
Settings = createPageElement(AppStatus.Settings, {
|
||||||
|
onBack = function()
|
||||||
|
self:setState({
|
||||||
|
appStatus = AppStatus.NotConnected,
|
||||||
|
})
|
||||||
|
end,
|
||||||
|
}),
|
||||||
|
|
||||||
|
Error = createPageElement(AppStatus.Error, {
|
||||||
|
errorMessage = self.state.errorMessage,
|
||||||
|
|
||||||
onClose = function()
|
onClose = function()
|
||||||
self:setState({
|
self:setState({
|
||||||
guiEnabled = false,
|
appStatus = AppStatus.NotConnected,
|
||||||
|
toolbarIcon = Assets.Images.PluginButton,
|
||||||
})
|
})
|
||||||
end,
|
end,
|
||||||
}, {
|
|
||||||
NotConnectedPage = PluginSettings.with(function(settings)
|
|
||||||
return createPageElement(AppStatus.NotConnected, {
|
|
||||||
onConnect = function(host, port)
|
|
||||||
self:startSession(host, port, {
|
|
||||||
openScriptsExternally = settings:get("openScriptsExternally"),
|
|
||||||
twoWaySync = settings:get("twoWaySync"),
|
|
||||||
})
|
|
||||||
end,
|
|
||||||
|
|
||||||
onNavigateSettings = function()
|
|
||||||
self:setState({
|
|
||||||
appStatus = AppStatus.Settings,
|
|
||||||
})
|
|
||||||
end,
|
|
||||||
})
|
|
||||||
end),
|
|
||||||
|
|
||||||
Connecting = createPageElement(AppStatus.Connecting),
|
|
||||||
|
|
||||||
Connected = createPageElement(AppStatus.Connected, {
|
|
||||||
projectName = self.state.projectName,
|
|
||||||
address = self.state.address,
|
|
||||||
|
|
||||||
onDisconnect = function()
|
|
||||||
Log.trace("Disconnecting session")
|
|
||||||
|
|
||||||
self.serveSession:stop()
|
|
||||||
self.serveSession = nil
|
|
||||||
self:setState({
|
|
||||||
appStatus = AppStatus.NotConnected,
|
|
||||||
})
|
|
||||||
|
|
||||||
Log.trace("Session terminated by user")
|
|
||||||
end,
|
|
||||||
}),
|
|
||||||
|
|
||||||
Settings = createPageElement(AppStatus.Settings, {
|
|
||||||
onBack = function()
|
|
||||||
self:setState({
|
|
||||||
appStatus = AppStatus.NotConnected,
|
|
||||||
})
|
|
||||||
end,
|
|
||||||
}),
|
|
||||||
|
|
||||||
Error = createPageElement(AppStatus.Error, {
|
|
||||||
errorMessage = self.state.errorMessage,
|
|
||||||
|
|
||||||
onClose = function()
|
|
||||||
self:setState({
|
|
||||||
appStatus = AppStatus.NotConnected,
|
|
||||||
})
|
|
||||||
end,
|
|
||||||
}),
|
|
||||||
|
|
||||||
Background = Theme.with(function(theme)
|
|
||||||
return e("Frame", {
|
|
||||||
Size = UDim2.new(1, 0, 1, 0),
|
|
||||||
BackgroundColor3 = theme.BackgroundColor,
|
|
||||||
ZIndex = 0,
|
|
||||||
BorderSizePixel = 0,
|
|
||||||
})
|
|
||||||
end),
|
|
||||||
}),
|
}),
|
||||||
|
|
||||||
toolbar = e(StudioToolbar, {
|
Background = Theme.with(function(theme)
|
||||||
name = pluginName,
|
return e("Frame", {
|
||||||
}, {
|
Size = UDim2.new(1, 0, 1, 0),
|
||||||
button = e(StudioToggleButton, {
|
BackgroundColor3 = theme.BackgroundColor,
|
||||||
name = "Rojo",
|
ZIndex = 0,
|
||||||
tooltip = "Show or hide the Rojo panel",
|
BorderSizePixel = 0,
|
||||||
icon = Assets.Images.PluginButton,
|
|
||||||
active = self.state.guiEnabled,
|
|
||||||
enabled = true,
|
|
||||||
onClick = function()
|
|
||||||
self:setState(function(state)
|
|
||||||
return {
|
|
||||||
guiEnabled = not state.guiEnabled,
|
|
||||||
}
|
|
||||||
end)
|
|
||||||
end,
|
|
||||||
})
|
})
|
||||||
}),
|
end),
|
||||||
|
}),
|
||||||
|
|
||||||
|
toggleAction = e(StudioPluginAction, {
|
||||||
|
name = "RojoConnection",
|
||||||
|
title = "Rojo: Connect/Disconnect",
|
||||||
|
description = "Toggles the server for a Rojo sync session",
|
||||||
|
icon = Assets.Images.PluginButton,
|
||||||
|
bindable = true,
|
||||||
|
onTriggered = function()
|
||||||
|
if self.serveSession == nil or self.serveSession:getStatus() == ServeSession.Status.NotStarted then
|
||||||
|
self:startSession()
|
||||||
|
elseif self.serveSession ~= nil and self.serveSession:getStatus() == ServeSession.Status.Connected then
|
||||||
|
self:endSession()
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
}),
|
||||||
|
|
||||||
|
connectAction = e(StudioPluginAction, {
|
||||||
|
name = "RojoConnect",
|
||||||
|
title = "Rojo: Connect",
|
||||||
|
description = "Connects the server for a Rojo sync session",
|
||||||
|
icon = Assets.Images.PluginButton,
|
||||||
|
bindable = true,
|
||||||
|
onTriggered = function()
|
||||||
|
if self.serveSession == nil or self.serveSession:getStatus() == ServeSession.Status.NotStarted then
|
||||||
|
self:startSession()
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
}),
|
||||||
|
|
||||||
|
disconnectAction = e(StudioPluginAction, {
|
||||||
|
name = "RojoDisconnect",
|
||||||
|
title = "Rojo: Disconnect",
|
||||||
|
description = "Disconnects the server for a Rojo sync session",
|
||||||
|
icon = Assets.Images.PluginButton,
|
||||||
|
bindable = true,
|
||||||
|
onTriggered = function()
|
||||||
|
if self.serveSession ~= nil and self.serveSession:getStatus() == ServeSession.Status.Connected then
|
||||||
|
self:endSession()
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
}),
|
||||||
|
|
||||||
|
toolbar = e(StudioToolbar, {
|
||||||
|
name = pluginName,
|
||||||
|
}, {
|
||||||
|
button = e(StudioToggleButton, {
|
||||||
|
name = "Rojo",
|
||||||
|
tooltip = "Show or hide the Rojo panel",
|
||||||
|
icon = self.state.toolbarIcon,
|
||||||
|
active = self.state.guiEnabled,
|
||||||
|
enabled = true,
|
||||||
|
onClick = function()
|
||||||
|
self:setState(function(state)
|
||||||
|
return {
|
||||||
|
guiEnabled = not state.guiEnabled,
|
||||||
|
}
|
||||||
|
end)
|
||||||
|
end,
|
||||||
|
})
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
return App
|
return function(props)
|
||||||
|
return e(PluginSettings.StudioProvider, {
|
||||||
|
plugin = props.plugin,
|
||||||
|
}, {
|
||||||
|
App = PluginSettings.with(function(settings)
|
||||||
|
local settingsProps = Dictionary.merge(props, {
|
||||||
|
settings = settings,
|
||||||
|
})
|
||||||
|
return e(App, settingsProps)
|
||||||
|
end),
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|||||||
@@ -18,6 +18,8 @@ local Assets = {
|
|||||||
Images = {
|
Images = {
|
||||||
Logo = "rbxassetid://5990772764",
|
Logo = "rbxassetid://5990772764",
|
||||||
PluginButton = "rbxassetid://3405341609",
|
PluginButton = "rbxassetid://3405341609",
|
||||||
|
PluginButtonConnected = "rbxassetid://9529783993",
|
||||||
|
PluginButtonWarning = "rbxassetid://9529784530",
|
||||||
Icons = {
|
Icons = {
|
||||||
Close = "rbxassetid://6012985953",
|
Close = "rbxassetid://6012985953",
|
||||||
Back = "rbxassetid://6017213752",
|
Back = "rbxassetid://6017213752",
|
||||||
|
|||||||
40
plugin/src/ChangeBatcher/createPatchSet.lua
Normal file
40
plugin/src/ChangeBatcher/createPatchSet.lua
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
--[[
|
||||||
|
Take an InstanceMap and a dictionary mapping instances to sets of property
|
||||||
|
names. Populate a patch with the encoded values of all the given properties
|
||||||
|
on all the given instances (or, if any changes set Parent to nil, removals
|
||||||
|
of instances) and return the patch.
|
||||||
|
]]
|
||||||
|
|
||||||
|
local Log = require(script.Parent.Parent.Parent.Log)
|
||||||
|
|
||||||
|
local PatchSet = require(script.Parent.Parent.PatchSet)
|
||||||
|
|
||||||
|
local encodePatchUpdate = require(script.Parent.encodePatchUpdate)
|
||||||
|
|
||||||
|
return function(instanceMap, propertyChanges)
|
||||||
|
local patch = PatchSet.newEmpty()
|
||||||
|
|
||||||
|
for instance, properties in pairs(propertyChanges) do
|
||||||
|
local instanceId = instanceMap.fromInstances[instance]
|
||||||
|
|
||||||
|
if instanceId == nil then
|
||||||
|
Log.warn("Ignoring change for instance {:?} as it is unknown to Rojo", instance)
|
||||||
|
continue
|
||||||
|
end
|
||||||
|
|
||||||
|
if properties.Parent then
|
||||||
|
if instance.Parent == nil then
|
||||||
|
table.insert(patch.removed, instanceId)
|
||||||
|
else
|
||||||
|
Log.warn("Cannot sync non-nil Parent property changes yet")
|
||||||
|
end
|
||||||
|
else
|
||||||
|
local update = encodePatchUpdate(instance, instanceId, properties)
|
||||||
|
table.insert(patch.updated, update)
|
||||||
|
end
|
||||||
|
|
||||||
|
propertyChanges[instance] = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
return patch
|
||||||
|
end
|
||||||
74
plugin/src/ChangeBatcher/createPatchSet.spec.lua
Normal file
74
plugin/src/ChangeBatcher/createPatchSet.spec.lua
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
return function()
|
||||||
|
local PatchSet = require(script.Parent.Parent.PatchSet)
|
||||||
|
local InstanceMap = require(script.Parent.Parent.InstanceMap)
|
||||||
|
|
||||||
|
local createPatchSet = require(script.Parent.createPatchSet)
|
||||||
|
|
||||||
|
it("should return a patch", function()
|
||||||
|
local patch = createPatchSet(InstanceMap.new(), {})
|
||||||
|
|
||||||
|
assert(PatchSet.validate(patch))
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should contain updates for every instance with property changes", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
|
||||||
|
local part1 = Instance.new("Part")
|
||||||
|
instanceMap:insert("PART_1", part1)
|
||||||
|
|
||||||
|
local part2 = Instance.new("Part")
|
||||||
|
instanceMap:insert("PART_2", part2)
|
||||||
|
|
||||||
|
local changes = {
|
||||||
|
[part1] = {
|
||||||
|
Position = true,
|
||||||
|
Size = true,
|
||||||
|
Color = true,
|
||||||
|
},
|
||||||
|
[part2] = {
|
||||||
|
CFrame = true,
|
||||||
|
Velocity = true,
|
||||||
|
Transparency = true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
local patch = createPatchSet(instanceMap, changes)
|
||||||
|
|
||||||
|
expect(#patch.updated).to.equal(2)
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should not contain any updates for removed instances", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
|
||||||
|
local part1 = Instance.new("Part")
|
||||||
|
instanceMap:insert("PART_1", part1)
|
||||||
|
|
||||||
|
local changes = {
|
||||||
|
[part1] = {
|
||||||
|
Parent = true,
|
||||||
|
Position = true,
|
||||||
|
Size = true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
local patch = createPatchSet(instanceMap, changes)
|
||||||
|
|
||||||
|
expect(#patch.removed).to.equal(1)
|
||||||
|
expect(#patch.updated).to.equal(0)
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should remove instances from the property change table", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
|
||||||
|
local part1 = Instance.new("Part")
|
||||||
|
instanceMap:insert("PART_1", part1)
|
||||||
|
|
||||||
|
local changes = {
|
||||||
|
[part1] = {},
|
||||||
|
}
|
||||||
|
|
||||||
|
createPatchSet(instanceMap, changes)
|
||||||
|
|
||||||
|
expect(next(changes)).to.equal(nil)
|
||||||
|
end)
|
||||||
|
end
|
||||||
39
plugin/src/ChangeBatcher/encodePatchUpdate.lua
Normal file
39
plugin/src/ChangeBatcher/encodePatchUpdate.lua
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
local Log = require(script.Parent.Parent.Parent.Log)
|
||||||
|
local RbxDom = require(script.Parent.Parent.Parent.RbxDom)
|
||||||
|
|
||||||
|
local encodeProperty = require(script.Parent.encodeProperty)
|
||||||
|
|
||||||
|
return function(instance, instanceId, properties)
|
||||||
|
local update = {
|
||||||
|
id = instanceId,
|
||||||
|
changedProperties = {},
|
||||||
|
}
|
||||||
|
|
||||||
|
for propertyName in pairs(properties) do
|
||||||
|
if propertyName == "Name" then
|
||||||
|
update.changedName = instance.Name
|
||||||
|
else
|
||||||
|
local descriptor = RbxDom.findCanonicalPropertyDescriptor(instance.ClassName, propertyName)
|
||||||
|
|
||||||
|
if not descriptor then
|
||||||
|
Log.debug("Could not sync back property {:?}.{}", instance, propertyName)
|
||||||
|
continue
|
||||||
|
end
|
||||||
|
|
||||||
|
local encodeSuccess, encodeResult = encodeProperty(instance, propertyName, descriptor)
|
||||||
|
|
||||||
|
if not encodeSuccess then
|
||||||
|
Log.debug("Could not sync back property {:?}.{}: {}", instance, propertyName, encodeResult)
|
||||||
|
continue
|
||||||
|
end
|
||||||
|
|
||||||
|
update.changedProperties[propertyName] = encodeResult
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if next(update.changedProperties) == nil and update.changedName == nil then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
return update
|
||||||
|
end
|
||||||
62
plugin/src/ChangeBatcher/encodePatchUpdate.spec.lua
Normal file
62
plugin/src/ChangeBatcher/encodePatchUpdate.spec.lua
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
return function()
|
||||||
|
local encodePatchUpdate = require(script.Parent.encodePatchUpdate)
|
||||||
|
|
||||||
|
it("should return an update when there are property changes", function()
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
local properties = {
|
||||||
|
CFrame = true,
|
||||||
|
Color = true,
|
||||||
|
}
|
||||||
|
local update = encodePatchUpdate(part, "PART", properties)
|
||||||
|
|
||||||
|
expect(update.id).to.equal("PART")
|
||||||
|
expect(update.changedProperties.CFrame).to.be.ok()
|
||||||
|
expect(update.changedProperties.Color).to.be.ok()
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should return nil when there are no property changes", function()
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
local properties = {
|
||||||
|
NonExistentProperty = true,
|
||||||
|
}
|
||||||
|
local update = encodePatchUpdate(part, "PART", properties)
|
||||||
|
|
||||||
|
expect(update).to.equal(nil)
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should set changedName in the update when the instance's Name changes", function()
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
local properties = {
|
||||||
|
Name = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
part.Name = "We'reGettingToTheCoolPart"
|
||||||
|
|
||||||
|
local update = encodePatchUpdate(part, "PART", properties)
|
||||||
|
|
||||||
|
expect(update.changedName).to.equal("We'reGettingToTheCoolPart")
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should correctly encode property values", function()
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
local properties = {
|
||||||
|
Position = true,
|
||||||
|
Color = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
part.Position = Vector3.new(0, 100, 0)
|
||||||
|
part.Color = Color3.new(0.8, 0.2, 0.9)
|
||||||
|
|
||||||
|
local update = encodePatchUpdate(part, "PART", properties)
|
||||||
|
local position = update.changedProperties.Position
|
||||||
|
local color = update.changedProperties.Color
|
||||||
|
|
||||||
|
expect(position.Vector3[1]).to.equal(0)
|
||||||
|
expect(position.Vector3[2]).to.equal(100)
|
||||||
|
expect(position.Vector3[3]).to.equal(0)
|
||||||
|
|
||||||
|
expect(color.Color3[1]).to.be.near(0.8, 0.01)
|
||||||
|
expect(color.Color3[2]).to.be.near(0.2, 0.01)
|
||||||
|
expect(color.Color3[3]).to.be.near(0.9, 0.01)
|
||||||
|
end)
|
||||||
|
end
|
||||||
21
plugin/src/ChangeBatcher/encodeProperty.lua
Normal file
21
plugin/src/ChangeBatcher/encodeProperty.lua
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
local Log = require(script.Parent.Parent.Parent.Log)
|
||||||
|
local RbxDom = require(script.Parent.Parent.Parent.RbxDom)
|
||||||
|
|
||||||
|
return function(instance, propertyName, propertyDescriptor)
|
||||||
|
local readSuccess, readResult = propertyDescriptor:read(instance)
|
||||||
|
|
||||||
|
if not readSuccess then
|
||||||
|
Log.warn("Could not sync back property {:?}.{}: {}", instance, propertyName, readResult)
|
||||||
|
return false, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
local dataType = propertyDescriptor.dataType
|
||||||
|
local encodeSuccess, encodeResult = RbxDom.EncodedValue.encode(readResult, dataType)
|
||||||
|
|
||||||
|
if not encodeSuccess then
|
||||||
|
Log.warn("Could not sync back property {:?}.{}: {}", instance, propertyName, encodeResult)
|
||||||
|
return false, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
return true, encodeResult
|
||||||
|
end
|
||||||
81
plugin/src/ChangeBatcher/init.lua
Normal file
81
plugin/src/ChangeBatcher/init.lua
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
--[[
|
||||||
|
The ChangeBatcher is responsible for collecting and dispatching changes made
|
||||||
|
to tracked instances during two-way sync.
|
||||||
|
]]
|
||||||
|
|
||||||
|
local RunService = game:GetService("RunService")
|
||||||
|
|
||||||
|
local PatchSet = require(script.Parent.PatchSet)
|
||||||
|
|
||||||
|
local createPatchSet = require(script.createPatchSet)
|
||||||
|
|
||||||
|
local ChangeBatcher = {}
|
||||||
|
ChangeBatcher.__index = ChangeBatcher
|
||||||
|
|
||||||
|
local BATCH_INTERVAL = 0.2
|
||||||
|
|
||||||
|
function ChangeBatcher.new(instanceMap, onChangesFlushed)
|
||||||
|
local self
|
||||||
|
|
||||||
|
local renderSteppedConnection = RunService.RenderStepped:Connect(function(dt)
|
||||||
|
self:__cycle(dt)
|
||||||
|
end)
|
||||||
|
|
||||||
|
self = setmetatable({
|
||||||
|
__accumulator = 0,
|
||||||
|
__renderSteppedConnection = renderSteppedConnection,
|
||||||
|
__instanceMap = instanceMap,
|
||||||
|
__onChangesFlushed = onChangesFlushed,
|
||||||
|
__pendingPropertyChanges = {},
|
||||||
|
}, ChangeBatcher)
|
||||||
|
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
function ChangeBatcher:stop()
|
||||||
|
self.__renderSteppedConnection:Disconnect()
|
||||||
|
self.__pendingPropertyChanges = {}
|
||||||
|
end
|
||||||
|
|
||||||
|
function ChangeBatcher:add(instance, propertyName)
|
||||||
|
local properties = self.__pendingPropertyChanges[instance]
|
||||||
|
|
||||||
|
if not properties then
|
||||||
|
properties = {}
|
||||||
|
self.__pendingPropertyChanges[instance] = properties
|
||||||
|
end
|
||||||
|
|
||||||
|
properties[propertyName] = true
|
||||||
|
end
|
||||||
|
|
||||||
|
function ChangeBatcher:__cycle(dt)
|
||||||
|
self.__accumulator += dt
|
||||||
|
|
||||||
|
if self.__accumulator >= BATCH_INTERVAL then
|
||||||
|
self.__accumulator -= BATCH_INTERVAL
|
||||||
|
|
||||||
|
local patch = self:__flush()
|
||||||
|
|
||||||
|
if patch then
|
||||||
|
self.__onChangesFlushed(patch)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
self.__instanceMap:unpauseAllInstances()
|
||||||
|
end
|
||||||
|
|
||||||
|
function ChangeBatcher:__flush()
|
||||||
|
if next(self.__pendingPropertyChanges) == nil then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
local patch = createPatchSet(self.__instanceMap, self.__pendingPropertyChanges)
|
||||||
|
|
||||||
|
if PatchSet.isEmpty(patch) then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
return patch
|
||||||
|
end
|
||||||
|
|
||||||
|
return ChangeBatcher
|
||||||
101
plugin/src/ChangeBatcher/init.spec.lua
Normal file
101
plugin/src/ChangeBatcher/init.spec.lua
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
return function()
|
||||||
|
local ChangeBatcher = require(script.Parent)
|
||||||
|
local InstanceMap = require(script.Parent.Parent.InstanceMap)
|
||||||
|
local PatchSet = require(script.Parent.Parent.PatchSet)
|
||||||
|
|
||||||
|
local noop = function() end
|
||||||
|
|
||||||
|
describe("new", function()
|
||||||
|
it("should create a new ChangeBatcher", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
local changeBatcher = ChangeBatcher.new(instanceMap, noop)
|
||||||
|
|
||||||
|
expect(changeBatcher.__pendingPropertyChanges).to.be.a("table")
|
||||||
|
expect(next(changeBatcher.__pendingPropertyChanges)).to.equal(nil)
|
||||||
|
expect(changeBatcher.__onChangesFlushed).to.equal(noop)
|
||||||
|
expect(changeBatcher.__instanceMap).to.equal(instanceMap)
|
||||||
|
expect(typeof(changeBatcher.__renderSteppedConnection)).to.equal("RBXScriptConnection")
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
|
||||||
|
describe("stop", function()
|
||||||
|
it("should disconnect the RenderStepped connection", function()
|
||||||
|
local changeBatcher = ChangeBatcher.new(InstanceMap.new(), noop)
|
||||||
|
|
||||||
|
changeBatcher:stop()
|
||||||
|
|
||||||
|
expect(changeBatcher.__renderSteppedConnection.Connected).to.equal(false)
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
|
||||||
|
describe("add", function()
|
||||||
|
it("should add property changes to be considered for the current batch", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
local changeBatcher = ChangeBatcher.new(instanceMap, noop)
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
|
||||||
|
instanceMap:insert("PART", part)
|
||||||
|
changeBatcher:add(part, "Name")
|
||||||
|
|
||||||
|
local properties = changeBatcher.__pendingPropertyChanges[part]
|
||||||
|
|
||||||
|
expect(properties).to.be.a("table")
|
||||||
|
expect(properties.Name).to.be.ok()
|
||||||
|
|
||||||
|
changeBatcher:add(part, "Position")
|
||||||
|
expect(properties.Position).to.be.ok()
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
|
||||||
|
describe("__cycle", function()
|
||||||
|
it("should immediately unpause any paused instances after each cycle", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
local changeBatcher = ChangeBatcher.new(instanceMap, noop)
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
|
||||||
|
instanceMap.pausedUpdateInstances[part] = true
|
||||||
|
|
||||||
|
changeBatcher:__cycle(0)
|
||||||
|
|
||||||
|
expect(instanceMap.pausedUpdateInstances[part]).to.equal(nil)
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
|
||||||
|
describe("__flush", function()
|
||||||
|
it("should return nil when there are no changes to process", function()
|
||||||
|
local changeBatcher = ChangeBatcher.new(InstanceMap.new(), noop)
|
||||||
|
expect(changeBatcher:__flush()).to.equal(nil)
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should return a patch when there are changes to process and the resulting patch is non-empty", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
local changeBatcher = ChangeBatcher.new(instanceMap, noop)
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
|
||||||
|
instanceMap:insert("PART", part)
|
||||||
|
|
||||||
|
changeBatcher.__pendingPropertyChanges[part] = {
|
||||||
|
Position = true,
|
||||||
|
Name = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
local patch = changeBatcher:__flush()
|
||||||
|
|
||||||
|
assert(PatchSet.validate(patch))
|
||||||
|
end)
|
||||||
|
|
||||||
|
it("should return nil when there are changes to process and the resulting patch is empty", function()
|
||||||
|
local instanceMap = InstanceMap.new()
|
||||||
|
local changeBatcher = ChangeBatcher.new(instanceMap, noop)
|
||||||
|
local part = Instance.new("Part")
|
||||||
|
|
||||||
|
instanceMap:insert("PART", part)
|
||||||
|
|
||||||
|
changeBatcher.__pendingPropertyChanges[part] = {
|
||||||
|
NonExistentProperty = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(changeBatcher:__flush()).to.equal(nil)
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
end
|
||||||
@@ -5,7 +5,7 @@ local isDevBuild = script.Parent.Parent:FindFirstChild("ROJO_DEV_BUILD") ~= nil
|
|||||||
return strict("Config", {
|
return strict("Config", {
|
||||||
isDevBuild = isDevBuild,
|
isDevBuild = isDevBuild,
|
||||||
codename = "Epiphany",
|
codename = "Epiphany",
|
||||||
version = {7, 0, 0, "-rc.1"},
|
version = {7, 1, 1},
|
||||||
expectedServerVersionString = "7.0 or newer",
|
expectedServerVersionString = "7.0 or newer",
|
||||||
protocolVersion = 4,
|
protocolVersion = 4,
|
||||||
defaultHost = "localhost",
|
defaultHost = "localhost",
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
local RunService = game:GetService("RunService")
|
||||||
|
|
||||||
local Log = require(script.Parent.Parent.Log)
|
local Log = require(script.Parent.Parent.Log)
|
||||||
|
|
||||||
--[[
|
--[[
|
||||||
@@ -135,29 +137,31 @@ function InstanceMap:destroyId(id)
|
|||||||
end
|
end
|
||||||
|
|
||||||
--[[
|
--[[
|
||||||
Pause updates for an instance momentarily and invoke a callback.
|
Pause updates for an instance.
|
||||||
|
|
||||||
If the callback throws an error, InstanceMap will still be kept in a
|
|
||||||
consistent state.
|
|
||||||
]]
|
]]
|
||||||
function InstanceMap:pauseInstance(instance, callback)
|
function InstanceMap:pauseInstance(instance)
|
||||||
local id = self.fromInstances[instance]
|
local id = self.fromInstances[instance]
|
||||||
|
|
||||||
-- If we don't know about this instance, ignore it and do not invoke the
|
-- If we don't know about this instance, ignore it.
|
||||||
-- callback.
|
|
||||||
if id == nil then
|
if id == nil then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
self.pausedUpdateInstances[instance] = true
|
self.pausedUpdateInstances[instance] = true
|
||||||
local success, result = xpcall(callback, debug.traceback)
|
end
|
||||||
self.pausedUpdateInstances[instance] = false
|
|
||||||
|
|
||||||
if success then
|
--[[
|
||||||
return result
|
Unpause updates for an instance.
|
||||||
else
|
]]
|
||||||
error(result, 2)
|
function InstanceMap:unpauseInstance(instance)
|
||||||
end
|
self.pausedUpdateInstances[instance] = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[
|
||||||
|
Unpause updates for all instances.
|
||||||
|
]]
|
||||||
|
function InstanceMap:unpauseAllInstances()
|
||||||
|
table.clear(self.pausedUpdateInstances)
|
||||||
end
|
end
|
||||||
|
|
||||||
function InstanceMap:__connectSignals(instance)
|
function InstanceMap:__connectSignals(instance)
|
||||||
@@ -200,6 +204,12 @@ function InstanceMap:__maybeFireInstanceChanged(instance, propertyName)
|
|||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if RunService:IsRunning() then
|
||||||
|
-- We probably don't want to pick up property changes to save to the
|
||||||
|
-- filesystem in a running game.
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
self.onInstanceChanged(instance, propertyName)
|
self.onInstanceChanged(instance, propertyName)
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -222,4 +232,4 @@ function InstanceMap:__disconnectSignals(instance)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
return InstanceMap
|
return InstanceMap
|
||||||
|
|||||||
@@ -63,7 +63,7 @@ local function applyPatch(instanceMap, patch)
|
|||||||
local failedToReify = reify(instanceMap, patch.added, id, parentInstance)
|
local failedToReify = reify(instanceMap, patch.added, id, parentInstance)
|
||||||
|
|
||||||
if not PatchSet.isEmpty(failedToReify) then
|
if not PatchSet.isEmpty(failedToReify) then
|
||||||
Log.debug("Failed to reify as part of applying a patch: {}", failedToReify)
|
Log.debug("Failed to reify as part of applying a patch: {:#?}", failedToReify)
|
||||||
PatchSet.assign(unappliedPatch, failedToReify)
|
PatchSet.assign(unappliedPatch, failedToReify)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -77,6 +77,10 @@ local function applyPatch(instanceMap, patch)
|
|||||||
continue
|
continue
|
||||||
end
|
end
|
||||||
|
|
||||||
|
-- Pause updates on this instance to avoid picking up our changes when
|
||||||
|
-- two-way sync is enabled.
|
||||||
|
instanceMap:pauseInstance(instance)
|
||||||
|
|
||||||
-- Track any part of this update that could not be applied.
|
-- Track any part of this update that could not be applied.
|
||||||
local unappliedUpdate = {
|
local unappliedUpdate = {
|
||||||
id = update.id,
|
id = update.id,
|
||||||
@@ -197,4 +201,4 @@ local function applyPatch(instanceMap, patch)
|
|||||||
return unappliedPatch
|
return unappliedPatch
|
||||||
end
|
end
|
||||||
|
|
||||||
return applyPatch
|
return applyPatch
|
||||||
|
|||||||
@@ -75,9 +75,13 @@ local function diff(instanceMap, virtualInstances, rootId)
|
|||||||
changedProperties[propertyName] = virtualValue
|
changedProperties[propertyName] = virtualValue
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
-- virtualValue can be empty in certain cases, and this may print out nil to the user.
|
|
||||||
local propertyType = next(virtualValue)
|
local propertyType = next(virtualValue)
|
||||||
Log.warn("Failed to decode property of type {}", propertyType)
|
Log.warn(
|
||||||
|
"Failed to decode property {}.{}. Encoded property was: {:#?}",
|
||||||
|
virtualInstance.ClassName,
|
||||||
|
propertyName,
|
||||||
|
virtualValue
|
||||||
|
)
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
local err = existingValueOrErr
|
local err = existingValueOrErr
|
||||||
|
|||||||
@@ -40,6 +40,13 @@ local function getProperty(instance, propertyName)
|
|||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if err.kind == RbxDom.Error.Kind.Roblox and err.extra:find("is not a valid member of") then
|
||||||
|
return false, Error.new(Error.UnknownProperty, {
|
||||||
|
className = instance.ClassName,
|
||||||
|
propertyName = propertyName,
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
return false, Error.new(Error.OtherPropertyError, {
|
return false, Error.new(Error.OtherPropertyError, {
|
||||||
className = instance.ClassName,
|
className = instance.ClassName,
|
||||||
propertyName = propertyName,
|
propertyName = propertyName,
|
||||||
|
|||||||
@@ -255,7 +255,7 @@ return function()
|
|||||||
Name = "Child A",
|
Name = "Child A",
|
||||||
Properties = {
|
Properties = {
|
||||||
Value = {
|
Value = {
|
||||||
Ref = "Child B",
|
Ref = "CHILD_B",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
Children = {},
|
Children = {},
|
||||||
@@ -287,7 +287,7 @@ return function()
|
|||||||
-- constructed as part of a recursive call before the parent has totally
|
-- constructed as part of a recursive call before the parent has totally
|
||||||
-- finished. Given deferred refs, this should not fail, but it is a good
|
-- finished. Given deferred refs, this should not fail, but it is a good
|
||||||
-- case to test.
|
-- case to test.
|
||||||
it("should apply properties containing refs to later siblings correctly", function()
|
it("should apply properties containing refs to later children correctly", function()
|
||||||
local virtualInstances = {
|
local virtualInstances = {
|
||||||
ROOT = {
|
ROOT = {
|
||||||
ClassName = "ObjectValue",
|
ClassName = "ObjectValue",
|
||||||
@@ -344,4 +344,4 @@ return function()
|
|||||||
expect(update.id).to.equal("ROOT")
|
expect(update.id).to.equal("ROOT")
|
||||||
expect(update.changedProperties.Value).to.equal(virtualInstances["ROOT"].Properties.Value)
|
expect(update.changedProperties.Value).to.equal(virtualInstances["ROOT"].Properties.Value)
|
||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ local Log = require(script.Parent.Parent.Log)
|
|||||||
local Fmt = require(script.Parent.Parent.Fmt)
|
local Fmt = require(script.Parent.Parent.Fmt)
|
||||||
local t = require(script.Parent.Parent.t)
|
local t = require(script.Parent.Parent.t)
|
||||||
|
|
||||||
|
local ChangeBatcher = require(script.Parent.ChangeBatcher)
|
||||||
local InstanceMap = require(script.Parent.InstanceMap)
|
local InstanceMap = require(script.Parent.InstanceMap)
|
||||||
local PatchSet = require(script.Parent.PatchSet)
|
local PatchSet = require(script.Parent.PatchSet)
|
||||||
local Reconciler = require(script.Parent.Reconciler)
|
local Reconciler = require(script.Parent.Reconciler)
|
||||||
@@ -56,10 +57,19 @@ function ServeSession.new(options)
|
|||||||
-- Declare self ahead of time to capture it in a closure
|
-- Declare self ahead of time to capture it in a closure
|
||||||
local self
|
local self
|
||||||
local function onInstanceChanged(instance, propertyName)
|
local function onInstanceChanged(instance, propertyName)
|
||||||
self:__onInstanceChanged(instance, propertyName)
|
if not self.__twoWaySync then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
self.__changeBatcher:add(instance, propertyName)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function onChangesFlushed(patch)
|
||||||
|
self.__apiContext:write(patch)
|
||||||
end
|
end
|
||||||
|
|
||||||
local instanceMap = InstanceMap.new(onInstanceChanged)
|
local instanceMap = InstanceMap.new(onInstanceChanged)
|
||||||
|
local changeBatcher = ChangeBatcher.new(instanceMap, onChangesFlushed)
|
||||||
local reconciler = Reconciler.new(instanceMap)
|
local reconciler = Reconciler.new(instanceMap)
|
||||||
|
|
||||||
local connections = {}
|
local connections = {}
|
||||||
@@ -82,6 +92,7 @@ function ServeSession.new(options)
|
|||||||
__twoWaySync = options.twoWaySync,
|
__twoWaySync = options.twoWaySync,
|
||||||
__reconciler = reconciler,
|
__reconciler = reconciler,
|
||||||
__instanceMap = instanceMap,
|
__instanceMap = instanceMap,
|
||||||
|
__changeBatcher = changeBatcher,
|
||||||
__statusChangedCallback = nil,
|
__statusChangedCallback = nil,
|
||||||
__connections = connections,
|
__connections = connections,
|
||||||
}
|
}
|
||||||
@@ -102,6 +113,10 @@ function ServeSession:__fmtDebug(output)
|
|||||||
output:write("}")
|
output:write("}")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
function ServeSession:getStatus()
|
||||||
|
return self.__status
|
||||||
|
end
|
||||||
|
|
||||||
function ServeSession:onStatusChanged(callback)
|
function ServeSession:onStatusChanged(callback)
|
||||||
self.__statusChangedCallback = callback
|
self.__statusChangedCallback = callback
|
||||||
end
|
end
|
||||||
@@ -179,55 +194,6 @@ function ServeSession:__onActiveScriptChanged(activeScript)
|
|||||||
self.__apiContext:open(scriptId)
|
self.__apiContext:open(scriptId)
|
||||||
end
|
end
|
||||||
|
|
||||||
function ServeSession:__onInstanceChanged(instance, propertyName)
|
|
||||||
if not self.__twoWaySync then
|
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
local instanceId = self.__instanceMap.fromInstances[instance]
|
|
||||||
|
|
||||||
if instanceId == nil then
|
|
||||||
Log.warn("Ignoring change for instance {:?} as it is unknown to Rojo", instance)
|
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
local remove = nil
|
|
||||||
|
|
||||||
local update = {
|
|
||||||
id = instanceId,
|
|
||||||
changedProperties = {},
|
|
||||||
}
|
|
||||||
|
|
||||||
if propertyName == "Name" then
|
|
||||||
update.changedName = instance.Name
|
|
||||||
elseif propertyName == "Parent" then
|
|
||||||
if instance.Parent == nil then
|
|
||||||
update = nil
|
|
||||||
remove = instanceId
|
|
||||||
else
|
|
||||||
Log.warn("Cannot sync non-nil Parent property changes yet")
|
|
||||||
return
|
|
||||||
end
|
|
||||||
else
|
|
||||||
local success, encoded = self.__reconciler:encodeApiValue(instance[propertyName])
|
|
||||||
|
|
||||||
if not success then
|
|
||||||
Log.warn("Could not sync back property {:?}.{}", instance, propertyName)
|
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
update.changedProperties[propertyName] = encoded
|
|
||||||
end
|
|
||||||
|
|
||||||
local patch = {
|
|
||||||
removed = {remove},
|
|
||||||
added = {},
|
|
||||||
updated = {update},
|
|
||||||
}
|
|
||||||
|
|
||||||
self.__apiContext:write(patch)
|
|
||||||
end
|
|
||||||
|
|
||||||
function ServeSession:__initialSync(rootInstanceId)
|
function ServeSession:__initialSync(rootInstanceId)
|
||||||
return self.__apiContext:read({ rootInstanceId })
|
return self.__apiContext:read({ rootInstanceId })
|
||||||
:andThen(function(readResponseBody)
|
:andThen(function(readResponseBody)
|
||||||
@@ -290,6 +256,7 @@ function ServeSession:__stopInternal(err)
|
|||||||
self:__setStatus(Status.Disconnected, err)
|
self:__setStatus(Status.Disconnected, err)
|
||||||
self.__apiContext:disconnect()
|
self.__apiContext:disconnect()
|
||||||
self.__instanceMap:stop()
|
self.__instanceMap:stop()
|
||||||
|
self.__changeBatcher:stop()
|
||||||
|
|
||||||
for _, connection in ipairs(self.__connections) do
|
for _, connection in ipairs(self.__connections) do
|
||||||
connection:Disconnect()
|
connection:Disconnect()
|
||||||
@@ -305,4 +272,4 @@ function ServeSession:__setStatus(status, detail)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
return ServeSession
|
return ServeSession
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
},
|
},
|
||||||
|
|
||||||
"TestEZ": {
|
"TestEZ": {
|
||||||
"$path": "modules/testez/lib"
|
"$path": "modules/testez"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -25,4 +25,4 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/build.rs
|
||||||
|
expression: contents
|
||||||
|
|
||||||
|
---
|
||||||
|
<roblox version="4">
|
||||||
|
<Item class="Folder" referent="0">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">optional</string>
|
||||||
|
</Properties>
|
||||||
|
<Item class="StringValue" referent="1">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">foo-optional</string>
|
||||||
|
<string name="Value">Hello, from foo.txt!</string>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
<Item class="StringValue" referent="2">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">foo-required</string>
|
||||||
|
<string name="Value">Hello, from foo.txt!</string>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
</Item>
|
||||||
|
</roblox>
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/build.rs
|
||||||
|
expression: contents
|
||||||
|
|
||||||
|
---
|
||||||
|
<roblox version="4">
|
||||||
|
<Item class="Folder" referent="0">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">root</string>
|
||||||
|
</Properties>
|
||||||
|
<Item class="Folder" referent="1">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">folder</string>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
</Item>
|
||||||
|
</roblox>
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/build.rs
|
||||||
|
expression: contents
|
||||||
|
|
||||||
|
---
|
||||||
|
<roblox version="4">
|
||||||
|
<Item class="Folder" referent="0">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">root</string>
|
||||||
|
</Properties>
|
||||||
|
<Item class="Folder" referent="1">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">folder</string>
|
||||||
|
</Properties>
|
||||||
|
<Item class="Folder" referent="2">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">child-projectname</string>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
</Item>
|
||||||
|
</Item>
|
||||||
|
</roblox>
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/build.rs
|
||||||
|
expression: contents
|
||||||
|
|
||||||
|
---
|
||||||
|
<roblox version="4">
|
||||||
|
<Item class="Folder" referent="0">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">root</string>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
</roblox>
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
---
|
---
|
||||||
source: tests/tests/build.rs
|
source: tests/tests/build.rs
|
||||||
expression: contents
|
expression: contents
|
||||||
|
|
||||||
---
|
---
|
||||||
<roblox version="4">
|
<roblox version="4">
|
||||||
<Item class="Folder" referent="0">
|
<Item class="Folder" referent="0">
|
||||||
@@ -25,14 +26,12 @@ expression: contents
|
|||||||
<R22>1</R22>
|
<R22>1</R22>
|
||||||
</CoordinateFrame>
|
</CoordinateFrame>
|
||||||
<Ref name="PrimaryPart">null</Ref>
|
<Ref name="PrimaryPart">null</Ref>
|
||||||
<BinaryString name="Tags">
|
<BinaryString name="Tags"></BinaryString>
|
||||||
</BinaryString>
|
|
||||||
</Properties>
|
</Properties>
|
||||||
<Item class="StringValue" referent="2">
|
<Item class="StringValue" referent="2">
|
||||||
<Properties>
|
<Properties>
|
||||||
<string name="Name">Cool StringValue</string>
|
<string name="Name">Cool StringValue</string>
|
||||||
<BinaryString name="Tags">
|
<BinaryString name="Tags"></BinaryString>
|
||||||
</BinaryString>
|
|
||||||
<string name="Value">Did you know that BaseValue.Changed is different than Instance.Changed?</string>
|
<string name="Value">Did you know that BaseValue.Changed is different than Instance.Changed?</string>
|
||||||
</Properties>
|
</Properties>
|
||||||
</Item>
|
</Item>
|
||||||
|
|||||||
@@ -1,27 +1,25 @@
|
|||||||
---
|
---
|
||||||
source: tests/tests/build.rs
|
source: tests/tests/build.rs
|
||||||
expression: contents
|
expression: contents
|
||||||
|
|
||||||
---
|
---
|
||||||
<roblox version="4">
|
<roblox version="4">
|
||||||
<Item class="Folder" referent="0">
|
<Item class="Folder" referent="0">
|
||||||
<Properties>
|
<Properties>
|
||||||
<string name="Name">rbxmx_ref</string>
|
<string name="Name">rbxmx_ref</string>
|
||||||
<BinaryString name="Tags">
|
<BinaryString name="Tags"></BinaryString>
|
||||||
</BinaryString>
|
|
||||||
</Properties>
|
</Properties>
|
||||||
<Item class="StringValue" referent="1">
|
<Item class="StringValue" referent="1">
|
||||||
<Properties>
|
<Properties>
|
||||||
<string name="Name">Target</string>
|
<string name="Name">Target</string>
|
||||||
<BinaryString name="Tags">
|
<BinaryString name="Tags"></BinaryString>
|
||||||
</BinaryString>
|
|
||||||
<string name="Value">Pointed to by ObjectValue</string>
|
<string name="Value">Pointed to by ObjectValue</string>
|
||||||
</Properties>
|
</Properties>
|
||||||
</Item>
|
</Item>
|
||||||
<Item class="ObjectValue" referent="2">
|
<Item class="ObjectValue" referent="2">
|
||||||
<Properties>
|
<Properties>
|
||||||
<string name="Name">Pointer</string>
|
<string name="Name">Pointer</string>
|
||||||
<BinaryString name="Tags">
|
<BinaryString name="Tags"></BinaryString>
|
||||||
</BinaryString>
|
|
||||||
<Ref name="Value">1</Ref>
|
<Ref name="Value">1</Ref>
|
||||||
</Properties>
|
</Properties>
|
||||||
</Item>
|
</Item>
|
||||||
|
|||||||
@@ -32,6 +32,20 @@ expression: contents
|
|||||||
<Item class="Part" referent="4">
|
<Item class="Part" referent="4">
|
||||||
<Properties>
|
<Properties>
|
||||||
<string name="Name">Color</string>
|
<string name="Name">Color</string>
|
||||||
|
<CoordinateFrame name="CFrame">
|
||||||
|
<X>1</X>
|
||||||
|
<Y>2</Y>
|
||||||
|
<Z>3</Z>
|
||||||
|
<R00>0</R00>
|
||||||
|
<R01>1</R01>
|
||||||
|
<R02>0</R02>
|
||||||
|
<R10>0</R10>
|
||||||
|
<R11>0</R11>
|
||||||
|
<R12>1</R12>
|
||||||
|
<R20>1</R20>
|
||||||
|
<R21>0</R21>
|
||||||
|
<R22>0</R22>
|
||||||
|
</CoordinateFrame>
|
||||||
<Color3uint8 name="Color3uint8">8404992</Color3uint8>
|
<Color3uint8 name="Color3uint8">8404992</Color3uint8>
|
||||||
</Properties>
|
</Properties>
|
||||||
</Item>
|
</Item>
|
||||||
|
|||||||
@@ -0,0 +1,230 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/build.rs
|
||||||
|
expression: contents
|
||||||
|
|
||||||
|
---
|
||||||
|
<roblox version="4">
|
||||||
|
<Item class="Folder" referent="0">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">weldconstraint</string>
|
||||||
|
<BinaryString name="AttributesSerialize">
|
||||||
|
</BinaryString>
|
||||||
|
<int64 name="SourceAssetId">-1</int64>
|
||||||
|
<BinaryString name="Tags"></BinaryString>
|
||||||
|
</Properties>
|
||||||
|
<Item class="Part" referent="1">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">A</string>
|
||||||
|
<bool name="Anchored">false</bool>
|
||||||
|
<BinaryString name="AttributesSerialize">
|
||||||
|
</BinaryString>
|
||||||
|
<float name="BackParamA">-0.5</float>
|
||||||
|
<float name="BackParamB">0.5</float>
|
||||||
|
<token name="BackSurface">0</token>
|
||||||
|
<token name="BackSurfaceInput">0</token>
|
||||||
|
<float name="BottomParamA">-0.5</float>
|
||||||
|
<float name="BottomParamB">0.5</float>
|
||||||
|
<token name="BottomSurface">0</token>
|
||||||
|
<token name="BottomSurfaceInput">0</token>
|
||||||
|
<CoordinateFrame name="CFrame">
|
||||||
|
<X>-14</X>
|
||||||
|
<Y>0.5</Y>
|
||||||
|
<Z>-5</Z>
|
||||||
|
<R00>1</R00>
|
||||||
|
<R01>0</R01>
|
||||||
|
<R02>0</R02>
|
||||||
|
<R10>0</R10>
|
||||||
|
<R11>1</R11>
|
||||||
|
<R12>0</R12>
|
||||||
|
<R20>0</R20>
|
||||||
|
<R21>0</R21>
|
||||||
|
<R22>1</R22>
|
||||||
|
</CoordinateFrame>
|
||||||
|
<bool name="CanCollide">true</bool>
|
||||||
|
<bool name="CanQuery">true</bool>
|
||||||
|
<bool name="CanTouch">true</bool>
|
||||||
|
<bool name="CastShadow">true</bool>
|
||||||
|
<int name="CollisionGroupId">0</int>
|
||||||
|
<Color3uint8 name="Color3uint8">10724005</Color3uint8>
|
||||||
|
<PhysicalProperties name="CustomPhysicalProperties">
|
||||||
|
<CustomPhysics>false</CustomPhysics>
|
||||||
|
</PhysicalProperties>
|
||||||
|
<token name="formFactorRaw">1</token>
|
||||||
|
<float name="FrontParamA">-0.5</float>
|
||||||
|
<float name="FrontParamB">0.5</float>
|
||||||
|
<token name="FrontSurface">0</token>
|
||||||
|
<token name="FrontSurfaceInput">0</token>
|
||||||
|
<float name="LeftParamA">-0.5</float>
|
||||||
|
<float name="LeftParamB">0.5</float>
|
||||||
|
<token name="LeftSurface">0</token>
|
||||||
|
<token name="LeftSurfaceInput">0</token>
|
||||||
|
<bool name="Locked">false</bool>
|
||||||
|
<bool name="Massless">false</bool>
|
||||||
|
<token name="Material">256</token>
|
||||||
|
<CoordinateFrame name="PivotOffset">
|
||||||
|
<X>0</X>
|
||||||
|
<Y>0</Y>
|
||||||
|
<Z>0</Z>
|
||||||
|
<R00>1</R00>
|
||||||
|
<R01>0</R01>
|
||||||
|
<R02>0</R02>
|
||||||
|
<R10>0</R10>
|
||||||
|
<R11>1</R11>
|
||||||
|
<R12>0</R12>
|
||||||
|
<R20>0</R20>
|
||||||
|
<R21>0</R21>
|
||||||
|
<R22>1</R22>
|
||||||
|
</CoordinateFrame>
|
||||||
|
<float name="Reflectance">0</float>
|
||||||
|
<float name="RightParamA">-0.5</float>
|
||||||
|
<float name="RightParamB">0.5</float>
|
||||||
|
<token name="RightSurface">0</token>
|
||||||
|
<token name="RightSurfaceInput">0</token>
|
||||||
|
<int name="RootPriority">0</int>
|
||||||
|
<Vector3 name="RotVelocity">
|
||||||
|
<X>0</X>
|
||||||
|
<Y>0</Y>
|
||||||
|
<Z>0</Z>
|
||||||
|
</Vector3>
|
||||||
|
<token name="shape">1</token>
|
||||||
|
<Vector3 name="size">
|
||||||
|
<X>4</X>
|
||||||
|
<Y>1</Y>
|
||||||
|
<Z>2</Z>
|
||||||
|
</Vector3>
|
||||||
|
<int64 name="SourceAssetId">-1</int64>
|
||||||
|
<BinaryString name="Tags"></BinaryString>
|
||||||
|
<float name="TopParamA">-0.5</float>
|
||||||
|
<float name="TopParamB">0.5</float>
|
||||||
|
<token name="TopSurface">0</token>
|
||||||
|
<token name="TopSurfaceInput">0</token>
|
||||||
|
<float name="Transparency">0</float>
|
||||||
|
<Vector3 name="Velocity">
|
||||||
|
<X>0</X>
|
||||||
|
<Y>0</Y>
|
||||||
|
<Z>0</Z>
|
||||||
|
</Vector3>
|
||||||
|
</Properties>
|
||||||
|
<Item class="WeldConstraint" referent="2">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">WeldConstraint</string>
|
||||||
|
<BinaryString name="AttributesSerialize">
|
||||||
|
</BinaryString>
|
||||||
|
<CoordinateFrame name="CFrame0">
|
||||||
|
<X>7</X>
|
||||||
|
<Y>0.000001013279</Y>
|
||||||
|
<Z>-3</Z>
|
||||||
|
<R00>1</R00>
|
||||||
|
<R01>0</R01>
|
||||||
|
<R02>0</R02>
|
||||||
|
<R10>0</R10>
|
||||||
|
<R11>1</R11>
|
||||||
|
<R12>0</R12>
|
||||||
|
<R20>0</R20>
|
||||||
|
<R21>0</R21>
|
||||||
|
<R22>1</R22>
|
||||||
|
</CoordinateFrame>
|
||||||
|
<Ref name="Part0Internal">1</Ref>
|
||||||
|
<Ref name="Part1Internal">3</Ref>
|
||||||
|
<int64 name="SourceAssetId">-1</int64>
|
||||||
|
<int name="State">3</int>
|
||||||
|
<BinaryString name="Tags"></BinaryString>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
</Item>
|
||||||
|
<Item class="Part" referent="3">
|
||||||
|
<Properties>
|
||||||
|
<string name="Name">B</string>
|
||||||
|
<bool name="Anchored">false</bool>
|
||||||
|
<BinaryString name="AttributesSerialize">
|
||||||
|
</BinaryString>
|
||||||
|
<float name="BackParamA">-0.5</float>
|
||||||
|
<float name="BackParamB">0.5</float>
|
||||||
|
<token name="BackSurface">0</token>
|
||||||
|
<token name="BackSurfaceInput">0</token>
|
||||||
|
<float name="BottomParamA">-0.5</float>
|
||||||
|
<float name="BottomParamB">0.5</float>
|
||||||
|
<token name="BottomSurface">0</token>
|
||||||
|
<token name="BottomSurfaceInput">0</token>
|
||||||
|
<CoordinateFrame name="CFrame">
|
||||||
|
<X>-7</X>
|
||||||
|
<Y>0.500001</Y>
|
||||||
|
<Z>-8</Z>
|
||||||
|
<R00>1</R00>
|
||||||
|
<R01>0</R01>
|
||||||
|
<R02>0</R02>
|
||||||
|
<R10>0</R10>
|
||||||
|
<R11>1</R11>
|
||||||
|
<R12>0</R12>
|
||||||
|
<R20>0</R20>
|
||||||
|
<R21>0</R21>
|
||||||
|
<R22>1</R22>
|
||||||
|
</CoordinateFrame>
|
||||||
|
<bool name="CanCollide">true</bool>
|
||||||
|
<bool name="CanQuery">true</bool>
|
||||||
|
<bool name="CanTouch">true</bool>
|
||||||
|
<bool name="CastShadow">true</bool>
|
||||||
|
<int name="CollisionGroupId">0</int>
|
||||||
|
<Color3uint8 name="Color3uint8">10724005</Color3uint8>
|
||||||
|
<PhysicalProperties name="CustomPhysicalProperties">
|
||||||
|
<CustomPhysics>false</CustomPhysics>
|
||||||
|
</PhysicalProperties>
|
||||||
|
<token name="formFactorRaw">1</token>
|
||||||
|
<float name="FrontParamA">-0.5</float>
|
||||||
|
<float name="FrontParamB">0.5</float>
|
||||||
|
<token name="FrontSurface">0</token>
|
||||||
|
<token name="FrontSurfaceInput">0</token>
|
||||||
|
<float name="LeftParamA">-0.5</float>
|
||||||
|
<float name="LeftParamB">0.5</float>
|
||||||
|
<token name="LeftSurface">0</token>
|
||||||
|
<token name="LeftSurfaceInput">0</token>
|
||||||
|
<bool name="Locked">false</bool>
|
||||||
|
<bool name="Massless">false</bool>
|
||||||
|
<token name="Material">256</token>
|
||||||
|
<CoordinateFrame name="PivotOffset">
|
||||||
|
<X>0</X>
|
||||||
|
<Y>0</Y>
|
||||||
|
<Z>0</Z>
|
||||||
|
<R00>1</R00>
|
||||||
|
<R01>0</R01>
|
||||||
|
<R02>0</R02>
|
||||||
|
<R10>0</R10>
|
||||||
|
<R11>1</R11>
|
||||||
|
<R12>0</R12>
|
||||||
|
<R20>0</R20>
|
||||||
|
<R21>0</R21>
|
||||||
|
<R22>1</R22>
|
||||||
|
</CoordinateFrame>
|
||||||
|
<float name="Reflectance">0</float>
|
||||||
|
<float name="RightParamA">-0.5</float>
|
||||||
|
<float name="RightParamB">0.5</float>
|
||||||
|
<token name="RightSurface">0</token>
|
||||||
|
<token name="RightSurfaceInput">0</token>
|
||||||
|
<int name="RootPriority">0</int>
|
||||||
|
<Vector3 name="RotVelocity">
|
||||||
|
<X>0</X>
|
||||||
|
<Y>0</Y>
|
||||||
|
<Z>0</Z>
|
||||||
|
</Vector3>
|
||||||
|
<token name="shape">1</token>
|
||||||
|
<Vector3 name="size">
|
||||||
|
<X>4</X>
|
||||||
|
<Y>1</Y>
|
||||||
|
<Z>2</Z>
|
||||||
|
</Vector3>
|
||||||
|
<int64 name="SourceAssetId">-1</int64>
|
||||||
|
<BinaryString name="Tags"></BinaryString>
|
||||||
|
<float name="TopParamA">-0.5</float>
|
||||||
|
<float name="TopParamB">0.5</float>
|
||||||
|
<token name="TopSurface">0</token>
|
||||||
|
<token name="TopSurfaceInput">0</token>
|
||||||
|
<float name="Transparency">0</float>
|
||||||
|
<Vector3 name="Velocity">
|
||||||
|
<X>0</X>
|
||||||
|
<Y>0</Y>
|
||||||
|
<Z>0</Z>
|
||||||
|
</Vector3>
|
||||||
|
</Properties>
|
||||||
|
</Item>
|
||||||
|
</Item>
|
||||||
|
</roblox>
|
||||||
15
rojo-test/build-tests/optional/default.project.json
Normal file
15
rojo-test/build-tests/optional/default.project.json
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"name": "optional",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder",
|
||||||
|
"foo-required": {
|
||||||
|
"$path": "foo.txt"
|
||||||
|
},
|
||||||
|
"foo-optional":{
|
||||||
|
"$path": { "optional": "foo.txt" }
|
||||||
|
},
|
||||||
|
"bar-optional":{
|
||||||
|
"$path": { "optional": "bar.txt" }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
1
rojo-test/build-tests/optional/foo.txt
Normal file
1
rojo-test/build-tests/optional/foo.txt
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Hello, from foo.txt!
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
{
|
||||||
|
"name": "root",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder",
|
||||||
|
"folder": {
|
||||||
|
"$path": "folder"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"name": "child-projectname",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
{
|
||||||
|
"name": "root",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder",
|
||||||
|
"folder": {
|
||||||
|
"$path": "folder"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"name": "child-projectname",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"name": "root",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -14,7 +14,13 @@
|
|||||||
"Color": {
|
"Color": {
|
||||||
"$className": "Part",
|
"$className": "Part",
|
||||||
"$properties": {
|
"$properties": {
|
||||||
"Color": [0.5, 0.25, 0]
|
"Color": [0.5, 0.25, 0],
|
||||||
|
"CFrame": [
|
||||||
|
1, 2, 3,
|
||||||
|
0, 1, 0,
|
||||||
|
0, 0, 1,
|
||||||
|
1, 0, 0
|
||||||
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"name": "weldconstraint",
|
||||||
|
"tree": {
|
||||||
|
"$path": "two-parts-welded.rbxmx"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/serve.rs
|
||||||
|
expression: redactions.redacted_yaml(info)
|
||||||
|
|
||||||
|
---
|
||||||
|
expectedPlaceIds: ~
|
||||||
|
gameId: ~
|
||||||
|
placeId: ~
|
||||||
|
projectName: optional
|
||||||
|
protocolVersion: 4
|
||||||
|
rootInstanceId: id-2
|
||||||
|
serverVersion: "[server-version]"
|
||||||
|
sessionId: id-1
|
||||||
|
|
||||||
@@ -0,0 +1,62 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/serve.rs
|
||||||
|
expression: "read_response.intern_and_redact(&mut redactions, root_id)"
|
||||||
|
|
||||||
|
---
|
||||||
|
instances:
|
||||||
|
id-2:
|
||||||
|
Children:
|
||||||
|
- id-3
|
||||||
|
ClassName: Folder
|
||||||
|
Id: id-2
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: true
|
||||||
|
Name: optional
|
||||||
|
Parent: "00000000000000000000000000000000"
|
||||||
|
Properties: {}
|
||||||
|
id-3:
|
||||||
|
Children:
|
||||||
|
- id-4
|
||||||
|
- id-5
|
||||||
|
ClassName: Folder
|
||||||
|
Id: id-3
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: src
|
||||||
|
Parent: id-2
|
||||||
|
Properties: {}
|
||||||
|
id-4:
|
||||||
|
Children: []
|
||||||
|
ClassName: StringValue
|
||||||
|
Id: id-4
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: foo
|
||||||
|
Parent: id-3
|
||||||
|
Properties:
|
||||||
|
Value:
|
||||||
|
String: "Hello, from foo.txt!"
|
||||||
|
id-5:
|
||||||
|
Children:
|
||||||
|
- id-6
|
||||||
|
ClassName: Folder
|
||||||
|
Id: id-5
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: node_modules
|
||||||
|
Parent: id-3
|
||||||
|
Properties: {}
|
||||||
|
id-6:
|
||||||
|
Children: []
|
||||||
|
ClassName: StringValue
|
||||||
|
Id: id-6
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: bar
|
||||||
|
Parent: id-5
|
||||||
|
Properties:
|
||||||
|
Value:
|
||||||
|
String: Hello from bar.txt
|
||||||
|
messageCursor: 2
|
||||||
|
sessionId: id-1
|
||||||
|
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/serve.rs
|
||||||
|
expression: "read_response.intern_and_redact(&mut redactions, root_id)"
|
||||||
|
|
||||||
|
---
|
||||||
|
instances:
|
||||||
|
id-2:
|
||||||
|
Children:
|
||||||
|
- id-3
|
||||||
|
ClassName: Folder
|
||||||
|
Id: id-2
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: true
|
||||||
|
Name: optional
|
||||||
|
Parent: "00000000000000000000000000000000"
|
||||||
|
Properties: {}
|
||||||
|
id-3:
|
||||||
|
Children:
|
||||||
|
- id-4
|
||||||
|
ClassName: Folder
|
||||||
|
Id: id-3
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: src
|
||||||
|
Parent: id-2
|
||||||
|
Properties: {}
|
||||||
|
id-4:
|
||||||
|
Children: []
|
||||||
|
ClassName: StringValue
|
||||||
|
Id: id-4
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: foo
|
||||||
|
Parent: id-3
|
||||||
|
Properties:
|
||||||
|
Value:
|
||||||
|
String: "Hello, from foo.txt!"
|
||||||
|
messageCursor: 0
|
||||||
|
sessionId: id-1
|
||||||
|
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
---
|
||||||
|
source: tests/tests/serve.rs
|
||||||
|
expression: "subscribe_response.intern_and_redact(&mut redactions, ())"
|
||||||
|
|
||||||
|
---
|
||||||
|
messageCursor: 2
|
||||||
|
messages:
|
||||||
|
- added:
|
||||||
|
id-5:
|
||||||
|
Children:
|
||||||
|
- id-6
|
||||||
|
ClassName: Folder
|
||||||
|
Id: id-5
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: node_modules
|
||||||
|
Parent: id-3
|
||||||
|
Properties: {}
|
||||||
|
id-6:
|
||||||
|
Children: []
|
||||||
|
ClassName: StringValue
|
||||||
|
Id: id-6
|
||||||
|
Metadata:
|
||||||
|
ignoreUnknownInstances: false
|
||||||
|
Name: bar
|
||||||
|
Parent: id-5
|
||||||
|
Properties:
|
||||||
|
Value:
|
||||||
|
String: Hello from bar.txt
|
||||||
|
removed: []
|
||||||
|
updated: []
|
||||||
|
- added: {}
|
||||||
|
removed: []
|
||||||
|
updated: []
|
||||||
|
sessionId: id-1
|
||||||
|
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
{
|
||||||
|
"name": "optional",
|
||||||
|
"tree": {
|
||||||
|
"$className": "Folder",
|
||||||
|
"create-later": {
|
||||||
|
"$path": { "optional": "create-later" }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
std = "roblox"
|
std = "roblox+testez"
|
||||||
|
|
||||||
[config]
|
[config]
|
||||||
unused_variable = { allow_unused_self = true }
|
unused_variable = { allow_unused_self = true }
|
||||||
|
|||||||
@@ -284,22 +284,14 @@ fn compute_and_apply_changes(tree: &mut RojoTree, vfs: &Vfs, id: Ref) -> Option<
|
|||||||
// that path and use it as the source for our patch.
|
// that path and use it as the source for our patch.
|
||||||
|
|
||||||
let snapshot = match snapshot_from_vfs(&metadata.context, &vfs, &path) {
|
let snapshot = match snapshot_from_vfs(&metadata.context, &vfs, &path) {
|
||||||
Ok(Some(snapshot)) => snapshot,
|
Ok(snapshot) => snapshot,
|
||||||
Ok(None) => {
|
|
||||||
log::error!(
|
|
||||||
"Snapshot did not return an instance from path {}",
|
|
||||||
path.display()
|
|
||||||
);
|
|
||||||
log::error!("This may be a bug!");
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
log::error!("Snapshot error: {:?}", err);
|
log::error!("Snapshot error: {:?}", err);
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let patch_set = compute_patch_set(&snapshot, &tree, id);
|
let patch_set = compute_patch_set(snapshot.as_ref(), &tree, id);
|
||||||
apply_patch_set(tree, patch_set)
|
apply_patch_set(tree, patch_set)
|
||||||
}
|
}
|
||||||
Ok(None) => {
|
Ok(None) => {
|
||||||
@@ -335,19 +327,14 @@ fn compute_and_apply_changes(tree: &mut RojoTree, vfs: &Vfs, id: Ref) -> Option<
|
|||||||
);
|
);
|
||||||
|
|
||||||
let snapshot = match snapshot_result {
|
let snapshot = match snapshot_result {
|
||||||
Ok(Some(snapshot)) => snapshot,
|
Ok(snapshot) => snapshot,
|
||||||
Ok(None) => {
|
|
||||||
log::error!("Snapshot did not return an instance from a project node.");
|
|
||||||
log::error!("This is a bug!");
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
log::error!("{:?}", err);
|
log::error!("{:?}", err);
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let patch_set = compute_patch_set(&snapshot, &tree, id);
|
let patch_set = compute_patch_set(snapshot.as_ref(), &tree, id);
|
||||||
apply_patch_set(tree, patch_set)
|
apply_patch_set(tree, patch_set)
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -4,9 +4,9 @@ use std::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
|
use clap::Parser;
|
||||||
use fs_err::File;
|
use fs_err::File;
|
||||||
use memofs::Vfs;
|
use memofs::Vfs;
|
||||||
use structopt::StructOpt;
|
|
||||||
use tokio::runtime::Runtime;
|
use tokio::runtime::Runtime;
|
||||||
|
|
||||||
use crate::serve_session::ServeSession;
|
use crate::serve_session::ServeSession;
|
||||||
@@ -17,20 +17,20 @@ const UNKNOWN_OUTPUT_KIND_ERR: &str = "Could not detect what kind of file to bui
|
|||||||
Expected output file to end in .rbxl, .rbxlx, .rbxm, or .rbxmx.";
|
Expected output file to end in .rbxl, .rbxlx, .rbxm, or .rbxmx.";
|
||||||
|
|
||||||
/// Generates a model or place file from the Rojo project.
|
/// Generates a model or place file from the Rojo project.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct BuildCommand {
|
pub struct BuildCommand {
|
||||||
/// Path to the project to serve. Defaults to the current directory.
|
/// Path to the project to serve. Defaults to the current directory.
|
||||||
#[structopt(default_value = "")]
|
#[clap(default_value = "")]
|
||||||
pub project: PathBuf,
|
pub project: PathBuf,
|
||||||
|
|
||||||
/// Where to output the result.
|
/// Where to output the result.
|
||||||
///
|
///
|
||||||
/// Should end in .rbxm, .rbxl, .rbxmx, or .rbxlx.
|
/// Should end in .rbxm, .rbxl, .rbxmx, or .rbxlx.
|
||||||
#[structopt(long, short)]
|
#[clap(long, short)]
|
||||||
pub output: PathBuf,
|
pub output: PathBuf,
|
||||||
|
|
||||||
/// Whether to automatically rebuild when any input files change.
|
/// Whether to automatically rebuild when any input files change.
|
||||||
#[structopt(long)]
|
#[clap(long)]
|
||||||
pub watch: bool,
|
pub watch: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
use structopt::StructOpt;
|
use clap::Parser;
|
||||||
|
|
||||||
/// Open Rojo's documentation in your browser.
|
/// Open Rojo's documentation in your browser.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct DocCommand {}
|
pub struct DocCommand {}
|
||||||
|
|
||||||
impl DocCommand {
|
impl DocCommand {
|
||||||
|
|||||||
@@ -1,15 +1,15 @@
|
|||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use structopt::StructOpt;
|
use clap::Parser;
|
||||||
|
|
||||||
use crate::project::Project;
|
use crate::project::Project;
|
||||||
|
|
||||||
/// Reformat a Rojo project using the standard JSON formatting rules.
|
/// Reformat a Rojo project using the standard JSON formatting rules.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct FmtProjectCommand {
|
pub struct FmtProjectCommand {
|
||||||
/// Path to the project to format. Defaults to the current directory.
|
/// Path to the project to format. Defaults to the current directory.
|
||||||
#[structopt(default_value = "")]
|
#[clap(default_value = "")]
|
||||||
pub project: PathBuf,
|
pub project: PathBuf,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,9 +4,9 @@ use std::process::{Command, Stdio};
|
|||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use anyhow::{bail, format_err};
|
use anyhow::{bail, format_err};
|
||||||
|
use clap::Parser;
|
||||||
use fs_err as fs;
|
use fs_err as fs;
|
||||||
use fs_err::OpenOptions;
|
use fs_err::OpenOptions;
|
||||||
use structopt::StructOpt;
|
|
||||||
|
|
||||||
use super::resolve_path;
|
use super::resolve_path;
|
||||||
|
|
||||||
@@ -22,14 +22,14 @@ static PLACE_README: &str = include_str!("../../assets/default-place-project/REA
|
|||||||
static PLACE_GIT_IGNORE: &str = include_str!("../../assets/default-place-project/gitignore.txt");
|
static PLACE_GIT_IGNORE: &str = include_str!("../../assets/default-place-project/gitignore.txt");
|
||||||
|
|
||||||
/// Initializes a new Rojo project.
|
/// Initializes a new Rojo project.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct InitCommand {
|
pub struct InitCommand {
|
||||||
/// Path to the place to create the project. Defaults to the current directory.
|
/// Path to the place to create the project. Defaults to the current directory.
|
||||||
#[structopt(default_value = "")]
|
#[clap(default_value = "")]
|
||||||
pub path: PathBuf,
|
pub path: PathBuf,
|
||||||
|
|
||||||
/// The kind of project to create, 'place' or 'model'. Defaults to place.
|
/// The kind of project to create, 'place' or 'model'. Defaults to place.
|
||||||
#[structopt(long, default_value = "place")]
|
#[clap(long, default_value = "place")]
|
||||||
pub kind: InitKind,
|
pub kind: InitKind,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
//! Defines Rojo's CLI through structopt types.
|
//! Defines Rojo's CLI through clap types.
|
||||||
|
|
||||||
mod build;
|
mod build;
|
||||||
mod doc;
|
mod doc;
|
||||||
@@ -6,11 +6,12 @@ mod fmt_project;
|
|||||||
mod init;
|
mod init;
|
||||||
mod plugin;
|
mod plugin;
|
||||||
mod serve;
|
mod serve;
|
||||||
|
mod sourcemap;
|
||||||
mod upload;
|
mod upload;
|
||||||
|
|
||||||
use std::{borrow::Cow, env, path::Path, str::FromStr};
|
use std::{borrow::Cow, env, path::Path, str::FromStr};
|
||||||
|
|
||||||
use structopt::StructOpt;
|
use clap::Parser;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
|
||||||
pub use self::build::BuildCommand;
|
pub use self::build::BuildCommand;
|
||||||
@@ -19,17 +20,18 @@ pub use self::fmt_project::FmtProjectCommand;
|
|||||||
pub use self::init::{InitCommand, InitKind};
|
pub use self::init::{InitCommand, InitKind};
|
||||||
pub use self::plugin::{PluginCommand, PluginSubcommand};
|
pub use self::plugin::{PluginCommand, PluginSubcommand};
|
||||||
pub use self::serve::ServeCommand;
|
pub use self::serve::ServeCommand;
|
||||||
|
pub use self::sourcemap::SourcemapCommand;
|
||||||
pub use self::upload::UploadCommand;
|
pub use self::upload::UploadCommand;
|
||||||
|
|
||||||
/// Command line options that Rojo accepts, defined using the structopt crate.
|
/// Command line options that Rojo accepts, defined using the clap crate.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
#[structopt(name = "Rojo", about, author)]
|
#[clap(name = "Rojo", version, about, author)]
|
||||||
pub struct Options {
|
pub struct Options {
|
||||||
#[structopt(flatten)]
|
#[clap(flatten)]
|
||||||
pub global: GlobalOptions,
|
pub global: GlobalOptions,
|
||||||
|
|
||||||
/// Subcommand to run in this invocation.
|
/// Subcommand to run in this invocation.
|
||||||
#[structopt(subcommand)]
|
#[clap(subcommand)]
|
||||||
pub subcommand: Subcommand,
|
pub subcommand: Subcommand,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,6 +42,7 @@ impl Options {
|
|||||||
Subcommand::Serve(subcommand) => subcommand.run(self.global),
|
Subcommand::Serve(subcommand) => subcommand.run(self.global),
|
||||||
Subcommand::Build(subcommand) => subcommand.run(),
|
Subcommand::Build(subcommand) => subcommand.run(),
|
||||||
Subcommand::Upload(subcommand) => subcommand.run(),
|
Subcommand::Upload(subcommand) => subcommand.run(),
|
||||||
|
Subcommand::Sourcemap(subcommand) => subcommand.run(),
|
||||||
Subcommand::FmtProject(subcommand) => subcommand.run(),
|
Subcommand::FmtProject(subcommand) => subcommand.run(),
|
||||||
Subcommand::Doc(subcommand) => subcommand.run(),
|
Subcommand::Doc(subcommand) => subcommand.run(),
|
||||||
Subcommand::Plugin(subcommand) => subcommand.run(),
|
Subcommand::Plugin(subcommand) => subcommand.run(),
|
||||||
@@ -47,14 +50,14 @@ impl Options {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct GlobalOptions {
|
pub struct GlobalOptions {
|
||||||
/// Sets verbosity level. Can be specified multiple times.
|
/// Sets verbosity level. Can be specified multiple times.
|
||||||
#[structopt(long("verbose"), short, global(true), parse(from_occurrences))]
|
#[clap(long("verbose"), short, global(true), parse(from_occurrences))]
|
||||||
pub verbosity: u8,
|
pub verbosity: u8,
|
||||||
|
|
||||||
/// Set color behavior. Valid values are auto, always, and never.
|
/// Set color behavior. Valid values are auto, always, and never.
|
||||||
#[structopt(long("color"), global(true), default_value("auto"))]
|
#[clap(long("color"), global(true), default_value("auto"))]
|
||||||
pub color: ColorChoice,
|
pub color: ColorChoice,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -106,12 +109,13 @@ pub struct ColorChoiceParseError {
|
|||||||
attempted: String,
|
attempted: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub enum Subcommand {
|
pub enum Subcommand {
|
||||||
Init(InitCommand),
|
Init(InitCommand),
|
||||||
Serve(ServeCommand),
|
Serve(ServeCommand),
|
||||||
Build(BuildCommand),
|
Build(BuildCommand),
|
||||||
Upload(UploadCommand),
|
Upload(UploadCommand),
|
||||||
|
Sourcemap(SourcemapCommand),
|
||||||
FmtProject(FmtProjectCommand),
|
FmtProject(FmtProjectCommand),
|
||||||
Doc(DocCommand),
|
Doc(DocCommand),
|
||||||
Plugin(PluginCommand),
|
Plugin(PluginCommand),
|
||||||
|
|||||||
@@ -3,9 +3,9 @@ use std::{
|
|||||||
io::BufWriter,
|
io::BufWriter,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use clap::Parser;
|
||||||
use memofs::{InMemoryFs, Vfs, VfsSnapshot};
|
use memofs::{InMemoryFs, Vfs, VfsSnapshot};
|
||||||
use roblox_install::RobloxStudio;
|
use roblox_install::RobloxStudio;
|
||||||
use structopt::StructOpt;
|
|
||||||
|
|
||||||
use crate::serve_session::ServeSession;
|
use crate::serve_session::ServeSession;
|
||||||
|
|
||||||
@@ -13,14 +13,14 @@ static PLUGIN_BINCODE: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/plugin.
|
|||||||
static PLUGIN_FILE_NAME: &str = "RojoManagedPlugin.rbxm";
|
static PLUGIN_FILE_NAME: &str = "RojoManagedPlugin.rbxm";
|
||||||
|
|
||||||
/// Install Rojo's plugin.
|
/// Install Rojo's plugin.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct PluginCommand {
|
pub struct PluginCommand {
|
||||||
#[structopt(subcommand)]
|
#[clap(subcommand)]
|
||||||
subcommand: PluginSubcommand,
|
subcommand: PluginSubcommand,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Manages Rojo's Roblox Studio plugin.
|
/// Manages Rojo's Roblox Studio plugin.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub enum PluginSubcommand {
|
pub enum PluginSubcommand {
|
||||||
/// Install the plugin in Roblox Studio's plugins folder. If the plugin is
|
/// Install the plugin in Roblox Studio's plugins folder. If the plugin is
|
||||||
/// already installed, installing it again will overwrite the current plugin
|
/// already installed, installing it again will overwrite the current plugin
|
||||||
|
|||||||
@@ -5,8 +5,8 @@ use std::{
|
|||||||
sync::Arc,
|
sync::Arc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use clap::Parser;
|
||||||
use memofs::Vfs;
|
use memofs::Vfs;
|
||||||
use structopt::StructOpt;
|
|
||||||
use termcolor::{BufferWriter, Color, ColorChoice, ColorSpec, WriteColor};
|
use termcolor::{BufferWriter, Color, ColorChoice, ColorSpec, WriteColor};
|
||||||
|
|
||||||
use crate::{serve_session::ServeSession, web::LiveServer};
|
use crate::{serve_session::ServeSession, web::LiveServer};
|
||||||
@@ -17,19 +17,19 @@ const DEFAULT_BIND_ADDRESS: Ipv4Addr = Ipv4Addr::new(127, 0, 0, 1);
|
|||||||
const DEFAULT_PORT: u16 = 34872;
|
const DEFAULT_PORT: u16 = 34872;
|
||||||
|
|
||||||
/// Expose a Rojo project to the Rojo Studio plugin.
|
/// Expose a Rojo project to the Rojo Studio plugin.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct ServeCommand {
|
pub struct ServeCommand {
|
||||||
/// Path to the project to serve. Defaults to the current directory.
|
/// Path to the project to serve. Defaults to the current directory.
|
||||||
#[structopt(default_value = "")]
|
#[clap(default_value = "")]
|
||||||
pub project: PathBuf,
|
pub project: PathBuf,
|
||||||
|
|
||||||
/// The IP address to listen on. Defaults to `127.0.0.1`.
|
/// The IP address to listen on. Defaults to `127.0.0.1`.
|
||||||
#[structopt(long)]
|
#[clap(long)]
|
||||||
pub address: Option<IpAddr>,
|
pub address: Option<IpAddr>,
|
||||||
|
|
||||||
/// The port to listen on. Defaults to the project's preference, or `34872` if
|
/// The port to listen on. Defaults to the project's preference, or `34872` if
|
||||||
/// it has none.
|
/// it has none.
|
||||||
#[structopt(long)]
|
#[clap(long)]
|
||||||
pub port: Option<u16>,
|
pub port: Option<u16>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -41,7 +41,10 @@ impl ServeCommand {
|
|||||||
|
|
||||||
let session = Arc::new(ServeSession::new(vfs, &project_path)?);
|
let session = Arc::new(ServeSession::new(vfs, &project_path)?);
|
||||||
|
|
||||||
let ip = self.address.unwrap_or(DEFAULT_BIND_ADDRESS.into());
|
let ip = self
|
||||||
|
.address
|
||||||
|
.or_else(|| session.serve_address())
|
||||||
|
.unwrap_or(DEFAULT_BIND_ADDRESS.into());
|
||||||
|
|
||||||
let port = self
|
let port = self
|
||||||
.port
|
.port
|
||||||
|
|||||||
136
src/cli/sourcemap.rs
Normal file
136
src/cli/sourcemap.rs
Normal file
@@ -0,0 +1,136 @@
|
|||||||
|
use std::{
|
||||||
|
io::{BufWriter, Write},
|
||||||
|
path::{Path, PathBuf},
|
||||||
|
};
|
||||||
|
|
||||||
|
use clap::Parser;
|
||||||
|
use fs_err::File;
|
||||||
|
use memofs::Vfs;
|
||||||
|
use rbx_dom_weak::types::Ref;
|
||||||
|
use serde::Serialize;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
serve_session::ServeSession,
|
||||||
|
snapshot::{InstanceWithMeta, RojoTree},
|
||||||
|
};
|
||||||
|
|
||||||
|
use super::resolve_path;
|
||||||
|
|
||||||
|
const PATH_STRIP_FAILED_ERR: &str = "Failed to create relative paths for project file!";
|
||||||
|
|
||||||
|
/// Representation of a node in the generated sourcemap tree.
|
||||||
|
#[derive(Serialize)]
|
||||||
|
#[serde(rename_all = "camelCase")]
|
||||||
|
struct SourcemapNode {
|
||||||
|
name: String,
|
||||||
|
class_name: String,
|
||||||
|
|
||||||
|
#[serde(skip_serializing_if = "Vec::is_empty")]
|
||||||
|
file_paths: Vec<PathBuf>,
|
||||||
|
|
||||||
|
#[serde(skip_serializing_if = "Vec::is_empty")]
|
||||||
|
children: Vec<SourcemapNode>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Generates a sourcemap file from the Rojo project.
|
||||||
|
#[derive(Debug, Parser)]
|
||||||
|
pub struct SourcemapCommand {
|
||||||
|
/// Path to the project to use for the sourcemap. Defaults to the current
|
||||||
|
/// directory.
|
||||||
|
#[clap(default_value = "")]
|
||||||
|
pub project: PathBuf,
|
||||||
|
|
||||||
|
/// Where to output the sourcemap. Omit this to use stdout instead of
|
||||||
|
/// writing to a file.
|
||||||
|
///
|
||||||
|
/// Should end in .json.
|
||||||
|
#[clap(long, short)]
|
||||||
|
pub output: Option<PathBuf>,
|
||||||
|
|
||||||
|
/// If non-script files should be included or not. Defaults to false.
|
||||||
|
#[clap(long)]
|
||||||
|
pub include_non_scripts: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SourcemapCommand {
|
||||||
|
pub fn run(self) -> anyhow::Result<()> {
|
||||||
|
let project_path = resolve_path(&self.project);
|
||||||
|
|
||||||
|
log::trace!("Constructing in-memory filesystem");
|
||||||
|
let vfs = Vfs::new_default();
|
||||||
|
|
||||||
|
let session = ServeSession::new(vfs, &project_path)?;
|
||||||
|
let tree = session.tree();
|
||||||
|
|
||||||
|
let filter = if self.include_non_scripts {
|
||||||
|
filter_nothing
|
||||||
|
} else {
|
||||||
|
filter_non_scripts
|
||||||
|
};
|
||||||
|
|
||||||
|
let root_node = recurse_create_node(&tree, tree.get_root_id(), session.root_dir(), filter);
|
||||||
|
|
||||||
|
if let Some(output_path) = self.output {
|
||||||
|
let mut file = BufWriter::new(File::create(&output_path)?);
|
||||||
|
serde_json::to_writer(&mut file, &root_node)?;
|
||||||
|
file.flush()?;
|
||||||
|
|
||||||
|
println!("Created sourcemap at {}", output_path.display());
|
||||||
|
} else {
|
||||||
|
let output = serde_json::to_string(&root_node)?;
|
||||||
|
println!("{}", output);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn filter_nothing(_instance: &InstanceWithMeta) -> bool {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
fn filter_non_scripts(instance: &InstanceWithMeta) -> bool {
|
||||||
|
match instance.class_name() {
|
||||||
|
"Script" | "LocalScript" | "ModuleScript" => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn recurse_create_node(
|
||||||
|
tree: &RojoTree,
|
||||||
|
referent: Ref,
|
||||||
|
project_dir: &Path,
|
||||||
|
filter: fn(&InstanceWithMeta) -> bool,
|
||||||
|
) -> Option<SourcemapNode> {
|
||||||
|
let instance = tree.get_instance(referent).expect("instance did not exist");
|
||||||
|
|
||||||
|
let mut children = Vec::new();
|
||||||
|
for &child_id in instance.children() {
|
||||||
|
if let Some(child_node) = recurse_create_node(tree, child_id, &project_dir, filter) {
|
||||||
|
children.push(child_node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this object has no children and doesn't pass the filter, it doesn't
|
||||||
|
// contain any information we're looking for.
|
||||||
|
if children.is_empty() && !filter(&instance) {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let file_paths = instance
|
||||||
|
.metadata()
|
||||||
|
.relevant_paths
|
||||||
|
.iter()
|
||||||
|
// Not all paths listed as relevant are guaranteed to exist.
|
||||||
|
.filter(|path| path.is_file())
|
||||||
|
.map(|path| path.strip_prefix(project_dir).expect(PATH_STRIP_FAILED_ERR))
|
||||||
|
.map(|path| path.to_path_buf())
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
Some(SourcemapNode {
|
||||||
|
name: instance.name().to_string(),
|
||||||
|
class_name: instance.class_name().to_string(),
|
||||||
|
file_paths,
|
||||||
|
children,
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -2,30 +2,38 @@ use std::path::PathBuf;
|
|||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use anyhow::{bail, format_err, Context};
|
use anyhow::{bail, format_err, Context};
|
||||||
|
use clap::Parser;
|
||||||
use memofs::Vfs;
|
use memofs::Vfs;
|
||||||
use reqwest::{
|
use reqwest::{
|
||||||
header::{ACCEPT, CONTENT_TYPE, COOKIE, USER_AGENT},
|
header::{ACCEPT, CONTENT_TYPE, COOKIE, USER_AGENT},
|
||||||
StatusCode,
|
StatusCode,
|
||||||
};
|
};
|
||||||
use structopt::StructOpt;
|
|
||||||
|
|
||||||
use crate::{auth_cookie::get_auth_cookie, serve_session::ServeSession};
|
use crate::{auth_cookie::get_auth_cookie, serve_session::ServeSession};
|
||||||
|
|
||||||
use super::resolve_path;
|
use super::resolve_path;
|
||||||
|
|
||||||
/// Builds the project and uploads it to Roblox.
|
/// Builds the project and uploads it to Roblox.
|
||||||
#[derive(Debug, StructOpt)]
|
#[derive(Debug, Parser)]
|
||||||
pub struct UploadCommand {
|
pub struct UploadCommand {
|
||||||
/// Path to the project to upload. Defaults to the current directory.
|
/// Path to the project to upload. Defaults to the current directory.
|
||||||
#[structopt(default_value = "")]
|
#[clap(default_value = "")]
|
||||||
pub project: PathBuf,
|
pub project: PathBuf,
|
||||||
|
|
||||||
/// Authenication cookie to use. If not specified, Rojo will attempt to find one from the system automatically.
|
/// Authenication cookie to use. If not specified, Rojo will attempt to find one from the system automatically.
|
||||||
#[structopt(long)]
|
#[clap(long)]
|
||||||
pub cookie: Option<String>,
|
pub cookie: Option<String>,
|
||||||
|
|
||||||
|
/// API key obtained from create.roblox.com/credentials. Rojo will use the Open Cloud API when this is provided. Only supports uploading to a place.
|
||||||
|
#[clap(long = "api_key")]
|
||||||
|
pub api_key: Option<String>,
|
||||||
|
|
||||||
|
/// The Universe ID of the given place. Required when using the Open Cloud API.
|
||||||
|
#[clap(long = "universe_id")]
|
||||||
|
pub universe_id: Option<u64>,
|
||||||
|
|
||||||
/// Asset ID to upload to.
|
/// Asset ID to upload to.
|
||||||
#[structopt(long = "asset_id")]
|
#[clap(long = "asset_id")]
|
||||||
pub asset_id: u64,
|
pub asset_id: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,10 +41,6 @@ impl UploadCommand {
|
|||||||
pub fn run(self) -> Result<(), anyhow::Error> {
|
pub fn run(self) -> Result<(), anyhow::Error> {
|
||||||
let project_path = resolve_path(&self.project);
|
let project_path = resolve_path(&self.project);
|
||||||
|
|
||||||
let cookie = self.cookie.or_else(get_auth_cookie).context(
|
|
||||||
"Rojo could not find your Roblox auth cookie. Please pass one via --cookie.",
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let vfs = Vfs::new_default();
|
let vfs = Vfs::new_default();
|
||||||
|
|
||||||
let session = ServeSession::new(vfs, project_path)?;
|
let session = ServeSession::new(vfs, project_path)?;
|
||||||
@@ -54,7 +58,36 @@ impl UploadCommand {
|
|||||||
|
|
||||||
log::trace!("Encoding binary model");
|
log::trace!("Encoding binary model");
|
||||||
rbx_binary::to_writer(&mut buffer, tree.inner(), &encode_ids)?;
|
rbx_binary::to_writer(&mut buffer, tree.inner(), &encode_ids)?;
|
||||||
do_upload(buffer, self.asset_id, &cookie)
|
|
||||||
|
match (self.cookie, self.api_key, self.universe_id) {
|
||||||
|
(cookie, None, universe) => {
|
||||||
|
// using legacy. notify if universe is provided.
|
||||||
|
if universe.is_some() {
|
||||||
|
log::warn!(
|
||||||
|
"--universe_id was provided but is ignored when using legacy upload"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let cookie = cookie.or_else(get_auth_cookie).context(
|
||||||
|
"Rojo could not find your Roblox auth cookie. Please pass one via --cookie.",
|
||||||
|
)?;
|
||||||
|
do_upload(buffer, self.asset_id, &cookie)
|
||||||
|
}
|
||||||
|
|
||||||
|
(cookie, Some(api_key), Some(universe_id)) => {
|
||||||
|
// using open cloud. notify if cookie is provided.
|
||||||
|
if cookie.is_some() {
|
||||||
|
log::warn!("--cookie was provided but is ignored when using Open Cloud API");
|
||||||
|
}
|
||||||
|
|
||||||
|
do_upload_open_cloud(buffer, universe_id, self.asset_id, &api_key)
|
||||||
|
}
|
||||||
|
|
||||||
|
(_, Some(_), None) => {
|
||||||
|
// API key is provided, universe id is not.
|
||||||
|
bail!("--universe_id must be provided to use the Open Cloud API");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -90,7 +123,7 @@ fn do_upload(buffer: Vec<u8>, asset_id: u64, cookie: &str) -> anyhow::Result<()>
|
|||||||
asset_id
|
asset_id
|
||||||
);
|
);
|
||||||
|
|
||||||
let client = reqwest::Client::new();
|
let client = reqwest::blocking::Client::new();
|
||||||
|
|
||||||
let build_request = move || {
|
let build_request = move || {
|
||||||
client
|
client
|
||||||
@@ -125,3 +158,38 @@ fn do_upload(buffer: Vec<u8>, asset_id: u64, cookie: &str) -> anyhow::Result<()>
|
|||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Implementation of do_upload that supports the new open cloud api.
|
||||||
|
/// see https://developer.roblox.com/en-us/articles/open-cloud
|
||||||
|
fn do_upload_open_cloud(
|
||||||
|
buffer: Vec<u8>,
|
||||||
|
universe_id: u64,
|
||||||
|
asset_id: u64,
|
||||||
|
api_key: &str,
|
||||||
|
) -> anyhow::Result<()> {
|
||||||
|
let url = format!(
|
||||||
|
"https://apis.roblox.com/universes/v1/{}/places/{}/versions?versionType=Published",
|
||||||
|
universe_id, asset_id
|
||||||
|
);
|
||||||
|
|
||||||
|
let client = reqwest::blocking::Client::new();
|
||||||
|
|
||||||
|
log::debug!("Uploading to Roblox...");
|
||||||
|
let response = client
|
||||||
|
.post(&url)
|
||||||
|
.header("x-api-key", api_key)
|
||||||
|
.header(CONTENT_TYPE, "application/xml")
|
||||||
|
.header(ACCEPT, "application/json")
|
||||||
|
.body(buffer)
|
||||||
|
.send()?;
|
||||||
|
|
||||||
|
let status = response.status();
|
||||||
|
if !status.is_success() {
|
||||||
|
bail!(
|
||||||
|
"The Roblox API returned an unexpected error: {}",
|
||||||
|
response.text()?
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user