keyserver/README.md

254 lines
8.9 KiB
Markdown
Raw Normal View History

2016-05-25 14:13:49 +00:00
Mailvelope Keyserver [![Build Status](https://travis-ci.org/mailvelope/keyserver.svg?branch=master)](https://travis-ci.org/mailvelope/keyserver)
==============
2016-06-01 18:44:14 +00:00
A simple OpenPGP public key server that validates email address ownership of uploaded keys.
## Why not use Web of Trust?
There are already OpenPGP key servers like the [SKS keyserver](https://bitbucket.org/skskeyserver/sks-keyserver/wiki/Home) that employ the [Web of Trust](https://en.wikipedia.org/wiki/Web_of_trust) to provide a way to authenticate a user's PGP keys. The problem with these servers are discussed [here](https://en.wikipedia.org/wiki/Key_server_(cryptographic)#Problems_with_keyservers).
### Privacy
The web of trust raises some valid privacy concerns. Not only is a user's social network made public, common SKS servers are also not compliant with the [EU Data Protection Directive](https://en.wikipedia.org/wiki/Data_Protection_Directive) due to lack of key deletion. This key server addresses these issues by not employing the web of trust and by allowing key removal.
### Usability
The main issue with the Web of Trust though is that it does not scale in terms of usability. The goal of this key server is to enable a better user experience for OpenPGP user agents by providing a more reliable source of public keys. Similar to messengers like Signal, users verify their email address by clicking on a link of a PGP encrypted message. This prevents user A from uploading a public key for user B. With this property in place, automatic key lookup is more reliable than with standard SKS servers.
2016-06-01 18:44:14 +00:00
This requires more trust to be placed in the service provider that hosts a key server, but we believe that this trade-off is necessary to improve the user experience for average users. Tech-savvy users or users with a threat model that requires stronger security may still choose to verify PGP key fingerprints just as before.
## Standardization and (De)centralization
The idea is that an identity provider such as an email provider can host their own key directory under a common `openpgpkeys` subdomain. An OpenPGP supporting user agent should attempt to lookup keys under the user's domain e.g. `https://openpgpkeys.example.com` for `user@example.com` first. User agents can host their own fallback key server as well, in case a mail provider does not provide its own key directory.
2016-06-01 18:44:14 +00:00
2016-06-09 16:08:15 +00:00
# Demo
Try out the server here: [https://keys.mailvelope.com](https://keys.mailvelope.com)
2016-06-01 18:44:14 +00:00
# Api
2016-06-07 13:36:38 +00:00
The key server provides a modern RESTful api, but is also backwards compatible to the OpenPGP HTTP Keyserver Protocol (HKP).
2016-06-01 18:44:14 +00:00
## HKP api
2016-06-07 13:36:38 +00:00
The HKP apis are not documented here. Please refer to the [HKP specification](https://tools.ietf.org/html/draft-shaw-openpgp-hkp-00) to learn more. The server generally implements the full specification, but has some constraints to improve the security for automatic key lookup:
2016-06-01 18:44:14 +00:00
2016-06-07 13:36:38 +00:00
#### Accepted `search` parameters
* Email addresses
* V4 Fingerprints
* Key ID strings with at least 16 digits (64-bit long key ID)
2016-06-01 18:44:14 +00:00
2016-06-07 13:36:38 +00:00
#### Accepted `op` parameters
* get
* index
* vindex
2016-06-01 18:44:14 +00:00
2016-06-07 13:36:38 +00:00
#### Accepted `options` parameters
* mr
2016-06-01 18:44:14 +00:00
## REST api
2016-06-08 21:01:32 +00:00
### Lookup a key
2016-06-09 10:21:16 +00:00
#### By key ID
2016-06-01 18:44:14 +00:00
```
2016-06-09 10:21:16 +00:00
GET /api/v1/key?keyId=b8e4105cc9dedc77
```
#### By fingerprint
```
GET /api/v1/key?fingerprint=e3317db04d3958fd5f662c37b8e4105cc9dedc77
2016-06-01 18:44:14 +00:00
```
2016-06-08 21:01:32 +00:00
#### By email address
2016-06-01 18:44:14 +00:00
2016-06-08 21:01:32 +00:00
```
GET /api/v1/key?email=user@example.com
```
2016-06-01 18:44:14 +00:00
2016-06-08 21:01:32 +00:00
#### By email address (shorthand link for sharing)
```
GET /user/user@example.com
```
#### Payload (JSON):
2016-06-01 18:44:14 +00:00
```json
{
2016-06-08 21:01:32 +00:00
"keyId": "b8e4105cc9dedc77",
"fingerprint": "e3317db04d3958fd5f662c37b8e4105cc9dedc77",
"userIds": [
{
"name": "Jon Smith",
2016-06-09 16:08:15 +00:00
"email": "jon@smith.com",
"verified": "true"
2016-06-08 21:01:32 +00:00
}
],
"created": "Sat Oct 17 2015 12:17:03 GMT+0200 (CEST)",
"algorithm": "rsa_encrypt_sign",
"keySize": "4096",
"publicKeyArmored": "-----BEGIN PGP PUBLIC KEY BLOCK----- ... -----END PGP PUBLIC KEY BLOCK-----"
2016-06-01 18:44:14 +00:00
}
```
2016-06-08 21:01:32 +00:00
* **keyId**: The 16 char key id in hex
* **fingerprint**: The 40 char key fingerprint in hex
2016-06-09 16:08:15 +00:00
* **userIds.name**: The user ID's name
* **userIds.email**: The user ID's email address
* **userIds.verified**: If the user ID's email address has been verified
2016-06-08 21:01:32 +00:00
* **created**: The key creation time as a JavaScript Date
* **algorithm**: The primary key alogrithm
* **keySize**: The key length in bits
* **publicKeyArmored**: The ascii armored public key block
### Upload new key
2016-06-01 18:44:14 +00:00
```
2016-06-08 21:01:32 +00:00
POST /api/v1/key
2016-06-01 18:44:14 +00:00
```
2016-06-08 21:01:32 +00:00
#### Payload (JSON):
2016-06-01 18:44:14 +00:00
2016-06-08 21:01:32 +00:00
```json
{
"publicKeyArmored": "-----BEGIN PGP PUBLIC KEY BLOCK----- ... -----END PGP PUBLIC KEY BLOCK-----",
"primaryEmail": "user@example.com"
}
2016-06-01 18:44:14 +00:00
```
2016-06-08 21:01:32 +00:00
* **publicKeyArmored**: The ascii armored public PGP key to be uploaded
* **primaryEmail (optional)**: The ascii armored block is parsed to check for user ids, so this parameter is purely optional. Normally a verification email is sent to every user id found in the pgp key. To prevent this behaviour, user agents can specify the user's primary email address to send out only one email.
2016-06-01 18:44:14 +00:00
2016-06-08 21:01:32 +00:00
### Verify uploaded key
2016-06-01 18:44:14 +00:00
```
GET /api/v1/verify?keyId=b8e4105cc9dedc77&nonce=6a314915c09368224b11df0feedbc53c
2016-06-01 18:44:14 +00:00
```
### Request key removal
2016-06-09 16:08:15 +00:00
#### Via delete request
2016-06-01 18:44:14 +00:00
```
2016-06-09 16:08:15 +00:00
DELETE /api/v1/key?keyId=b8e4105cc9dedc77 OR ?email=user@example.com
2016-06-01 18:44:14 +00:00
```
2016-06-09 16:08:15 +00:00
#### Via link
2016-06-01 18:44:14 +00:00
```
2016-06-09 16:08:15 +00:00
GET /api/v1/removeKey?keyId=b8e4105cc9dedc77 OR ?email=user@example.com
```
2016-06-01 18:44:14 +00:00
### Verify key removal
```
GET /api/v1/verifyRemove?keyId=b8e4105cc9dedc77&nonce=6a314915c09368224b11df0feedbc53c
2016-06-01 18:44:14 +00:00
```
# Development
2016-06-01 18:44:14 +00:00
The server is written is in JavaScript ES6 and runs on [Node.js](https://nodejs.org/) v4+. It uses [MongoDB](https://www.mongodb.com/) v2.4+ as its database.
## Install Node.js (Mac OS)
This is how to install node on Mac OS using [homebrew](http://brew.sh/). For other operating systems, please refer to the [Node.js download page](https://nodejs.org/en/download/).
```shell
brew update
brew install node
```
## Setup local MongoDB (Mac OS)
This is the installation guide to get a local development installation on Mac OS using [homebrew](http://brew.sh/). For other operating systems, please refer to the [MongoDB Getting Started Guide](https://docs.mongodb.com/getting-started/shell/).
```shell
brew update
brew install mongodb
mongod --config /usr/local/etc/mongod.conf
```
Now the mongo daemon should be running in the background. To have mongo start automatically as a background service on startup you can also do:
```shell
brew services start mongodb
```
Now you can use the `mongo` CLI client to create a new test database. **The username and password used here match the ones in the `config/development.js` file. Be sure to change them for production use**:
2016-06-01 18:44:14 +00:00
```shell
mongo
use keyserver-test
db.createUser({ user:"keyserver-user", pwd:"trfepCpjhVrqgpXFWsEF", roles:[{ role:"readWrite", db:"keyserver-test" }] })
```
## Setup SMTP user
The key server uses [nodemailer](https://nodemailer.com) to send out emails upon public key upload to verify email address ownership. To test this feature locally, open the `config/development.js` file and change the `email.auth.user` and `email.auth.pass` attributes to your Gmail test account. Make sure that `email.auth.user` and `email.sender.email` match. Otherwise the Gmail SMTP server will block any emails you try to send. Also, make sure to enable `Allow less secure apps` in the [Gmail security settings](https://myaccount.google.com/security#connectedapps). You can read more on this in the [Nodemailer documentation](https://nodemailer.com/using-gmail/).
2016-06-01 18:44:14 +00:00
For production you should use a service like [Amazon SES](https://aws.amazon.com/ses/), [Mailgun](https://www.mailgun.com/) or [Sendgrid](https://sendgrid.com/solutions/transactional-email/). Nodemailer supports all of these out of the box.
## Install dependencies and run tests
```shell
npm install && npm test
```
## Start local server
```shell
npm start
```
# Production
The `config/development.js` file can be used to configure a local development installation. For production use, the following environment variables need to be set:
2016-06-01 18:44:14 +00:00
* NODE_ENV=production
2016-06-10 15:48:41 +00:00
* UPGRADE_HTTPS=true (upgrade HTTP to HTTPS and use [HSTS](https://developer.mozilla.org/en-US/docs/Web/Security/HTTP_strict_transport_security))
* PUBLIC_KEY_PIN=<base64 encoded sha256> (use [HPKP](https://developer.mozilla.org/de/docs/Web/Security/Public_Key_Pinning))
2016-06-01 18:44:14 +00:00
* MONGO_URI=127.0.0.1:27017/test_db
* MONGO_USER=db_user
* MONGO_PASS=db_password
* SMTP_HOST=127.0.0.1
* SMTP_PORT=465
* SMTP_TLS=true
2016-06-02 11:00:22 +00:00
* SMTP_STARTTLS=true
* SMTP_PGP=true
2016-06-01 18:44:14 +00:00
* SMTP_USER=smtp_user
* SMTP_PASS=smtp_pass
* SENDER_NAME="OpenPGP Key Server"
2016-06-01 18:44:14 +00:00
* SENDER_EMAIL=noreply@example.com
# License
AGPL v3.0
See the [LICENSE](https://raw.githubusercontent.com/mailvelope/keyserver/master/LICENSE) file for details
## Libraries
Among others, this project relies on the following open source libraries:
* [OpenPGP.js](https://openpgpjs.org/)
* [Nodemailer](https://nodemailer.com/)
* [addressparser](https://github.com/nodemailer/addressparser)
* [koa](http://koajs.com/)
* [mongodb](https://mongodb.github.io/node-mongodb-native/)