Every DATEX pointer can be shared and synchronized between multiple endpoints.
This chapter serves as an introduction to shared pointers and explains the concepts behind pointer synchronisation.
The Supranet can be regarded as one continuously running global process that has access to a shared memory. Endpoints within the Supranet are running separate threads within this global process. They all have (limited) access to the data within the shared memory.
Every DATEX pointer that is created on an endpoint becomes part of this globally shared memory. This does not mean that pointers are automatically sent to other endpoints when they are created, but they can always be accessed by other endpoints when needed (as long as the endpoint has the required permissions to access the pointer).
For this demonstration, you need two separate browser clients with a developer console.
Alternatively, you can start a Deno CLI process by running the deno
command.
To initialize the endpoints, first import the DATEX runtime:
await import("https://cdn.unyt.org/datex-core-js-legacy/datex.js")
Now, you can connect to the supranet:
await Datex.Supranet.connect()
After a successful initialization, you should see the connection info printed to the console:
Note
You can learn more about supranet connections in the chapter Supranet Networking.
As explained in the chapter Pointers, you can now create a new pointer containing a number value.
const x = $(42)
x.val // -> 42
Each DATEX pointer has a globally unique address (pointer id) that is linked to an endpoint. With this id, a pointer can be found in the supranet.
The pointer id can be accessed via the id
property of a pointer instance:
x.id // -> e.g. "D5A3CB02310Dx480B651422749F9x40C85600300"
Per default, pointers have read/write restrictions and can only be accessed by the endpoint that created them or other instances of the same endpoint.
You can explicitly grant access for a pointer to a specific endpoint:
const y = $(43);
grantAccess(y, '@user1'); // @user1 can now read/write x
revokeAccess(y, '@user1'); // @user1 can no longer read/write x
You can also make a pointer publicly accessible by all endpoints, which
we will do for our example pointer x
:
grantPublicAccess(x); // any endpoint can now read/write x
Note
To allow all remote endpoints to access pointers that are created on the local endpoint, you can disable the runtime option PROTECT_POINTERS
:
Datex.Runtime.OPTIONS.PROTECT_POINTERS = false
We want to emphasize that this is a major security risk and should only be used in controlled environments and for testing purposes.
Now that we know the id of the pointer x
and have made it publicly accessible,
we can access it from another endpoint:
Open a new browser tab or Deno CLI and follow the same steps to load the DATEX runtime and connect to the supranet:
await import("https://cdn.unyt.org/datex-core-js-legacy/datex.js")
await Datex.Supranet.connect()
You can use the $
shortcut to load the remote pointer with the previously determined id, e.g.:
const x = await $.D5A3CB02310Dx480B651422749F9x40C85600300
The variable x
now holds a reference to the same pointer that we
created on the other client before.
You can verify this by reading the pointer value:
x.val // -> 42
The pointer that we stored in the variable x
is not just a static value - its value is updated bidirectionally.
That means that any changes on the original endpoint are reflected on the second endpoint, and vice-versa - try it out for yourself:
Update value on first client:
x.val = 10
Read value on second client:
x.val // -> 10
Pointer synchronisation does not just work with primitive values, but also with objects, maps, sets, etc.
DATEX has a global garbage collection mechanism that handles shared pointers across the network.
Global garbage collection works with both primitive and complex pointers.
-
If a pointer was loaded from a remote endpoint, it is garbage collected if there are no local references to the pointer
-
If a pointer was created by the local endpoint, it is only garbage collected if the following conditions are fulfilled:
- There are no local references to the pointer
- There are no references to the pointer on other endpoints
The DATEX JS Runtime guarantees that two pointers with the same id are always pointing to the same instance:
const x1 = await $.ABCDEF
const x2 = await $.ABCDEF
assert (x1 === x2)
The identity of a pointer is always preserved. Even if you receive the pointer from a remote endpoint call, it is still identical to the local instance:
// remote endpoint
type User = {name: string, age: number}
const happyBirthday = $(
function (user: User) {
user.age++;
return user;
}
)
// local endpoint
const user = $({
name: "Luke",
age: 20
})
const happyBirthday = await $.DEFABCDEF // <- pointer id for the remote "happyBirthday" function
const olderUser = await happyBirthday(user);
user === olderUser // true
olderUser.age === 21 // true
user.age === 21 // true
Like we would expect if this was a normal, local JavaScript function call, the returned
user
object is identical to the object we passed to the function.
This is not only true for this simple example, but also for more complex scenarios.
For example, reference identities are also preserved within eternal values.