The Cerberus Management Service (CMS) is a core component of the Cerberus REST API that facilities user and AWS IAM role authentication and the management of Safe Deposit Boxes (SDBs), an abstraction on top of Hashicorp's Vault.
To learn more about Cerberus, please visit the Cerberus website.
If you do not wish to install and run MySQL and Vault locally you can run these embedded using gradlew see the Running CMS Locally
section
gradlew runVaultAndMySQL
If you wish to persist data permanently you can install Vault and MySQL locally
MySQL is required to run the application locally.
To get MySQL setup on OS X:
$ brew install mysql
$ mysql.server restart
$ mysql_secure_installation
You'll need to create a database and user for it. Run the following SQL against your mysql database:
CREATE DATABASE IF NOT EXISTS cms;
CREATE USER 'cms'@'localhost' IDENTIFIED BY '<YOUR DB PASSWORD HERE>';
GRANT ALL ON cms.* TO 'cms'@'localhost';
Vault is required to run the application locally.
To get Vault setup on OS X:
$ brew install vault
$ vault server -dev
This will output a root token, you can now setup vault using the supplied script:
$ /path/to/project/vault-setup.sh
That will setup the default policy and generate a token for CMS and output:
export VAULT_ADDR="http://localhost:8200"
export VAULT_TOKEN="<token>"
There are a few parameters that need to be configured for CMS to run properly, they are defined in this table.
property | required | notes |
---|---|---|
JDBC.url | Yes | The JDBC url for the mysql db |
JDBC.username | Yes | The JDBC user name for the mysql db |
JDBC.password | Yes | The JDBC JDBC.password for the mysql db |
root.user.arn | Yes | The arn for the root AWS user, needed to make the KMS keys deletable. |
admin.role.arn | Yes | The arn for an AWS user, needed to make the KMS keys deletable. |
cms.role.arn | Yes | The arn for the Instance profile for CMS instances, so they can admin KMS keys that they create. |
cms.admin.group | Yes | Group that user can be identified by to get admin privileges, currently this just enables users to access /v1/metadata see API.md |
cms.admin.roles | No | Comma separated list of ARNs that can auth and access admin endpoints. |
cms.auth.connector | Yes | The user authentication connector implementation to use for user auth. |
cms.user.token.ttl.override | No | By default user tokens are created with a TTL of 1h, you can override that with this param |
cms.iam.token.ttl.override | No | By default IAM tokens are created with a TTL of 1h, you can override that with this param |
cms.kms.policy.validation.interval.millis.override | No | By default CMS validates KMS key policies no more than once per minute, you can override that with this param |
KMS Policies are bound to IAM Principal IDs rather than ARNs themselves. Because of this, we validate the policy at authentication time
to ensure that if an IAM role has been deleted and re-created, that we grant access to the new principal ID.
The API limit for this call is low, so the cms.kms.policy.validation.interval.millis.override
property is used to throttle this validation.
For local dev see Running CMS Locally
.
For deployed environments they are configured via the CLI, which will generate a props file and stuff it into S3 encrypted with KMS.
cerberus --debug \
-e demo \
-r us-west-2 \
create-cms-config \
--admin-group cerberus-admins \
-P cms.auth.connector=com.nike.cerberus.auth.connector.onelogin.OneLoginAuthConnector \
-P auth.connector.onelogin.api_region=us \
-P auth.connector.onelogin.client_id=$ONE_LOGIN_CLIENT_ID \
-P auth.connector.onelogin.client_secret=$ONE_LOGIN_CLIENT_SECRET \
-P auth.connector.onelogin.subdomain=nike
See Creating an environment for more information.
CMS will download the props file at startup time and load the props into Guice.
The User authentication contract is defined by the AuthConnector interface.
The only included implementation of this interface targets OneLogin. We expect to implement more connectors in the near future.
property | required | notes |
---|---|---|
cms.auth.connector | Yes | com.nike.cerberus.auth.connector.onelogin.OneLoginAuthConnector |
auth.connector.onelogin.api_region | Yes | us or eu |
auth.connector.onelogin.client_id | Yes | The OneLogin API client id |
auth.connector.onelogin.client_secret | Yes | The OneLogin API client secret |
auth.connector.onelogin.subdomain | Yes | Your orgs OneLogin subdomain [xxxxx].onelogin.com |
Assumption: The current implementation looks up group membership for a user via the member_of field on the getUserById API response.
Multi-factor authentication is only enabled if it is required in Okta for the authenticating user.
property | required | notes |
---|---|---|
cms.auth.connector | Yes | com.nike.cerberus.auth.connector.okta.OktaAuthConnector |
auth.connector.okta.api_key | Yes | The Okta API key |
auth.connector.okta.base_url | Yes | The Okta base url (e.g. "https://example.okta.com" or "https://example.oktapreview.com" ) |
Multi-factor authentication is enabled for all users, even if it is not required in Okta.
property | required | notes |
---|---|---|
cms.auth.connector | Yes | com.nike.cerberus.auth.connector.okta.OktaMFAAuthConnector |
auth.connector.okta.api_key | Yes | The Okta API key |
auth.connector.okta.base_url | Yes | The Okta base url (e.g. "https://example.okta.com" or "https://example.oktapreview.com" ) |
First, a few properties must be configured in src/main/resources/cms-local-overrides.conf
You'll need a few pieces of information before you can run the application:
- The DB password you setup earlier
- The group that identifies which users are administrators
- The root user ARN for your AWS account
- The AWS IAM role ARN that represents administrators and CMS instances
- The authentication connector class that is used to authenticate users and get their group membership
# Database connection details.
JDBC.url="jdbc:mysql://localhost:3306/cms?useUnicode=true&characterEncoding=utf8&useLegacyDatetimeCode=false&serverTimezone=UTC&useSSL=false"
JDBC.username="cms"
JDBC.password="<YOUR DB PASSWORD HERE>"
# Group that user can be identified by to get admin privileges.
cms.admin.group="<YOUR ADMIN GROUP>"
# AWS ARNs used when setting up KMS keys for IAM role authentication.
root.user.arn="arn:aws:iam::<YOUR AWS ACCOUNT ID>:root"
admin.role.arn="arn:aws:iam::<YOUR AWS ACCOUNT ID>:role/<YOUR IAM ROLE FOR ADMINS>"
cms.role.arn="arn:aws:iam::<YOUR AWS ACCOUNT ID>:role/<YOUR IAM ROLE FOR CMS>"
# Auth Connector
cms.auth.connector=<YOUR AUTH CONNECTOR CLASS>
This option is for running locally uses embedded Vault, MySQL, and runs the Dashboard with reverse proxy.
Run each of the following tasks in new command line terminals (each are blocking tasks).
Steps:
gradlew runVaultAndMySQL
- Downloads and configures embedded MySQL and Vault,
- You can control Vault version with
vaultVersion
ingradle/develop.gradle
- This task needs to be run as Admin in Windows, ensure that you start the IDE or Terminals as Admin
- Once you see
core: post-unseal setup complete
proceed to next step
gradlew runCMS
- Auto-sets the Vault system props from
runVaultAndMySQL
and starts CMS - To debug attach remote debugger to port 5005
- If you wish to do IAM auth in dev mode you will need to make sure you set your env as described http://docs.aws.amazon.com/sdk-for-java/v1/developer-guide/credentials.html
- Now you should have a complete CMS system running locally. The next steps are optional.
- Auto-sets the Vault system props from
gradlew runDashboardAndReverseProxy
(optional)- Runs the dashboard and reverse into interact with CMS, sometimes better than curling or using postman.
- Downloads the dashboard from GitHub releases and runs an express server and reverse proxy to expose
http://localhost:9001/dashboard/
- You can change dashboard version with
dashboardRelease
ingradle/develop.gradle
gradlew bootstrapData
(optional)- Adds some data test data to Cerberus since
runVaultAndMySQL
is ephemeral and deletes everything when the process ends.
- Adds some data test data to Cerberus since
Above should work on Windows, Mac, and Linux.
With Vault and MySQL running
Simply run com.nike.cerberus.Main
. The following VM arguments should be set:
-D@appId=cms -D@environment=local -Dvault.addr=http://localhost:8200 -Dvault.token=<token>
With Vault and MySQL running
./gradlew clean build
./debugShadowJar.sh -Dvault.addr=http://localhost:8200 -Dvault.token=<token>
Import the build.gradle file.
See API.md
Cerberus Management Service is released under the Apache License, Version 2.0