I started using Bitwarden, an open source password manager, and have been quite happy with it as I'm self-hosting and it has native apps for my devices in addition to a browser UI. I'm leaving the code here for posterity, but it's definitely not production ready or supported.
- Centralized storage of passwords and other sensitive data
- Accessible by clients that possess keys and perform all crypto operations client-side
- Distrust the Server: assume the server will be compromised at some point
- Less is More: client and server code should be as simple as possible (complexity == bugs)
- Website passwords
- Credit cards
- Sensitive data (SSN, etc.)
- Master Password (MP): known to user only, used to generate Data Encryption Key
- Data Encryption Key (DEK): 256 bit key generated using MP and PBKDF2, used to encrypt all sensitive data
- Request Authentication Key (RAK): 80 bit shared secret between client and server, generated when new client established, used to HMAC requests for data from the server. Note: 80 bits is the approximate length of 16 monocased alphanumeric characters. This is assumed to be a reasonable amount of data to manually transfer to a new client (4 rows of 4 characters).
- Install application files on web server.
- In configuration files configure database host, credentials, table prefix.
- Optionally configure admin email contact to validate new accounts. Leaving blank leaves server open to public for signups.
- At this point an anonymous request to the project root displays the basic info needed to configure new clients.
- Install client application
- Configure base URL for server
- Client confirms valid server certificate. Invalid certificate results in a hard failure.
- Client presents option for new or existing account.
- If new account:
- Client generates new RAK, stores it locally
- Prompts user for desired username
- Sends request for new account to server
- Server validates request:
- username available
- RAK strong enough
- client below request threshold
- If request is valid:
- server creates account, stores username and RAK
- responds with a success message
- Otherwise, respond with an error message
- If existing account:
- Client prompts user for username and RAK from existing client (provides information on where to find RAK)
- Client confirms correct username and RAK by making an authenticated request to the server and checking for a success response code
- User enters data to be stored into UI and indicates “Save Data”
- Client checks if client session is active, i.e. DEK is in memory
- If no active client session:
- Prompt user for master password and generate DEK
- Otherwise retrieve DEK from client session
- Encrypt data with DEK and create payload to send to server for storage
- Send payload to server and authenticate request with HMAC using RAK
- User interacts with UI and indicates “Retrieve Data”
- Based on UI context, client determines which records to request from server
- Creates payload to send to server to request records
- Send payload to server and authenticate request with HMAC using RAK
- Client checks if client session is active, i.e. DEK is in memory
- If no active client session:
- Prompt user for master password and generate DEK
- Otherwise retrieve DEK from client session
- Decrypt records with DEK and update the UI with the
The following schema defines the data types stored by the server:
- TBD
The server implements a basic REST API. It supports multiple API versions by exposing a version number as the second chunk of the path.
/api/1/data
- POST: create a new stored item
/api/1/data/{id}
- GET: retrieve the stored item with given id
- PUT: update the stored item with given id
- DELETE: delete the stored item with given id
/api/1/data/search
- POST: retrieve the set of stored items that match the query
- params: query (string)
- POST: retrieve the set of stored items that match the query
/api/1/auth
- PUT: update the current user’s RAK
The client is responsible for all cryptographic operations other than those performed by the server during request validation. The following API is optional for clients to implement, but will help organize client application code around the standard workflows.
TODO move the response codes to the Server API, they don’t belong here
create_item
: create a new secure item and store it on the server- args:
- type (int): type of record, i.e. website, credit card, data
- public_name (string): plaintext identifier for the record
- data (string): plaintext JSON data to store
- returns:
- 200: item ID (int) if success
- 400: error code if failure
- args:
get_item
: return a secure item from the server- args:
- id (int): item ID for the record
- returns:
- 200: item (JSON)
- 400: error code if failure
- args:
edit_item
: modify the attributes of a secure item on the server- args:
- id (int): item ID for the record
- updates (JSON): name-value pairs of fields to modify
- returns:
- 200: item ID (int) if success
- 400: error code if failure
- args:
delete_item
: delete a secure item from the server- args:
- id (int): item ID of the record to delete
- returns:
- item ID if success
- error code if failure
- args:
get_items
: return a list of secure items from the server that match a query string- args:
- query (string): Optional. Query string to match secure items. Empty query returns all items.
- returns:
- list of secure items
- args:
decrypt_data
: decrypt data using DEK and return data in JSON format- args:
- data (string): encrypted JSON data
- returns:
- plaintext JSON data (string)
- args:
encrypt_data
: encrypt data using DEK and return encrypted bytes- args:
- data (string): plaintext JSON data
- returns:
- encrypted JSON data (string)
- args:
get_master_password
: prompt user for MP and return the string entered- args:
- None
- returns:
- characters entered (string)
- args:
get_dek
: generate and return DEK based on MP- args:
- None
- returns:
- DEK (string)
- args:
- TBD