-
Notifications
You must be signed in to change notification settings - Fork 1
API Protocol
GET /v1/accounts/{transport}/code/{number}
The client requests an SMS or Voice verification code for the client's PSTN number.
-
transport
is the stringsms
orvoice
, depending on how the client would like a verification code delivered. -
number
is the client's PSTN number.
Returns:
-
200
request was processed successfully. -
400
badly formattednumber
. -
415
invalidtransport
. -
413
rate limit exceeded. Too many requests.
PUT /v1/acccounts/code/{verification_code}
Authorization: Basic {basic_auth}
{
"signalingKey" : "{bas64_encoded_52_byte_key}"
"supportsSms" : false
}
The client submits the verification code it received via voice or SMS to the server for confirmation.
-
verification_code
is the code it received via voice or SMS, numeric only. -
basic_auth
are the authorization credentials the client would like to create. These are in the form ofBase64({number}:{password})
, wherenumber
is the client's verified PSTN number andpassword
is a randomly generated 16 byte string. -
signalingKey
is a randomly generated 32 byte AES key and a 20 byte HMAC-SHA1 MAC key, concatenated together and Base64 encoded. -
supportsSms
indicates whether a client supports SMS as a transport. -
registrationId
is a 14 bit integer that's randomly generated at client install time. This will be used for clients to detect whether an app has reinstalled and lost their session state.
Returns:
-
200
account successfully verified. -
401
badly formattedbasic_auth
. -
403
incorrectverification_code
. -
413
rate limit exceeded.
PUT /v1/accounts/apn/
Authorization: Basic {basic_auth}
{
apnRegistrationId: "{apn_registration_id}"
}
or
PUT /v1/accounts/gcm/
Authorization: Basic {basic_auth}
{
gcmRegistrationId: "{gcm_registration_id}"
}
The client submits its APN or GCM push registration ID.
-
basic_auth
is the client's authorization credentials (see above). -
gcm_reistration_id
orapn_registration_id
is the client's registration ID.
Returns:
-
200
request succeeded. -
401
invalid authentication credentials. -
415
badly formatted JSON.
PUT /v1/keys/
Authorization: Basic {basic_auth}
{
lastResortKey : {
keyId: 0xFFFFFF
publicKey: "{public_key}"
identityKey: "{identity_key}"
},
keys: [
{
keyId: {key_id},
publicKey: "{public_key}",
identityKey: "{public_key}"
},
...]
}
-
public_key
is a randomly generated Curve25519 public key with a leading byte of0x05
to indicate its type. This is a total of 33 bytes, base64 encoded without padding (no ==). -
identity_key
is a Curve25519 public key with a leading byte of0x05
to indicate its type. This is a total of 33 bytes, base64 encoded without padding (no ==). Each client should have a single identity key generated at install time. -
key_id
each prekey has a unique 24bit identifier. The last resort key is always 0xFFFFFF.
Returns:
-
200
request succeeded. -
401
invalid authentication credentials. -
415
badly formatted JSON.
PUT /v1/directory/tokens
Authorization: Basic {basic_auth}
{
"contacts": [{"{token}", "{token}", ..., "{token}"]
}
-
token
is Base64(SHA1(E164number)[0:10]) without Base64 padding.
Returns:
-
400
badly formatted token(s). -
401
invalid authentication credentials. -
415
badly formatted JSON. -
200
request succeeded. The structure below is returned.
{
contacts: [{token="{token}", relay="{relay}", supportsSms="true"},
{token="{token}"},
...,
{token="tokenN", relay="{relay}"}]
}
-
token
is Base64(SHA1(E164number)[0:10]) without Base64 padding. -
relay
is the name of a federated node which this contact is associated with. -
supportsSms
indicates that the contact supports the SMS transport.
At this point the client should be fully registered.
Messages bodies sent and received by clients are a protocol buffer structure:
message PushMessageContent {
optional string body = 1;
message AttachmentPointer {
optional fixed64 id = 1;
optional string contentType = 2;
optional bytes key = 3;
}
repeated AttachmentPointer attachments = 2;
}
If a client does not have an existing session with a recipient, the client will need to retrieve a PreKey for the recipient in order to start one.
GET /v1/keys/{number}?relay={relay}
Authorization: Basic {basic_auth}
-
number
is the PSTN number of the recipient. -
relay
(optional) is the federated relay the recipient is associated with. Therelay
param should only be included if the destination is at a federated node other than the sender.
Returns:
-
401
invalid authentication credentials. -
413
rate limit exceeded. -
404
unknown/unregisterednumber
. -
200
request succeeded. The structure below is returned.
{
keyId: {key_id},
publicKey: "{public_key}",
identityKey: "{public_key}"
}
POST /v1/messages/
Authorization Basic {basic_auth}
{
messages: [{
type: {type},
destination: "{destination_number}",
body: "{base64_encoded_message_body}", // Encrypted PushMessageContent
relay: "{relay}",
timestamp: "{time_sent_millis_since_epoc}"
},
...,
]
}
-
type
is the type of message. Supported types are enumerated below. -
destination_number
is the PSTN number of the message recipient. -
body
is the Base64 encoded (without padding) and encryptedPushMessageContent
(above). -
relay
(optional) is the relay the message recipient is registered with. -
timestamp_sent_millis_since_epoch
is the timestamp of the message in millis since the epoch.
Returns:
-
401
invalid authentication credentials. -
413
rate limit exceeded. -
415
badly formatted JSON. -
200
request succeeded. The structure below is returned.
{
"success" : [{destination_number}, {destination_number}, ..., {destination_number}],
"failure" : [{destination_number},...,{destination_number}]
}
Supported types:
int TYPE_MESSAGE_PLAINTEXT = 0;
int TYPE_MESSAGE_CIPHERTEXT = 1;
int TYPE_MESSAGE_PREKEY_BUNDLE = 3;
APN clients will receive a push notification:
{
alert: "You have a new message!",
"m": "{payload}"
}
GCM clients will receive a push notification:
{payload}
-
payload
is a Base64 encoded (without padding)IncomingPushMessageSignal
, which is encrypted and MAC'd using thesignalingKey
submitted during registration.
Encrypted IncomingPushMessageSignal format:
struct {
opaque version[1];
opaque iv[16];
opaque ciphertext[...]; // The IncomingPushMessageSignal
opaque mac[10];
The IncomingPushMessageSignal protocol buffer:
message OutgoingMessageSignal {
optional uint32 type = 1;
optional string source = 2;
optional string relay = 3;
repeated string destinations = 4;
optional uint64 timestamp = 5;
optional bytes message = 6; // Encrypted PushMessageContent (above)
}
Recall that a push message is transmitted as the following structure:
message PushMessageContent {
optional string body = 1;
message AttachmentPointer {
optional fixed64 id = 1;
optional string contentType = 2;
optional bytes key = 3;
}
repeated AttachmentPointer attachments = 2;
}
To fill out the AttachmentPointer
structure, the client takes the following steps:
- Generates a single-use 32 byte AES key and 32 byte Hmac-SHA256 key.
- Encrypts the attachment using AES in CBC mode with PKCS#5 padding and a random IV, then formats the encrypted blob as
IV || Ciphertext || MAC
. - Requests an attachment allocation from the server.
- Uploads the attachment to the allocation.
- Constructs the
AttachmentPointer
with the attachment allocationid
, the attachment's MIMEcontentType
, and the concatenated 32 byte AES and 32 byte Hmac-SHA256key
.
GET /v1/attachments/
Authorization: {basic_auth}
Returns:
-
401
invalid authentication credentials. -
413
rate limit exceeded. -
200
request succeeded. The structure below is returned.
{
"id" : "{attachment_id}",
"location" : "{attachment_url}"
}
PUT {attachment_url}
Content-Type: application/octet-stream
The client PUT
s the encrypted binary blob to the attachment_url
returned from the attachment allocation step.
GET /v1/attachments/{attachment_id}
Authorization: {basic_auth}
-
attachment_id
is theid
in a receivedAttachmentPointer
protocol buffer.
Returns
-
401
invalid authentication credentials. -
413
rate limit exceeded. -
200
request succeeded. The structure below is returned.
{
"location" : "{attachment_url}"
}
The client can now GET {attachment_url}
to retrieve the encrypted binary blob.