Services
Services Defined
Services are singleton provider objects that serve a specific purpose on the server. For instance, a game might have a PointsService, which manages in-game points for the players.
A game might have many services. They will serve as the backbone of a game.
For the sake of example, we will slowly develop PointsService to show how a service is constructed.
Creating Services
In its simplest form, a service can be created like so:
local PointsService = Crystal.CreateService { Name = "PointsService", Client = {} }
return PointsService
The Client
table is optional for the constructor. However, it will be added by Crystal if left out. For the sake of code clarity, it is recommended to keep it in the constructor as shown above.
If the Client
table is omitted, the service will be interpreted as server-side only. This means that the client will not be able to access this service using Crystal.GetService
on the client.
The Name
field is required. This name is how code outside of your service will find it. This name must be unique from all other services. It is best practice to name your variable the same as the service name (e.g. local PointsService
matches Name = "PointsService"
).
The last line (return PointsService
) assumes this code is written in a ModuleScript, which is best practice for containing services.
Adding methods
Services are just simple tables at the end of the day. As such, it is very easy to add methods to services.
function PointsService:AddPoints(player, amount)
-- TODO: add points
end
function PointsService:GetPoints(player)
return 0
end
Adding properties
Again, services are just tables. So we can simply add in properties as we want. In our above method, we are returning 0
for GetPoints()
because we have nowhere to store/retrieve points. Likewise, our AddPoints()
method can't do anything. Let's change that. Let's create a property that holds a table of points per player:
PointsService.PointsPerPlayer = {}
Using methods and properties
Now we can change our AddPoints()
and GetPoints()
methods to use this field.
PointsService.PointsPerPlayer = {}
function PointsService:AddPoints(player, amount)
local points = self:GetPoints(player) -- Current amount of points
points += amount -- Add points
self.PointsPerPlayer[player] = points -- Store points
end
function PointsService:GetPoints(player)
local points = self.PointsPerPlayer[player]
return points or 0 -- Return 0 if no points found for player
end
Using events
What if we want to fire an event when the amount of points changes? This is easy. We can assign an event named PointsChanged
as a property of our service, and have our AddPoints()
method fire the event:
-- Load the Signal module and create PointsChanged signal:
local Signal = require(Crystal.Util.Signal)
PointsService.PointsChanged = Signal.new()
-- Modify AddPoints:
function PointsService:AddPoints(player, amount)
local points = self:GetPoints(player)
points += amount
self.PointsPerPlayer[player] = points
-- Fire event signal, as long as we actually changed the points:
if (amount ~= 0) then
self.PointsChanged:Fire(player, points)
end
end
Another service could then listen for the changes on that event:
function SomeOtherService:CrystalStart()
local PointsService = Crystal.GetService("PointsService")
PointsService.PointsChanged:Connect(function(player, points)
print("Points changed for " .. player.Name .. ":", points)
end)
end
CrystalInit and CrystalStart
In that last code snippet, there's an odd CrystalStart()
method. This is part of the Crystal lifecycle (read more under execution model). These methods are optional, but very useful for orchestrating communication between other services.
When a service is first created, it is not guaranteed that other services are also created and ready to be used. The CrystalInit
and CrystalStart
methods come to save the day! After all services are created and the Crystal.Start()
method is fired, the CrystalInit
methods of all services will be fired.
From the CrystalInit
method, we can guarantee that all other services have been created. However, we still cannot guarantee that those services are ready to be consumed. Therefore, we can reference them within the Init
step, but we should never use them (e.g. use the methods or events attached to those other services).
After all CrystalInit
methods have finished, all CrystalStart
methods are then fired. At this point, we can guarantee that all CrystalInits
are done, and thus can freely access other services.
In order to maintain this pattern, be sure to set up your service in the Init
method (or earlier; just in the ModuleScript itself). By the time CrystalStart
methods are being fired, your services should be available for use.
Cleaning Up Unused Memory
Alright, back to our PointsService! We have a problem... We have created a memory leak! When we add points for a player, we add the player to the table. What happens when the player leaves? Nothing! And that's a problem. That player's data is forever held onto within that PointsPerPlayer
table. To fix this, we need to clear out that data when the player leaves. We can use the CrystalInit
method to hook up to the Players.PlayerRemoving
event and remove the data:
function PointsService:CrystalInit()
game:GetService("Players").PlayerRemoving:Connect(function(player)
-- Clear out the data for the player when the player leaves:
self.PointsPerPlayer[player] = nil
end)
end
While memory management is not unique to Crystal, it is still an important aspect to consider when making your game. Even a garbage-collected language like Lua can have memory leaks introduced by the developer.
Client Communication
Alright, so we can store and add points on the server for a player. But who cares? Players have no visibility to these points at the moment. We need to open a line of communication between our service and the clients (AKA players). This functionality is so fundamental to Crystal, that it's where the name came from: The need to Crystal together communication.
This is where we are going to use that Client
table defined at the beginning.
Methods
Let's say that we want to create a method that lets players fetch how many points they have, and when their points change. First, let's make a method to fetch points:
function PointsService.Client:GetPoints(player)
-- We can just call our other method from here:
return self.Server:GetPoints(player)
end
This creates a client-exposed method called GetPoints
. Within it, we reach back to our top-level service using self.Server
and then invoke our other GetPoints
method that we wrote before. In this example, we've basically just created a proxy for another method; however, this will not always be the case. There will be many times where a client method will exist alone without an equivalent server-side-only method.
Under the hood, Crystal will create a RemoteFunction and bind this method to it.
On the client, we could then invoke the service as such:
-- From a LocalScript
local Crystal = require(game:GetService("ReplicatedStorage").Packages.Crystal)
local PointsService = Crystal.GetService("PointsService")
PointsService:GetPoints():andThen(function(points)
print("Points for myself:", points)
end)
Signals (Server-to-Client)
We should also create a signal that we can fire events for the clients when their points change. We can use Crystal:CreateSignal()
to indicate we want a signal created for the service.
local PointsService = Crystal.CreateService {
Name = "PointsService",
Client = {
PointsChanged = Crystal.CreateSignal(), -- Create the signal
},
}
See the RemoteSignal documentation for more info on how to use the RemoteSignal object.
Under the hood, Crystal is using the Comm
module, which is creating a RemoteEvent object linked to this event. This is a two-way signal (like a transceiver), so we can both send and receive data on both the server and the client.
We can then modify our AddPoints
method again to fire this signal too:
function PointsService:AddPoints(player, amount)
local points = self:GetPoints(player)
points += amount
self.PointsPerPlayer[player] = points
if amount ~= 0 then
self.PointsChanged:Fire(player, points)
-- Fire the client signal:
self.Client.PointsChanged:Fire(player, points)
end
end
And from the client, we can listen for an event on the signal:
-- From a LocalScript
local Crystal = require(game:GetService("ReplicatedStorage").Packages.Crystal)
local PointsService = Crystal.GetService("PointsService")
PointsService.PointsChanged:Connect(function(points)
print("Points for myself now:", points)
end)
Signals (Client-to-Server)
Signal events can also be fired from the client. This is useful when the client needs to give the server information, but doesn't care about any response from the server. For instance, maybe the client wants to tell the PointsService that it wants some points. This is an odd use-case, but let's just roll with it.
We will create another client-exposed signal called GiveMePoints
which will randomly give the player points. Again, this is nonsense in the context of an actual game, but useful for example.
Let's create the signal on the PointsService:
local PointsService = Crystal.CreateService {
Name = "PointsService",
Client = {
PointsChanged = Crystal.CreateSignal(),
GiveMePoints = Crystal.CreateSignal(), -- Create the new signal
},
}
Now, let's listen for the client to fire this signal. We can hook this up in our CrystalInit
method:
function PointsService:CrystalInit()
local rng = Random.new()
-- Listen for the client to fire this signal, then give random points:
self.Client.GiveMePoints:Connect(function(player)
local points = rng:NextInteger(0, 10)
self:AddPoints(player, points)
print("Gave " .. player.Name .. " " .. points .. " points")
end)
-- ...other code for cleaning up player data here
end
From the client, we can fire the signal like so:
-- From a LocalScript
local Crystal = require(game:GetService("ReplicatedStorage").Packages.Crystal)
local PointsService = Crystal.GetService("PointsService")
-- Fire the signal:
PointsService.GiveMePoints:Fire()
See the ClientRemoteSignal documentation for more info on how to use the ClientRemoteSignal object.
Properties
It is often useful to replicate data to all or individual players. Instead of creating methods and signals to communicate this data, RemoteProperties can be used.
For example, let's refactor the AddPoints
method to set a RemoteProperty of the number of points the player has. The client will then be able to
easily read this property:
-- Create the RemoteProperty:
PointsService.Client.Points = Crystal.CreateProperty(0)
function PointsService:AddPoints(player, amount)
local points = self:GetPoints(player)
points += amount
self.PointsPerPlayer[player] = points
self.Client.Points:SetFor(player, points)
end
On the client, we can now easily read the Points
property:
-- LocalScript
local Crystal = require(game:GetService("ReplicatedStorage").Packages.Crystal)
local PointsService = Crystal.GetService("PointsService")
-- The 'Observe' method will fire for the current value and any time the value changes:
PointsService.Points:Observe(function(points)
print("Current number of points:", points)
end)
Using Observe
is the easiest way to track the value of a RemoteProperty on the client.
See the RemoteProperty and ClientRemoteProperty documentation for more info on how to use the RemoteProperty and ClientRemoteProperty objects.
Full Example
PointsService
At the end of this tutorial, we should have a PointsService that looks something like this:
local Crystal = require(game:GetService("ReplicatedStorage").Packages.Crystal)
local Signal = require(Crystal.Util.Signal)
local PointsService = Crystal.CreateService {
Name = "PointsService",
-- Define some properties:
PointsPerPlayer = {},
PointsChanged = Signal.new(),
Client = {
-- Expose signals to the client:
PointsChanged = Crystal.CreateSignal(),
GiveMePoints = Crystal.CreateSignal(),
Points = Crystal.CreateProperty(0),
},
}
-- Client exposed GetPoints method:
function PointsService.Client:GetPoints(player)
return self.Server:GetPoints(player)
end
-- Add Points:
function PointsService:AddPoints(player, amount)
local points = self:GetPoints(player)
points += amount
self.PointsPerPlayer[player] = points
if amount ~= 0 then
self.PointsChanged:Fire(player, points)
self.Client.PointsChanged:Fire(player, points)
end
self.Client.Points:SetFor(player, points)
end
-- Get Points:
function PointsService:GetPoints(player)
local points = self.PointsPerPlayer[player]
return points or 0
end
-- Initialize
function PointsService:CrystalInit()
local rng = Random.new()
-- Give player random amount of points:
self.Client.GiveMePoints:Connect(function(player)
local points = rng:NextInteger(0, 10)
self:AddPoints(player, points)
print("Gave " .. player.Name .. " " .. points .. " points")
end)
-- Clean up data when player leaves:
game:GetService("Players").PlayerRemoving:Connect(function(player)
self.PointsPerPlayer[player] = nil
end)
end
return PointsService
Client Consumer
Example of client-side LocalScript consuming the PointsService:
-- From a LocalScript
local Crystal = require(game:GetService("ReplicatedStorage").Packages.Crystal)
Crystal.Start():catch(warn):await()
local PointsService = Crystal.GetService("PointsService")
local function PointsChanged(points)
print("My points:", points)
end
-- Get points and listen for changes:
PointsService:GetPoints():andThen(PointsChanged)
PointsService.PointsChanged:Connect(PointsChanged)
-- Ask server to give points randomly:
PointsService.GiveMePoints:Fire()