LSL Protocol/Restrained Love Open Relay Group/ORG Requirements/0004 draft
STATUS: draft
VERSION: 0004
Summary
This page is a list of requirements for a relay or an in-world device to have the right to be called "ORG compliant". It is mostly about adding support for an ORG-centered informational metacommand and specifying Marine Kelley's RLVR protocol a bit more that it is now, while of course preserving full compatibility.
For a quick overview of the motivations and general working of this protocol and a few examples, we refer the reader to the original specification page by Marine Kelley.
If you are making furniture, traps, or other RLVR controllers and you are not interested in what optional ORG x-tensions can offer, then you can stick with the original specification by Marine Kelley and still be compatible with ORG relays. However, to have the right to call your controller "ORG compatible", there are still a few light additional requirements such as:
- using only llRegionSayTo for communications with the relay,
- not spamming the RLVR channel with messages that do not belong to the protocol,
- and remain responsive to ping messages at every time if you want the session to remain locked.
If you are interested in ORG x-tensions, then you should also read the part about !x-orgversions.
If you are making a relay, sorry we are afraid you will have to read all this page.
Definition of an ORG device
A RLVR device is compliant with ORG specifications if it implements the consolidated RLVR protocol described below along with the optional ORG x-tensions it announces supporting.
Note in particular that
- the consolidated protocol is backward compatible with Marine Kelley's RLVR protocol version 1.100 but resolves most loose ends of that document;
- ORG relays handle controller messages whose second field is ffffffff-ffff-ffff-ffff-ffffffffffff exactly the same as if it was the key of the wearer (NULL_KEY is a wildcard);
- one-to-one chat messages (not using a wildcard) are sent by using llRegionSayTo LSL function;
- there is an additional meta-command, !x-orgversions for a relay to announce supported x-tensions;
- each supported x-tension can imply other supported metacommands, ORG x-tensions meta-command all having names starting with prefix "!x-";
- other proprietary metacommands with prefix "!_-" (replace "_" by any letter that is not "x") can be supported;
Supported x-tensions can constrain the core specification (this document) even more, but cannot alter it or make it less constraining unless as a result of the relay executing a meta-command of this x-tension sent by the controller. Thus, although the protocol is extensible, a controller can always rely on the fact an ORG relay abides by the requirements in this document until it asks the relay to do otherwise.
ORG devices may be advertised by the use of the ORG logo for advertising compliance with ORG requirements. ORG devices makers are encouraged to do so.
Consolidated RLVR protocol
Protocol Syntax
Summary of differences with original RLVR:
- we force use of llRegionSayTo,
- we explicitly forbid incorrect syntax on the relay channel
- the relay is supposed to ignore bad commands
- wildcards are supported
Definitions
- The Restrained Love Relay (RLVR) protocol implies two partners (SL objects): the relay, attachment worn by the avatar commands are intended for, and a controller, which sends commands to the relay.
- The goal of the protocol is to force the relay wearer to make perform actions and to toggle restrictions on their behavior. Both actions and restrictions can be enforced either through use of viewer hooks (Restrained Love API, or RLV) or through LSL scripting.
- Hence we distinguish between "RLV commands", starting with "@", which are commands following the syntax of the RLV API and which will ultimately be forwarded to the viewer, and "meta-commands", starting with "!", which are meant to be interpreted by LSL functions.
Requirements
RLVR protocol consists of messages exchanged between relay and controller
- on channel -1812221819 (called RLVR channel),
- using the LSL function llRegionSayTo,
- all sent from the same prim of the controller or relay (within a session)
- and having the following syntax (non-terminal <c_msg> for controller messages/command messages and <r_msg> for relay messages/acknowledgement messages):
<c_msg> ::= IDENT "," KEY "," <commands> <r_msg> ::= IDENT "," KEY "," <command> "," ACK <commands> ::= <command> | <command> "|" <commands> <command> ::= <rlvcommand> | <metacommand> <rlvcommand> ::= "@" RLVCOMMAND <metacommand> ::= "!" METACOMMANDNAME <metacommandarguments> <metacommandarguments> ::= "" | "/" ARG <metacommandarguments>
where we explicit the different terminal tokens:
IDENT: any string without "," KEY: any UUID ACK: any string without "," RLVCOMMAND: any string without "," METACOMMANDNAME: any string without "/" and "," ARG: any string without "/" and ","
- Relays can only send <r_msg>'s, and controllers <c_msg>'s. Nothing else is allowed on RLVR channel.
- In <c_msg>, KEY is always either the uuid of the relay wearer or ffffffff-ffff-ffff-ffff-ffffffffffff (wildcard) if it is a message intended for all relays who can hear it.
- In <r_msg>, KEY is always the uuid of the controller.
On the relay side only
Further requirements:
- A relay MUST ignore any message on RLVR channel which is not a <c_msg> with KEY being either the key of the avatar wearing the relay or ffffffff-ffff-ffff-ffff-ffffffffffff.
Then, for every <command>:
- if <command> is neither a valid <rlvcommand> or <metacommand>, the relay must ignore it and process the next <command>'s;
- if <command> is a <rlvcommand> or a <metacommand>, the relay MUST send back to the controller a <r_msg> (see above) such that:
- IDENT is the same as IDENT in the <c_msg> being processed,
- <command> is the subcommand being processed
- and ACK is a string called acknowledgement value. By default the acknowledgement value is "ok" if the relay will execute the <command >, "ko" otherwise.
Remarks:
- Many meta-commands may have other acknowledgements than "ok" or "ko" (or none, for !pong). Their definitions supersede this specification.
- Note that unknown meta-commands cannot be executed and thus wil be acknowledged by "ko".
- Known meta-commands which are sent with not enough parameters or parameters with types incompatible with those in the definition known by the relay also cannot be executed and will also be acknowledged by "ko". (ex: the relay knows !x-who/key, but receives !x-who/integer/string).
On executing commands
Summary of differences with original RLVR:
- explanation on how to handle meta-commands with variable number of parameters,
- the relay knows commands in both core specification and x-tensions,
- clarification on multi-controller relays.
Relay side
Requirements:
- If a <rlvcommand> is acknowledged with value "ok", the relay ensures the viewer is affected the same way as if the LSL instruction llOwnerSay(<rlvcommand>) was executed by a script in a primitive that handles only one controller.
- When processing a <metacommand>, if the relay knows at least one definition of '!'+METACOMMANDNAME for which the received <metacommandarguments> provide enough parameters, then it will be handled according to the definition handling the most parameters among the latter. Extra parameters are ignored.
- A relays knows the following meta-command definitions:
- definitions listed in the core requirements
- all definitions included in x-tensions announced by the relay.
- When acknowledging a <metacommand> by anything else than "ko", the relay commits itself to execute it.
Remarks:
- For first requirement, it must be understood that the reference behavior is that the relay would have if it had one controller per prim. However handling several controllers from one unique prim is possible but requires taking care that different controllers do not interfere with each other, releasing each other's restrictions, for instance.
- For the second requirement, the word "handled" means that if the meta-command with these parameters is known, then requirements in known definition cannot be ignored. Sometimes, this can imply that the meta-command cannot be refused, for instance.
Controller side
- The controller can send any command within allowed syntax.
- Only commands described in this specification and commands belonging to x-tensions announced by the relay are guaranteed to be understood by the relay. If not understood, the relay will just answer "ko", do nothing and skip to the next command.
Protocol session
Summary of differences with original RLVR:
- it is now clear when a session starts and stops (and what a session is!)
- it is clear when to expect interactive dialogs (and how to avoid spamming relay wearers with blue dialogs)
- no session for unreachable controllers (answer to ping is mandatory to ensure session persistence)
Definitions
- During an execution of the RLVR protocol, a relay and a controller establish a session together. A relay may manage several sessions with several controllers. Likewise, a controller could establish sessions with several relays.
- Restrictions on a relay wearer are relative to a session. So are most other variables (authorization state, key of the controller, ... ).
- A session is Opened whenever the relay accepts a RLVR command from a controller.
- A session is Closed (or released) as soon as the relay forgets every variables and releases all restrictions pertaining to that session.
- A session is Locked whenever it holds active restrictions.
- A session is Authed if the controller has been allowed to control the relay (not only store some variables with no consequence). Once authed, no interactive "ask" dialog should pop up anymore concerning this session. Most commands should now normally be accepted (and maybe a few automatically rejected).
- A controller is called reachable if it can receive messages from the relay and can prove it by answering ping requests from the relay. By "can", it is meant by the method of communication agreed upon by the relay and controller (by default: chat messages sent on RLVR channel by llRegionSayTo). X-tensions such as email and delay can modify what is meant by reachable.
Relay side requirements
On closing sessions:
- Closing a session clears all RLV restrictions that belong only to that session, clears all session variables and disables all restrictions relative to that session. This MUST happen on the following events and no other:
- on executing !release (received from the controller, or relay-triggered as in a safeword, see #!release)
- on noticing the controller is not reachable
- on timeout, if the session is not Locked
- Sessions that are not locked MUST be closed after a reasonable timeout.
On interactive Auth:
- Unless specified otherwise (see !x-who), no command can trigger an interactive dialog unless the command requires an Authed session and the session is not already Authed.
- Relay implementations MAY require an authed session for all <rlvcommand>'s of the form @xxx=force, @xxx=add and @xxx=n, and for meta-commands whose specifications say it is allowed.
- Other RLV commands MUST automatically accepted or rejected by any relay.
On removing and adding restrictions:
- Commands that remove restrictions are always accepted (RLV @...=<channel_number>, @clear[=...] and @...=y/rem commands, !release, and !x-.../clear[/...] meta-commands).
- No restriction can be released unless as the result of one of the above commands, or of closing a session.
- @xxx=n/add <rlvcommand>'s, after being accepted, add a restriction and therefore Lock the session.
- Some meta-commands described in x-tensions can also add restrictions and therefore Lock the session.
Reachability:
- A relay MUST provide a mechanism for checking that every controlling device is reachable. It is not specified whether this checking process should be automatic or manually triggered.
Controller side requirements
- A controller MUST eventually close any Locked session using the meta-command !release.
In particular, non-portable controllers should not let a Locked relay wearer run free out of reach. Moreover, any controller should either react to some kind of event that is bound to happen (timer expiration being the most common) or provide a user interface for closing the session. It should be clear at least to the person who set the trap how the victim will be released.
Core meta-commands and associated mechanisms
An ORG device MUST at least support the meta-commands !release, !pong, !version, !implversion and !x-orgversions described below and implement their associated mechanisms.
Particular meta-commands definitions and mechanisms may override general requirements above.
New in ORG compared to RLVR: !x-orgversions
!release
Sent by a controller, this command closes the session.
- !release obviously opens no session, requires no Auth and is not Locking.
- If there is an actual session to close !release MUST be accepted by the relay and acknowledeged by "ok".
- On accepting !release, the relay erases all data pertaining to the session and disables all restrictions of the session.
- When a relay decides to close a session on its own (user safeword, for instance) and it is not due to the controller being unreachable, the relay must send the following acknowledgement message:
release,<controller_key>,!release,ok
When receiving the latter message, the controller should act accordingly, for instance by resetting itself and making itself ready for the next session.
Note that !release is not the same as @clear. @clear is interpreted by the relay as a RLV command and, as such, can only clear RLV restrictions. If RLV restrictions were the only restrictions of the session, then the session will be closed, but some x-tensions also define pure LSL based restrictions.
ping/!pong
The ping/!pong mechanism exists so that relays can check presence and responsiveness of a controller. This was primarily meant to be used on relay wearer relog, in order to reactivate all restrictions so that sessions persist on relog if the controller still exists and is ready to continue the session.
!pong cannot require auth and does not lock a session
We call ping message a message from the relay, having the following syntax:
ping,<controller_key>,ping,ping
We call !pong message a message from the controller, having the following syntax:
ping,<relay_key>,!pong
Relay side
- A relay SHOULD NOT send an acknowledgement in response to !pong.
- A ping message can be sent any time by the relay to a known controller.
- On wearer relog, for each existing Locked session, the relay must send a ping message to the session controller.
- When a !pong message is received from a controller, if it was a reply to a previous ping message:
- if the session RLV restrictions contain unsit and the avatar was sitting on some object when the restriction was applied, then the relay MUST force the avatar to sit back on the same object (if still existing);
- the relay MUST make sure all restrictions belonging to the session are made active again.
- If a ping message is sent and is not answered by the controller, then the relay closes the session.
Note that the relay is responsible for storing restrictions and the key of the sit target across sessions, not the controller.
Controller side
- Unless it does not matter that the relay could close the session (or for another reason, the controller knows the relay will not close it), a controller SHOULD listen to ping messages and answer any ping message with a !pong message.
- A controller should ignore ping requests from relays it is not ready to handle.
Note controller don't have to poll for relay presence to resume a session but should just listen for ping messages instead. The relay should have stored all restrictions and handle sitting back to whatever furniture the relay wearer was tied up to.
Concerning the second bullet: for instance, if the controller is a furniture which is already in use by another relay wearer, it would be cause dysfunctions to let the pinging relay restore the restrictions without letting its wearer sit on the furniture. Or if the controller is not stateless and the state for the pinging relay has not been saved before, resuming the session from the initial state might be inconsistant.
!version
The purpose of this meta-command is to query the version of the RLVR protocol supported by a relay.
- !version opens no session (and requires no auth, neither can lock a session)
- A relay MUST always acknowledge !version with the version number of the RLVR protocol it supports ("1100" here, since ORG 0004 includes RLVR 1.100).
Note this is the most common command for a controller to scan for nearby relays, as non-ORG relays also accept this command and answer it the way described above.
!implversion
The purpose of this meta-command is to query the name and version of the relay implementation.
- !implversion opens no session (and requires no auth, neither can lock a session)
- A relay MUST always acknowledge !implversion with its implementation version string... which can be any string.
It is recommended this string includes the following elements:
- the relay name/brand/identifier,
- the version number of that relay name/brand/identifier,
- at the beginning, the string prefix "ORG=0004/"
The latter is convenient for checking for ORG compatibility using a command even non-ORG relay should understand.
!x-orgversions
This meta-command purpose it two-fold. First it allows a controlling device to ask a relay for the list of ORG x-tensions it supports and, second, similarily to !version, !x-orgversions can be used to know what version of ORG the relay complies to.
Example:
C<->R -> query,rUUID,!x-orgversions <- query,cUUID,!x-orgversions,ORG=0004/who=002/email=006
Requirements:
- !x-orgversions should always be accepted with no Auth required, it does not open a session or Lock one.
- The acknowledgement value in a reply to !x-orgversions is a "/"-list of items of the form <package>=<version>. The first <package> is always "ORG" (representing the core specification package of ORG relays), its version string is the version of the specification (4 digits). The other packages are ORG x-tension names, whose version string is 3 digits.
- If several incompatible versions of a same x-tension are supported, they must be listed ordered by version number. Ex: ORG=0004/who=002/email=005/email=006. If versions are backward compatible, only the last version should be reported.
Remarks:
- In a controller, it is however recommended to use !x-orgversions as soon as possible in order to avoid sending unsupported and useless metacommands later on.
- In the acknowledgement message, x-tension names are listed, not meta-command names (as one x-tension can have several meta-commands). Thus listed names do not include the prefix "!x-".