Install Unity 2022 LTS, clone this repository using git clone --recurse-submodules
, and open it in Unity.
Unity should automatically install required packages. This includes the ParrelSync package,
which is useful for testing multiplayer functionality in-editor.
The Rider IDE is recommended, it has direct integration with Unity.
The game can be started from Scenes/MainScene
. To run, select PlayMode type from Server, Client, or Server & Client in Multiplayer -> Window: PlayMode Tools
, and set an address,
by default 127.0.0.1:7979
. Then press the play button. Additional configuration can be specified in command line arguments,
which can be set in-editor on the Editor Args
field of the GameBootstrap -> Editor Cmd Args
singleton component.
Opencraft 2 is built by the Unity editor, using File -> Build Settings
with Platform
set to Windows, Mac, Linux
and
Target Platform
set to Linux
or Windows
(Mac is untested). For debugging and analysis, the Development Build
flag must be set. The builds folder is location under ./Builds/
. This folder also contains the Docker files for containerizing Opencraft 2.
Opencraft 2 can be run as a container. The main game container can be built from the ./Builds/
folder using
docker build -t jerriteic/opencraft2:base .
. This base image depends on jerriteic/gpu_ubuntu20.04
which allows
container application to run graphics applications with NVIDIA GPU hardware acceleration. That container can
be built in the same folder using the Dockerfile.ubuntugpu
dockerfile.
The game container runs a hardware-accelerated 3D graphics application through VirtualGL. This requires extensive configuration to the host platform:
- Install correct NVIDIA drivers for your platform and GPU.
- Container is tested on Ubuntu 20.04 on NVIDIA Tesla T4 with proprietary driver version 535.
- Install Docker + NVIDIA container toolkit
- On headless hosts, configure display settings according to this guide.
- Do not reinstall the NVIDIA drivers, skip the step marked
Install NVIDIA Driver from CUDA .run shell script
.
- Do not reinstall the NVIDIA drivers, skip the step marked
If the Docker requirements are met, run the container using the following command:
docker run -it \
--runtime=nvidia \
--name=super-container \
--security-opt seccomp=unconfined \
--init \
--net=host \
--privileged=true \
--rm=false \
-e DISPLAY=:0 \
-v /tmp/.X11-unix/X0:/tmp/.X11-unix/X0:ro \
-v /etc/localtime:/etc/localtime:ro \
-v /share-docker:/share-docker \
jerriteic/opencraft2:base
Warning
Running Docker containers with these flags gives them UNRESTRICTED HOST ROOT ACCESS! Act accordingly.
ParrelSync allows synchronizing multiple copies of a Unity project. These
can be run in parallel to test networked functionality. To create a new ParrelSync clone, in ParrelSync -> Clones Manager
select Add new clone
.
Note
ParrelSync clones require manual setup to fix package dependencies
Inside the new ParrelSync project clone folder, create a symlink to the original project's UnityRenderStreaming folder
using the following command:ln -s <ORIGINAL_PROJECT>/UnityRenderStreaming UnityRenderStreaming
for Linux,
or mklink /D UnityRenderStreaming <ORIGINAL_PROJECT>/UnityRenderStreaming UnityRenderStreaming
for Windows.
This project supports a single game client acting as a streamed gaming host for many players on guest clients.
Testing Multiplay is done using ParrelSync, in the clone's launch arguments in the clone manager add -multiplayRole Guest
to run it as a Multiplay guest client.
Multiplay functionality requires a signalling service to establish a direct connection between host and guest clients.
The signaling service is run as a webapp, the source is available in ./UnityRenderStreaming/WebApp/
and can be build using ./UnityRenderStreaming/pack_webapp.sh
which has npm
as a dependency.
The webapp can be run with a convenience script .Builds/Multiplay_WebApp/start.sh
or directly with .\webserver -p <PORT>
.
The port the webserver listens on must be the same as the signaling port configured using the
application command line argument -signalingPort <int>
.
Inspecting existing entities, components, systems, and queries in each existing world can be done using
Entities information pages under Window -> Entities
. Information not specific to ECS can be found under Window -> Analysis
,
with the Profiler
being particularly important for determining performance impact of various changes.
Burst compiled generated code can be viewed in Jobs -> Burst -> Open Inspector
, though this is mainly useful for low level optimization work.
Multiplayer functionality and behaviour can be visualized using the Window -> Multiplayer -> Window: NetDbg (Browser)
tool,
which automatically gathers state snapshot metrics from Netcode for Entities connections.
If your Unity or your IDE starts giving strange errors, particularly about packages,
it is worth trying Edit > Preferences -> External Tools -> Regenerate project files
. As a last resort, Reimport All
through the right-click menu in the Project
window.
The Opencraft 2 application can be configured in a variety of ways, with some methods being different in-editor and
when run standalone. In editor, the hierarchy of configuration is Deployment Graph
>Multiplayer PlayMode Tools
>Editor Args
> localConfigJson
.
Argument | Options | Default | Description |
---|---|---|---|
-deploymentJson | FilePath | null | Path to a deployment configuration Json file. |
-deploymentID | int | -1 | The ID of this node, used for deployment. |
-remoteConfig | true/false | false | Should this node fetch configuration from a deployment service. |
-deploymentURL | URL | 127.0.0.1 | URL of the deployment service. |
-deploymentPort | int | 7980 | Port of the deployment service. |
-debug | true/false | false | Enable/disable verbose logging. |
-seed | string | "42" | Seed used for terrain generation. |
-playType | ClientAndServer Client Server StreamedClient ThinClient |
Client | Mode to start the application in. Can be overriden by remote config. |
-serverUrl | URL | 127.0.0.1 | URL of the game server. |
-serverPort | int | 7979 | Port of the game server. |
-localConfigJson | FilePath | null | Path to a command args Json file. |
-networkTickRate | int | 60 | Rate of network snapshots. |
-simulationTickRate | int | 60 | Rate of simulation steps. |
-takeScreenshots | true/false | false | Take screenshots at intervals. No effect on Server-only builds. |
-screenshotInterval | int | 5 | Set screenshot interval. |
-screenshotFolder | FilePath | Application.persistentDataPath\screenshots | Set screenshot save location. |
-duration | int | -1 | Exit after a set amount of seconds. |
-userID | int | -1 | Player's user ID string. Acts as a username. |
-signalingUrl | URL | ws://127.0.0.1:7981 | URL of the stream signaling service. |
-multiplayRole | Disabled Host Guest |
Disabled | Stream gaming role. |
-switchToStream | int | 0 | Switch to a streamed client after this duration. |
-emulationType | None Record PlaybackExplore PlaybackCollect PlaybackBuild SimulationExplore SimulationCollect SimulationBuild |
None | Player emulation type mode. |
-emulationFile | FilePath | Application.persistentDataPath\recordedInputs.inputtrace | Path to a player emulation trace. |
-numThinClientPlayers | int | 0 | Number of thin clients to run, useful for debugging or experiments. |
-profiler-enable | true/false | false | Automatically starts the profiler module. |
-profiler-log-file | FilePath | null | Specifies a .raw file to write profiler data to. |
-profiler-maxusedmemory | int | 16000000 | Max memory used by profiler in bytes. Default is 16MB. |
-logStats | true/false | false | Log relevant Profiler statistics to file. |
-statsFile | FilePath | Application.persistentDataPath\stats.csv | What file to log statistics to. |
The deployment service constructs a deployment graph based on a configuration file. The deployment configuration file path is set
using the command line argument -deploymentJson <FilePath>
. In editor, a json file can be set on the Deployment Config
field of the GameBootstrap->Cmd Args Reader
singleton component.
The Json is expected to follow this formatting (excluding comments):
{
"nodes":[
{
"nodeID":0, // ID of this node
"nodeIP":"127.0.0.1", // Expected IP of this node, will throw warning if node with nodeID not at this IP.
"worldConfigs":[ // List of worlds to deploy on this node
{
"worldType":"Client", // None, Client, Server, ThinClient
"initializationMode":"CreateAndConnect", // CreateAndConnect, Connect
"multiplayStreamingRoles":"Guest", // None, Guest, Host
"serverNodeID":-1, // The ID of the node to connect a client world to
"streamingNodeID":1, // The ID of the node to connect a streamed guest client world to
"numThinClients":0,
"services":[], // Names of services, handled according to serviceFilterType
"serviceFilterType":"Includes", // Includes, Excludes, Only
"emulationBehaviours":"None" // None, PlaybackExplore, PlaybackCollect, PlaybackBuild, SimulationExplore, SimulationCollect, SimulationBuild
}
]
},
{
"nodeID":1,
"nodeIP":"127.0.0.1",
"worldConfigs":[
{
"worldType":"Server",
"initializationMode":"CreateAndConnect",
"multiplayStreamingRoles":"Disabled",
"serverNodeID":-1,
"streamingNodeID":-1,
"numThinClients":0,
"services":[],
"serviceFilterType":"Includes",
"emulationBehaviours":"None"
},
{
"worldType":"Client",
"initializationMode":"CreateAndConnect",
"multiplayStreamingRoles":"Host",
"serverNodeID":1,
"streamingNodeID":-1,
"numThinClients":0,
"services":[],
"serviceFilterType":"Includes",
"emulationBehaviours":"None"
}
]
}
]
}