This package provides components for building an email client using the Telios Network. Telios is an offline-capabale e2e encrypted email service built on hypercore-protocol for sending and receiving emails.
This SDK provides methods for interacting with the Telios Client-Server API. It comes with everything needed for sending/receiving encrypted data, registering a new account, creating mailboxes, and creating shared drives.
npm i @telios/client-sdk
const ClientSDK = require("@telios/client-sdk");
const teliosSDK = await new ClientSDK({
provider: 'https://apiv1.telios.io'})
const Account = teliosSDK.Account
const { secretBoxKeypair, signingKeypair, mnemonic } = Account.makeKeys();
// Verification code sent to the recovery email
const vcode = "Xf1sP4";
const initPayload = {
account: {
account_key: secretBoxKeypair.publicKey,
recovery_email: recoveryEmail,
device_drive_key: drivePublicKey,
device_signing_key: signingKeypair.publicKey
},
};
const { account, sig } = await Account.init(
initPayload,
signingKeypair.privateKey
);
const registerPayload = {
...account,
sig: sig,
vcode: vcode,
};
// Send the account object that was just signed to be stored and
// verified on the server for later authentication.
const res = await account.register(registerPayload);
The ClientSDK class instantiates the Account and Mailbox Class as well as gives access to a host of utility functions under the Crypto module.
opts
: allows you to set class variablesauth
andprovider
provider
: Base URL of the API providerauth
claims
device_signing_key
:account_key
:device_peer_key
:device_id
:
device_signing_priv_key
:sig
: Signature sent from the Telios server when this account was registered.
The variables are accessible through the instance of ClientSDK
as shown in the example below.
This is a method allowing you to set the provider within the instantiated class.
provider
: Base URL of the API provider
This is a method allowing you to set the auth payload within the instantiated class.
auth
claims
device_signing_key
:account_key
:device_peer_key
:device_id
:
device_signing_priv_key
:sig
: Signature sent from the Telios server when this account was registered.
const ClientSDK = require("@telios/client-sdk");
const teliosSDK = await new ClientSDK()
// Account Class will help create request payload for new account creation and more
const Account = teliosSDK.Account
// Mailbox Class provides functionality needed to process encrypted emails
const Mailbox = teliosSDK.Mailbox
// Variables set through the constructor are available through client
const {
auth,
provider
opts
} = teliosSDK.client
teliosSDK.setProvider('https://apiv1.telios.io')
const auth = {
claims: {
account_key: //SB_PUB_KEY,
device_signing_key: //SIG_PUB_KEY,
device_id: //DEVICE_ID
},
device_signing_priv_key: //SIG_PRIV_KEY,
sig: //ACCOUNT_SERVER_SIG
}
teliosSDK.setAuthPayload(auth)
The Account class handles communication with the Telios server and provides methods for creating request payloads.
Keypairs will need to be initially created before any other actions can be taken. These keys will be used for encrypting/decrypting data on the client and from other users. The private keys should be stored somewhere safe (and encrypted) and never shared. The public keys generated will be used for encrypting a recipient's data and can be shared publicly.
mnemonic
: Passing in a bip39 mnemonic will create deterministic keypairs.
Returns:
secretBoxKeypair
: Public/private keys for the accountsigningKeypair
: Public/private signing keys for the accountmnemonic
: If no mnemonic is passed in as an argument a new one is generated and passed back with the keys that were created from it.
Prepares an account registration payload
acctPayload
: Account Object to be signed for registrationaccount
account_key
: Public key for the accountdevice_drive_key
: Public key of the drive created for the devicedrive.publicKey
recovery_email
: Recovery email in plaintext. This is immediately hashed and stored once sent to the backenddevice_signing_key
: Public signing key for your device
privateKey
: Private key for the account
Returns:
account
: Public/private keys for the accountaccount_key
: Public key for the accountdevice_drive_key
: Public key of the drive created for the devicedrive.publicKey
recovery_email
: Recovery email in plaintext. This is immediately hashed and stored once sent to the backenddevice_signing_key
: Public signing key for your devicedevice_id
: UUID for this device
sig
: Public/private signing keys for the account
Registers a new account with the API server. This method requires a verification code (vcode
) in order for the backend to create the account. Examples on how to generate verification codes are listed below.
acctPayload
: Account Objectaccount
account_key
: Public key for the accountdevice_drive_key
: Public key of the drive created for the devicedrive.publicKey
recovery_email
: Recovery email in plaintext. This is immediately hashed and stored once sent to the backenddevice_signing_key
: Public signing key for your devicedevice_id
: UUID for this device
sig
: Signature returned fromAccount.init
vcode
: Verification code sent to the recovery email.
Example: Get verfication code - This request will send a verification code in the form of a captcha image to the recovery email listed in the request.
curl --location --request POST 'https://apiv1.telios.io/account/captcha' --data-raw '{ "addr": "[email protected]" }'
Example: Verifying the activation code
curl --location --request POST 'https://apiv1.telios.io/account/captcha/verify' --data-raw '{ "vcode": "Xf1sP4" }'
Account registration example usage:
const ClientSDK = require("@telios/client-sdk");
const teliosSDK = await new ClientSDK({
provider: 'https://apiv1.telios.io'})
const Account = teliosSDK.Account
const { secretBoxKeypair, signingKeypair, peerKeypair } = Account.makeKeys();
// Verification code sent to the recovery email
const vcode = "Xf1sP4";
const initPayload = {
account: {
account_key: secretBoxKeypair.publicKey,
recovery_email: recoveryEmail,
device_drive_key: driveKey,
device_signing_key: signingKeypair.publicKey,
device_peer_key: peerKeypair.publicKey,
device_id: deviceId,
},
};
const { account, sig } = await Account.init(
signingKeypair.privateKey,
initPayload
);
const registerPayload = {
...account,
sig: sig,
vcode: vcode,
};
// Send the account object that was just signed to be stored and
// verified on the server for later authentication.
const { _sig: serverSig } = await Account.register(registerPayload);
// Storing the auth information for later use
const auth = {
claims: {
account_key: initPayload.account.account_key,
device_signing_key: initPayload.account.device_signing_key,
device_id: initPayload.account.device_id
},
device_signing_priv_key: signingKeypair.privateKey,
sig: serverSig
}
teliosSDK.setAuthPayload(auth)
Example response:
{
// signature from server to be used for authentication
_sig: "[server_signature]";
// The server's drive diff key. To replicate your local drive with the server,
// use this key when doing drive.addPeer(diffKey)
_drive_diff_key: "[drive_diff_key]"
}
The sig
returned will be required for authentication and should be stored and encrypted locally. This, along with the account's signing key will be used to create a unique access token for every request.
Initaites the account recovery flow. If email and recovery_email
are valid and recovery code will be sent to the recovery_email
.
email
: User's Telios email addressrecovery_email
: User's recovery email used during registration.
Generates a new sync code on the server that expires in 10 minutes. This code should be given to the other peer device for pairing.
Retrieves the public keys and info needed to sync one or more devices/drives
code
: The recovery code sent torecovery_email
Example response:
{
// The drive public key needed for replication
drive_key: "[drive_key]";
// The Telios seeding peer public key. Use this key to let Telios through the drive's firewall.
peer_pub_key: "[peer_pub_key]"
}
Registers a new device with the provider. Returns a server signature that should be used for generating new sessions with the API server.
payload
type
: The device type -"MOBILE" | "DESKTOP"
device_id
: UUID for this devicedevice_drive_key
: Public key of the drive created for the device -drive.publicKey
device_signing_key
: Public signing key for your device
This method will retrieve stats about the account.
Example Response:
{
plan: "FREE", // Plan tier the account is on
daily_email_used: 1, // Number of email sent since last reset aka whether or not they went over the daily quota
daily_email_reset_date: "2021-12-21T19:00:35.000+00:00", // Datea at which the daily_email_used will reset
namespace_used: 1, // Number of Namespace registered by Account
aliases_used: 3, // Number of Aliases registered by Account
storage_space_used: 109635126 // Total Space used up by Account on backup server
}
The Mailbox class provides functionality needed for processing encrypted emails.
Example Usage:
// If Auth payload hasn't been previously set do the below
const auth = {
claims: {
device_signing_key: signingKeypair.publicKey,
account_key: secretBoxKeypair.publicKey,
device_peer_key: peerKeypair.publicKey,
device_id: "[device_id]",
},
device_signing_priv_key: signingKeypair.privateKey,
sig: "[sig]",
}
teliosSDK.setAuthPayload(auth)
// Once Auth Payload is set.
const Mailbox = teliosSDK.Mailbox;
const payload = {
account_key: secretBoxKeypair.publicKey,
addr: "[email protected]",
};
const res = await Mailbox.registerMailbox(payload);
Example response:
{
"registered": true
}
A recipient's account's public key is required for sending encrypted emails within the Telios network. getMailboxPubKeys
takes an array of recipient's addresses and returns their corresponding public key.
addresses
: An array of email addresses
Example usage:
const res = await Mailbox.getMailboxPubKeys([
"[email protected]",
"[email protected]",
]);
Example response:
[
{
address: "[email protected]",
account_key: "[account_public_key]",
},
{
address: "[email protected]",
account_key: "[account_public_key]",
},
];
Test if an email recovery address is valid and or has not already been used by another account.
email
: A recovery email address[email protected]
Example usage:
const res = await Mailbox.isValidRecoveryEmail("[email protected]");
When sending an email to multiple recipients, the recipient's email domain is checked if it matches telios.io. In this case the email is encrypted, stored on the local drive, and an encrypted message is sent that only the recipient can decipher. The deciphered metadata gives the recipient instructions how to to retrieve and decipher thier encrypted email.
In the instance of multiple recipients from non-compatible email providers (gmail, yahoo, etc..), the email is initially sent without encryption via normal SMTP. The reason for this is it doesn't make sense to encrypt an email that's being sent in cleartext to other recipients. If some of the recipients are using telios.io, the email WILL be encrypted at rest when picked up by the mailserver for Telios recipients.
email
: An email in JSON formatowner
: An email address showing who the email is from. Recipients can use this with the detached signature to verify messages.keypairs
: signing and secret box keypairssecretBoxKeypair
: The sender's signing keypair (Bob). Private key is only used during encryption and never sent or stored.signingKeypair
: The sender's signing keypair object (Bob). Public key is used for authenticity of sender
Email JSON should be in the following format:
const email = {
"subject": "Hello Bob",
"date": "2020-07-14T13:49:36.000Z",
"cid": "bafkreif7unfcuq3zw6fl4pmxh67niukus4ozw3lvqmycmsd52wmor4n6kl", // IPFS cid
"key": "6f6331eaf1bdf150f37312dfb560792f4acd18ed4bcf989e4afab1d290cbe3f1",
"header": "27b9b910f9658d489be208cde88819de9512c8b9f3954baf",
"path": "677bfc57-3ee9-4d53-84dc-1010bff8f4d6.json",
"to": [
{
"address": "[email protected]",
"name": "Bob"
}
],
"from": [
{
"address": "[email protected]",
"name": "Alice"
}
],
"cc": [],
"bcc": [],
"sender": [],
"text_body": "You're my favorite test person ever",
"html_body": "<h1>You're my favorite test person ever</h1>",
"attachments": [
{
"filename": "test.pdf",
"content": "--base64-data--",
"contentType": "application/pdf"
},
{
"filename": "test.txt",
"content": "--base64-data--",
"contentType": "text/plain"
}
]
}
Example usage:
// In this example Bob is sending an ecrypted email to two other Telios mailboxes.
const res = await Mailbox.send(email, {
owner: "[email protected]",
keypairs: {
secretBoxKeypair,
signingKeypair
},
// A Shared Drive.
drive: "[drive]",
// Destination path of the file on the local drive
dest: "/email/email.json",
});
acctPrivKey
: Your account's private keyacctPubKey
: Your account's public key
Example usage:
const acctPubKey = "[account_public_key]";
const acctPrivKey = "[account_private_key]";
const mail = await Mailbox.getNewMail(acctPrivKey, acctPubKey);
Example response:
[
{
headers: [
{
header: "x-spam-score",
value: "1.9",
},
],
subject: "Hello Bob",
date: "2020-07-14T13:49:36.000Z",
to: [
{
address: "[email protected]",
name: "Bob",
},
],
from: [
{
address: "[email protected]",
name: "Alice",
},
],
cc: [],
bcc: [],
sender: [],
text_body: "You're my favorite test person ever",
html_body: "<h1>You're my favorite test person ever</h1>",
attachments: [
{
filename: "test.pdf",
content: "--base64-data--",
contentType: "application/pdf",
},
{
filename: "test.txt",
content: "--base64-data--",
contentType: "text/plain",
},
],
},
];
After an email has been pulled down onto your local devices its meta record can be safely removed from the server.
ids
: an array of meta message ids on the server
Example usage:
// Pass in an array of message IDs to be marked as synced.
const res = await Mailbox.markAsSynced(["5f1210b7a29fe6222f199f80"]);
Example Alias: alice2000
#netflix
@telios.io
alice2000
= Alias Namespace
Registers a new alias namespace. Namespaces must be unique and are account bound. Each namespace can potentially hold an infinite number of alias addresses under each namespace.
alias_name
: Desired namespacealice2000
domain
: Domain for the namespace in the following formattelios.io
key
: Alias namespace's public key. Each namespace must have it's own unique public key
Example usage:
const mailbox = await initMailbox();
const { secretBoxKeypair } = Account.makeKeys();
const res = await mailbox.registerAliasName({
alias_name: 'alice2000',
domain: 'telios.io',
key: secretBoxKeypair.publicKey
});
Example response:
{
"registered": true,
"name": "aliceAlias",
"key": "[alias_public_key]"
}
Example Alias: alice2000
#netflix
@telios.io
netflix
= Alias address
Registers a new alias address. Addresses can be registered manually, or they can be created on-the-fly. If an account already has an alias name registered (alice2000
) and an email is sent to alice2000#[email protected]
, then the alias mailbox alice2000#[email protected]
will automatically be created. Auto-generated aliases do not count against alias usage until they are manually whitelisted
. Setting whitelisted
to false will silently fail to deliver any new emails coming to that address.
alias_addressed
: Desired addressnetflix
forwards_to
: Array of email address strings that this alias address should forward to. All alias addresses default to forwarding to the account's main mailbox address and any addresses added here will be addative to that main mailbox address.whitelisted
: true|false Allow emails to be delivered to this alias.
Example usage:
const res = await Mailbox.registerAliasAddress({
alias_address: 'alice2000#[email protected]',
forwards_to: [],
whitelisted: true
});
Example response:
{
"_xid": "155", // external ID of the alias address record on the mailserver.
"alias_key": "[alias_public_key]",
"registered": true,
"alias_address": "alice2000#[email protected]",
"forwards_to": [],
"whitelisted": true
}
Update an existing alias address.
alias_address
: the full alias address being updatedalice2000#[email protected]
,forwards_to
: Array of email address strings that this alias address should forward to. All alias addresses default to forwarding to the account's main mailbox address and any addresses added here will be addative to that main mailbox address.whitelisted
: true|false Allow emails to be delivered to this alias.
Example usage:
const res = await Mailbox.updateAliasAddress({
alias_address: 'alice2000#[email protected]',
forwards_to: ['[email protected]'],
whitelisted: true
});
Example response:
{
"address": "alice2000#[email protected]",
"forwards_to": ['[email protected]'],
"whitelisted": true
}
Removes an alias address.
address
: String of the full alias addressalice2000#[email protected]
Example usage:
const res = await Mailbox.removeAliasAddress('alice2000#[email protected]');
The Domain class provides functionality needed for registering custom domains and mailboxes.
Example Usage:
// If Auth payload hasn't been previously set do the below
const auth = {
claims: {
device_signing_key: signingKeypair.publicKey,
account_key: secretBoxKeypair.publicKey,
device_peer_key: peerKeypair.publicKey,
device_id: "[device_id]",
},
device_signing_priv_key: signingKeypair.privateKey,
sig: "[sig]",
}
teliosSDK.setAuthPayload(auth)
// Once Auth Payload is set.
const Domain = teliosSDK.Domain;
const payload = {
account_key: secretBoxKeypair.publicKey,
addr: "[email protected]",
};
const bool = await Domain.isAvailable('telios.app');
Check if custom a domain is available and has not been previously registered.
domain
: String of the domaintelios.app
Example usage:
const bool = await Domain.isAvailable('telios.app');
Register a new custom domain.
domain
: String of the domaintelios.app
Example usage:
const payload = {
domain:'telios.app'
};
const res = await Domain.register('telios.app');
Example response:
{
"domain": "telios.app",
"verification": {
"name": "@",
"type": "TXT",
"value": "telios-verification=d48808347d7d8a0b91f2e3af9d77ce33"
}
}
Verify user owns the domain.
domain
: String of the domaintelios.app
Example usage:
const { verified } = await Domain.verifyOwnership('telios.app');
Verify DNS records are correctly set.
domain
: String of the domaintelios.app
Example usage:
const payload = {
domain:'telios.app'
};
const res = await Domain.verifyDNS('telios.app');
Example response:
{
"mx": {
"type": "MX",
"name": "telios.app",
"value": "mailer.telios.app",
"verified": true
},
"spf": {
"type": "TXT",
"name": "telios.app",
"value": "v=spf1 include:mailer.telios.app ~all",
"verified": true
},
"dkim": {
"type": "TXT",
"name": "dkim._domainkey.telios.app",
"value": "",
"verified": true
},
"dmarc": {
"type": "TXT",
"name": "_dmarc.telios.app",
"value": "v=DMARC1;p=quarantine",
"verified": true
}
}
Register a mailbox under a custom domain.
name
: String of the mailbox display nameBob Test
addr
: String of the full mailbox address with custom domain[email protected]
account_key
: String of the mailbox account's public key
Example usage:
const payload = {
name: 'Bob Tester',
addr:'[email protected]',
account_key: '0000000000000000000000000000000000000000000000000000000000000000'
};
await Domain.registerMailbox(payload);
Update a custom domain mailbox.
addr
: String of the mailbox addressbob
disabled
: Boolean that determines if the mailbox is disabled and should receive mail
Example usage:
const payload = {
addr:'[email protected]',
disabled: true
};
const res = await Domain.update(payload);
Example response:
{
"address": "[email protected]",
"disabled": true
}
Delete a mailbox under a custom domain.
addr
: String of the mailbox address[email protected]
Example usage:
const payload = {
addr:'[email protected]'
};
await Domain.deleteMailbox(payload);
Send an email invite to another user to claim their mailbox. The invite email will send them a code and a temporary password to claim their account.
addr
: String of the claimable mailbox address[email protected]
inviteEmail
: String of the email address to send the invite to. Note that this also becomes the recovery email for this business userpassword
: String of the temporary password the new user will need to claim the account. Once their account has been claimed/sync they will immediately be asked to replace this temporary pass with their own.
Example usage:
const payload = {
addr:'[email protected]',
inviteEmail: '[email protected]',
password: 'letmein1234'
};
await Domain.sendMailboxInvite(payload);