The Off-the-Record messaging protocol (OTR) was originally introduced in the 2004 paper 'Off-the-Record Communication, or, Why Not To Use PGP' [1] and has since become an standard for performing end-to-end encryption. OTR provides confidentiality, deniability, authentication, forward secrecy and post-compromise security.
The OTR protocol itself (in its latest version) is currently described by the document: Off-the-Record Messaging Protocol version 4 [2] and will not be described in detail over here. Instead, this document aims to describe OTR's usage in its current version (version 4) and best practices of using it within XMPP.
This document will refer to OTR in its current version -4-, and only take into account behaviour for version 3 as well, as version 2 and 1 have been deprecated.
Though this document will not focus on the OTR protocol itself, a brief overview is warranted to better understand the protocols strengths and weaknesses.
OTR provides several cryptographic properties depending of what kind of conversation in established. In terms of deniability, OTR (in its version 4) provides offline deniability for any kind of conversations, but only online deniability for both parties during online conversations (it provides online deniability for only one party -the initiator- during offline conversations). It also provides confidentiality, integrity, and authentication (in a deniable matter). Furthermore, it provides forward secrecy and post-compromise security. To achieve all of these properties, OTR uses several cryptographic algorithms, such as DAKEs, SHAKE-256, ChaCha20, SMP, and more.
An OTR session can be held only between two parties, regardless of them being online or offline. Because of this, OTR is incompatible with Multi-User Chat (XEP-0045) [3] and Mediated Information eXchange (MIX) (XEP-0369) [4]; but OTR might support group chat in future versions.
Take into account that OTR, in its version 4, can be implemented over clients following a selection of modes: 'OTRv3-compatible mode', 'OTRv4-standalone mode', and 'OTRv4-interactive-only'. Information about these modes can be found on the main protocol [5].
To participate of an OTR conversation, clients need to set up an OTR library, to optionally decide of a mode that will be used and to generate an instance tag (which works as a device id). They should also publish their instance tag to a device list on the PEP node, so other devices can find it. And lastly, depending of the mode chosen, they should publish specific prekey values.
In order for other devices to be able to initiate a session with a given device (for offline conversations, and when not wanting to use a query message or whitespace tag), it first has to announce itself by adding its instace tag to the devices PEP node.
It is REQUIRED to set the access model of the urn:xmpp:otr:1:devices node to 'open' to give entities without presence subscription read access to the devices and allow them to establish an OTR session. Not having presence subscription is a common occurrence on the first few messages between two contacts.
Devices MUST check that their own device id is contained in the list whenever they receive a PEP update from their own account. If they have been removed, they MUST reannounce themselves.
The device element MAY contain an attribute called label, which is a user defined string describing a device that supports OTR. It is RECOMMENDED to keep the length of the label under 53 Unicode code points. It MUST not contain location, identity or private information, and only refer to the name of the client.
*Example 1. Adding the own instance tag to the device list*
<iq from='ahab@otr.im' type='set' id='announce1'>
<pubsub xmlns='http://jabber.org/protocol/pubsub'>
<publish node='urn:xmpp:otr:1:devices'>
<item id='current'>
<devices xmlns='urn:xmpp:otr:1'>
<device id='c45b041c' label='Beagle Client'/>
<device id='c1bb7b7a' />
<device id='7c91e4ce' label='ChatSecure Client' />
</devices>
</item>
</publish>
<publish-options>
<x xmlns='jabber:x:data' type='submit'>
<field var='FORM_TYPE' type='hidden'>
<value>http://jabber.org/protocol/pubsub#publish-options</value>
</field>
<field var='pubsub#access_model'>
<value>open</value>
</field>
</x>
</publish-options>
</pubsub>
</iq>
A device MUST publish its Prekey values, which can refer to the Client Profile, the Prekey Profile and a set of Prekey Messages. For this, an specific unstrusted Prekey Server should be used.
A participant will find information about a prekey server reading the Service Discovery specification (XEP-0030). The first lookup will be a look up of items of the containing server:
*Example 2. Lookup for server*
<iq from='alice@xmpp.org/notebook'
id='h7ns81g'
to='xmpp.org'
type='get'>
<query xmlns='http://jabber.org/protocol/disco#items'/>
</iq>
The server then returns the services that are associated with it:
*Example 3. Server returns associated services*
<iq from='xmpp.org'
id='h7ns81g'
to='alice@xmpp.org/notebook'
type='result'>
<query xmlns='http://jabber.org/protocol/disco#items'>
<item jid='prekey.xmpp.org'
name='OTR Prekey Server'/>
</query>
</iq>
In order to find an OTRv4 compliant prekey server, the device then needs to send a info request to all items returned from the original call:
*Example 4. Device checks for OTRv4 compliant prekey server*
<iq from='alice@xmpp.org/notebook'
id='info1'
to='prekey.xmpp.org'
type='get'>
<query xmlns='http://jabber.org/protocol/disco#info'/>
</iq>
For a compliant server, this will return the feature
http://jabber.org/protocol/otrv4-prekey-server
and an identity that has
category auth
and type otr-prekey
:
*Example 5. Server returns an OTRv4 compliant prekey server*
<iq type='result'
from='prekey.xmpp.org'
to='alice@xmpp.org/notebook'
id='info1'>
<query xmlns='http://jabber.org/protocol/disco#info'>
<identity
category='auth'
type='otr-prekey'
name='OTR Prekey Server'/>
<feature var='http://jabber.org/protocol/disco#info'/>
<feature var='http://jabber.org/protocol/disco#items'/>
<feature var='http://jabber.org/protocol/otrv4-prekey-server'/>
</query>
</iq>
Finally, before starting to use a Prekey server, you also need to lookup for the fingerprint for this server. This can be find by doing a lookup for items on the server:
*Example 6. Asking for additional items from the prekey server*
<iq from='alice@xmpp.org/notebook'
id='items1'
to='prekey.xmpp.org'
type='get'>
<query xmlns='http://jabber.org/protocol/disco#items'/>
</iq>
This should return an item where node has the value fingerprint
, and the name
will contain the hexadecimal representation of the fingerprint:
*Example 7. Prekey server returns its fingerprint*
<iq type='result'
from='prekey.xmpp.org'
to='alice@xmpp.org/notebook'
id='items1'>
<query xmlns='http://jabber.org/protocol/disco#items'>
<item jid='prekey.xmpp.org'
node='fingerprint'
name='3B72D580C05DE2823A14B02B682636BF58F291A7E831D237ECE8FC14DA50A187A50ACF665442AB2D140E140B813CFCCA993BC02AA4A3D35C'/>
</query>
</iq>
The fingerprint will, for OTRv4 keys, always be 112 hexadecimal digits that can be decoded into a 56-byte value, following the instructions in the OTRv4 specification.
If the server returns more than one prekey server in its list of items, anyone should be able to use it. All prekey servers exposed by an XMPP server have to share the same storage. Thus, a client should randomly choose one of the returned prekey servers to connect to, in order to distribute load for the server.
An entity authenticates to the server through an interactive DAKE. DAKE messages are sent in "message" stanzas. This follows the prekey server specification for OTR [6].
When calculating the phi
value for XMPP, the bare JID of the publisher and the
bare jid of the server has to be used:
*Example 8. Calculating Phi*
phi = DATA("alice@xmpp.org") || DATA("prekey.xmpp.org")
An entity starts the DAKE by sending the first encoded message in the body of a message:
*Example 9. First DAKE message*
<message
from='alice@xmpp.org/notebook'
id='nzd143v8'
to='prekey.xmpp.org'>
<body>AAQ1...</body>
</message>
The server responds with the subsequent DAKE message:
*Example 10. Second DAKE message*
<message
from='prekey.xmpp.org'
id='13fd16378'
to='alice@xmpp.org/notebook'>
<body>AAQ2...</body>
</message>
And the entity terminates the DAKE and sends the prekey values attached to the last DAKE message:
*Example 11. Thrid DAKE message and submission*
<message
from='alice@xmpp.org/notebook'
id='kud87ghduy'
to='prekey.xmpp.org'>
<body>AAQ3...</body>
</message>
And the Prekey Server responds with a "Success" message:
*Example 12. Success message*
<message
from='prekey.xmpp.org'
id='0kdytsmslkd'
to='alice@xmpp.org/notebook'>
<body>AAQG...</body>
</message>
An entity authenticates to the server through a DAKE. DAKE messages are send in "message" stanzas.
It sends the same DAKE messages as the previous section, except for the attached message in the last DAKE-3 message.
And the entity terminates the DAKE and asks for storage information:
*Example 13. Thrid DAKE message and ask for storage information*
<message
from='alice@xmpp.org/notebook'
id='kud87ghduy'
to='prekey.xmpp.org'>
<body>AAQ3...</body>
</message>
And the Prekey Server responds with a "Storage Status" message:
*Example 14. Storage status message*
<message
from='prekey.xmpp.org'
id='0kdytsmslkd'
to='alice@xmpp.org/notebook'>
<body>AAQL...</body>
</message>
Clients that support the OTR protocol can advertise it in different ways, as OTR can be implemented in different modes that impact how the advertisement works.
There are two main mechanisms used:
- Plaintext advertisement
- DAKE advertisement.
If you are implementing 'OTRv3 Compatible Mode' and 'Interactive-Only Mode', or starting an online conversations, you can use these kind of initialization messages, which use OTR's own discovery mechanisms.
- Whitespace Tag
If a client wishes to indicate support for OTR they include a special whitespace tag in their messages. This tag can appear anywhere in the body of the message stanza, but it is most often found at the end. The OTR tag comprises the following bytes:
*Example 14. OTR tag*
\x20\x09\x20\x20\x09\x09\x09\x09 \x20\x09\x20\x09\x20\x09\x20\x20
and is followed by one or more of the following sequences to indicate the version of OTR which the client supports:
*Example 15. OTR tag version 3*
\x20\x20\x09\x09\x20\x20\x09\x09
*Example 16. OTR tag version 3*
\x20\x20\x09\x09\x20\x09\x20\x20
When a client sees this special string in the body of a message stanza it may choose to start an OTR session immediately, or merely indicate support to the user and allow the user to manually start a session. This is done by sending a message stanza containing an OTR query message in the body which indicates the supported versions of OTR.
- Query message
As stated above, this is a message stanza containing an OTR query message in the body which indicates the supported versions of OTR (currently only 3 and 4 are supported).
*Example 17. OTR query*
?OTR?v34?
Any message which begins with the afforementioned string (note that the version number[s] may be different) should be treated as an OTR message. The initialization message can contain a payload, which should not refer to the identity of any participant.
If you are implementing 'Interactive-Only Mode', 'Standalone Mode', or starting offline conversations, you can use these kind of initialization messages, which use OTR's XMPP own discovery mechanisms.
In order to determine whether a given contact has devices that support OTRv4, the devices node in PEP is consulted. Devices MUST subscribe to urn:xmpp:otr:1:devices via PEP, so that they are informed whenever their contacts add a new device. They MUST cache the most up-to-date version of the device list.
*Example 18. Devicelist update received by subscribed clients*
<message from='ahab@otr.im'
to='ishmael@otr.im'
type='headline'
id='update_01'>
<event xmlns='http://jabber.org/protocol/pubsub#event'>
<items node='urn:xmpp:otr:1:devices'>
<item id='current'>
<devices xmlns='urn:xmpp:omemo:1'>
<device id='7c91e4ce' />
<device id='c8257518' label='Gajim Client' />
</devices>
</item>
</items>
</event>
</message>
After the device list has been received, and the client have decided to which device to send an initialization message (which depends on the policy and instance tag behaviour defined. It can be to all of them, to the trusted one, to the most recent one, etc.), an identity message or a Non-Interactive Auth message can be sent (for online conversations and offline conversations respectively). Note that prior to sending any of these messages, Prekey values should be asked for.
Peers that want to start an OTR conversation, need to have subscription to the contact's presence information. The best subscription state for this might be 'Both'.
In order to start a OTR conversation, prekey values should be fetched. In order to start an online conversation, only a Client Profile should be asked for. In order to start an offline conversation, a Client Profile, a Prekey Profile and a Prekey Message should be asked for.
An entity asks the server for Prekey Ensembles from a particular participant by
sending a "Prekey Ensemble Query Retrieval message" for an specific identity
and device (using the instance tag), for example, queequeg@otr.im/7c91e4ce
,
and specific versions, for example, "45".
*Example 19. Asking for Prekey values*
<message
from='alice@xmpp.org/notebook'
id='nzd143v8'
to='prekey.xmpp.org'>
<body>AAQQ...</body>
</message>
The server responds with a "Prekey Ensemble Retrieval message" if there are values in storage:
*Example 20. Server response for success*
<message
from='prekey.xmpp.org'
id='13fd16378'
to='alice@xmpp.org/notebook'>
<body>AAQT...</body>
</message>
The server responds with a "No Prekey-Ensembles in Storage message" if there are no values in storage:
*Example 21. Server response for failure*
<message
from='prekey.xmpp.org'
id='13fd16378'
to='alice@xmpp.org/notebook'>
<body>AAQO...</body>
</message>
From this point, messages should be sent according to the protocol.
Some clients in the wild have been known to insert XML in the node of a message. Clients that support OTR should tolerate encrypted payloads which expand to unescaped XML, and treat it as plain text.
XMPP is designed so that the client needs to know very little about where and how a message will be routed. Generally, clients are encouraged to send messages to the bare JID and allow the server to route the messages as it sees fit. However, OTR requires that messages be sent to a particular resource. Therefore clients should send OTR messages to a full JID, possibly allowing the user to determine which resource (determined by the instance tag) they wish to start an encrypted session with. Furthermore, if a client receives a request to start an OTR session in a carboned message (due to a server which does not support the aforementioned "private" directive, or a client which does not set it), it should be silently ignored.
If a device has a policy for multi-device synchronization, before sending a message (except for a query message or a whitespace tag) a peer MUST explicitly fetch device lists for this peer and for the peer is wanting to start a conversation with.
Message Processing Hints (XEP-0334) [7] defines a set of hints for how messages should be handled by XMPP servers. These hints are not hard and fast rules, but suggestions which the servers may or may not choose to follow. Best practice is to include the following hints on all OTR messages:
<no-copy xmlns="urn:xmpp:hints"/>
<no-permanent-store xmlns="urn:xmpp:hints"/>
Explicit Message Encryption (XEP-0380) [8] defines a hint to let clients without OTR support know that this message was encrypted, and display a friendly message instead of the raw encrypted data. It is RECOMMENDED that the client adds this hint alongside every encrypted message
<encryption xmlns="urn:xmpp:eme:0" namespace="urn:xmpp:otr:0"/>
All together, an example OTR message might look like this (with the majority of the body stripped out for readability):
*Example 22. OTR message with processing hints*
<message from='malvolio@otr.im/countesshousehold'
to='olivia@otr.im/veiled'>
<body>?OTR?v23?...</body>
<encryption xmlns="urn:xmpp:eme:0" namespace="urn:xmpp:otr:0"/>
<no-copy xmlns="urn:xmpp:hints"/>
<no-permanent-store xmlns="urn:xmpp:hints"/>
<private xmlns="urn:xmpp:carbons:2"/>
</message>
Most clients today provide options to automatically start an OTR session, to manually construct a session at the users request, or to always require the use of an OTR session even if the remote client does not support OTR.
In the interest of user experience, it is NOT RECOMMENDED to start an OTR session with a previously unseen resource or one for which we do not have OTR keys cached without first discovering if the remote end supports OTR using one of the mechanisms described in the "Discovery" section of this document except in security critical contexts where user experience is not a concern.
Instead, it is RECOMMENDED to always allow the user to manually start an OTR session and to indicate that OTR is known to be available when OTR support is discovered by any of the aforementioned mechanisms.
It is RECOMMENDED that the lifetime of OTR sessions be limited to the lifetime of the XMPP session in which the OTR session was established. If a resource associated with either end of the OTR session goes offline (a closing stream tag is received, or a fatal stream error occurs), it is RECOMMENDED to start an OTR offline session.
RFC 5122 [9] defines a Uniform Resource Identifier (URI) and Internationalized Resource Identifier (IRI) scheme for XMPP entities, and XMPP URI Query Components (XEP-0147) [10] defines various query components for use with XMPP URI's. When an entity has an associated OTR fingerprint its URI is often formed with "otr-fingerprint" in the query string.
Example 23. OTR Fingerprint¶
xmpp:feste@allfools.lit?otr-fingerprint=AEA4D503298797D4A4FC823BC1D24524B4C54338
The XMPP Registrar [11] maintains a registry of queries and key-value pairs for use in XMPP URIs at https://xmpp.org/registrar/querytypes.html. As of the date this document was authored, the 'otr-fingerprint' query string has not been formally defined and has therefore is not officially recognized by the registrar.
This document requires no interaction with the Internet Assigned Numbers Authority (IANA).
No namespaces or parameters need to be registered with the XMPP Registrar as a result of this document.
- Borisov, N., Goldberg, I. and Brewer, E. (20004). Off-the-Record Communication, or, Why Not To Use PGP, WPES’04. Available at: https://otr.cypherpunks.ca/otr-wpes.pdf
- OTR version 4. Available at https://bugs.otr.im/otrv4/otrv4/-/blob/master/otrv4.md
- XEP-0045: Multi-User Chat. Available at: https://xmpp.org/extensions/xep-0045.html
- XEP-0369: Mediated Information eXchange (MIX). Available at: https://xmpp.org/extensions/xep-0369.html
- OTR version 4 modes. Available at https://bugs.otr.im/otrv4/otrv4/-/tree/master/modes
- OTR Prekey Server specification. Available at https://bugs.otr.im/otrv4/otrv4-prekey-server/-/blob/master/otrv4-prekey-server.md
- XEP-0334: Message Processing Hints. Available at: https://xmpp.org/extensions/xep-0334.html
- XEP-0380: Explicit Message Encryption. Available at: https://xmpp.org/extensions/xep-0380.html
- RFC 5122: Internationalized Resource Identifiers (IRIs) and Uniform Resource Identifiers (URIs) for the Extensible Messaging and Presence Protocol (XMPP). Avaiable at: http://tools.ietf.org/html/rfc5122
- XEP-0147: XMPP URI Query Components. Available at: https://xmpp.org/extensions/xep-0147.html
- The XMPP Registrar maintains a list of reserved protocol namespaces as well as registries of parameters used in the context of XMPP extension protocols approved by the XMPP Standards Foundation. For further information, see: https://xmpp.org/registrar/