Skip to content

A small service for securely delivering Vault authorization keys.

License

Notifications You must be signed in to change notification settings

ssnippets/vault-gatekeeper-mesos

 
 

Repository files navigation

vault-gatekeeper-mesos

Build Status

Vault-Gatekeeper-Mesos (VGM) is a small service for delivering Vault token to other services who's lifecycles are managed by Mesos or one of it's frameworks (such as Marathon).

VGM takes the Cubbyhole Authenication approach outlined by Jeff Mitchell on Vault Blog. Specifically Vault response wrapping is used as outlined in the Vault documentation. In short, a service will request a vault token from VGM supplying its Mesos task id. VGM will then check with Mesos to ensure that the task has been recently started, and if so, request the creation of 2 tokens (temp and perm). The temp token, which can only be used twice, will be first used by Vault internally to write the perm token into it's own Cubbyhole at the path response. This temp token is then provided to the service which can use that token to retrieve the perm token. VGM can also ensure the correct policies are set on the perm token by using an internal configuration based on the task's name in Mesos.

Requires Vault 0.6.0 or greater

Deploying

You can grab a binary from the releases or deploy the docker image channelmeter/vault-gatekeeper-mesos. When deploying VGM you must consider how you will deliver it's Vault authorization token. To issue a token to VGM on startup you can use the VAULT_TOKEN or APP_ID (and accompayning USER_ID_*) environment variables. Otherwise VGM functions like Vault in that it starts "sealed" and it must be "unsealed". It can be unsealed via JSON API, or you can direct your browser to http://gate.keep.er:9201 and use the UI provided there.

VGM also supports the client environment variables used by vault such as, VAULT_ADDR, VAULT_SKIP_VERIFY, VAULT_CACERT and VAULT_CAPATH.

Arguments

LISTEN_ADDR | -listen - Default: :9091 - The address this service should listen on.

TLS_CERT | -tls-cert - Path to TLS certificate. If this value is set, gatekeeper will be served over TLS.

TLS_KEY | -tls-key - Path to TLS key. If this value is set, gatekeeper will be served over TLS.

MESOS_MASTER | -mesos - The address of the mesos master. Can be either a zookeeper link (zk://zoo1:2181,zoo2:2181/mesos) or a http link to a single or multiple mesos masters (http://leader.mesos:5050).

VAULT_ADDR | -vault - The address of the vault server.

VAULT_SKIP_VERIFY | tls-skip-verify - Do not verify TLS certificate.

VAULT_CACERT | -ca-cert - Path to a PEM encoded CA cert file to use to verify the Vault server SSL certificate.

VAULT_CAPATH | -ca-path - Path to a directory of PEM encoded CA cert files to verify the Vault server SSL certificate.

GATE_POLICIES | -policies - The path on the generic vault backend to load policies from (See Policies section).

TASK_LIFE | -task-life - Default: 2m - The maximum age of a task before VGM will refuse to issue tokens for it.

RECREATE_TOKEN | -self-recreate-token - Default: false - When the current token is reaching it's MAX_TTL (720h by default), recreate the token with the same policy instead of trying to renew (requires a sudo/root token, and for the token to have a ttl).

Vault Startup Authorization Methods

VAULT_TOKEN - Vault authorization token to make requests with.

CUBBY_TOKEN | -cubby-token - Temporary vault authorization token that has a cubbyhole secret in CUBBY_PATH that contains the permanent vault token.

CUBBY_PATH | -cubby-path - Path to key in cubbyhole. By default this is /vault-token.

WRAPPED_TOKEN_AUTH | -wrapped-token-auth - Temporary vault authorization token that has a wrapped permanent vault token.

APP_ID | -auth-appid - Use the app-id authorization method with this app id.

USER_ID_METHOD | -auth-userid-method - With the app-id authorization method, this argument decides how VGM should generate the user id. Valid values are mac and file.

USER_ID_INTERFACE | -auth-userid-interface - When USER_ID_METHOD is mac, this is the name of the interface that the mac address should be generated from.

USER_ID_PATH | -auth-userid-path - When USER_ID_METHOD is file, read the data from this file as the user_id.

USER_ID_HASH | -auth-userid-hash - Hash the user_id with this scheme. Valid values are sha256, sha1, and md5.

USER_ID_SALT | -auth-userid-salt - When provided, the user_id will be hashed with salt$user_id.

Unsealing

By default, VGM, like Vault, will start sealed. The APP_ID and VAULT_TOKEN arguments can be started with VGM in order to start unsealed.

While VGM is sealed there are two ways to unseal it, via the API or with your browser (by just opening the url VGM is listening on). See the POST /unseal section for information on the different unseal methods.

Policies

VGM will create token's with given policies by using the data in its policies config. This config is pulled from vault from the generic backend (and supplied by you). A policies config is a simple json structure, with the key name being the Mesos task name (with the Marathon framework this is your app name) and the value being select token options. Policy names support glob-style matching, with the longest pattern taking priority. .

{
	"web-server":{
		"policies":["web"],
		"meta":{"foo":"bar"},
		"ttl":3000,
		"num_uses":0,
	},
	"ChronosTask:my_special_task":{
		"policies":["special_task"],
		"meta":{"foo":"bar"},
		"ttl":3000,
		"num_uses":0,
	},
	"ChronosTask:*":{
		"policies":["batch"],
		"meta":{"foo":"bar"},
		"ttl":3000,
		"num_uses":0,
	},
	"*":{
		"policies":["default"],
		"ttl":1500,
	}
}

In the above, any task starting with ChronosTask: will get the batch policy except ChronosTask:my_special_task, which will get special_task.

You will have to use the Vault API in order to set th epolicies to your backend. Assuming your policy is saved as policy.json, here's how to save that information using cURL.

$ curl -X POST -H "X-Vault-Token: <MY TOKEN>" -H "Content-Type: application/json" -d @policy.json http://vault/v1/secret/gatekeeper

If you update the policy secret, you will need to restart VGM or reload the policies via the /policies/reload API (see below) to apply the changes.

API

GET /status.json

Gets the status of VGM.

Response -

{
	"ok":true,
	"started":"time VGM was started",
	"status":"Either Sealed or Unsealed",
	"uptime":"Duration of uptime",
	"stats":{
		"requests":"number of token requests",
		"successful":"number of successful requests",
		"denied":"number of denied requests"
	}
}

POST /seal

Seal the service. The token that was provided will be forgotten.

Response -

{
	"ok":true,
	"status":"Either Sealed or Unsealed",
	"error":"error if any"
}

POST /unseal

Unseal the service.

Parameters (application/json) -

  • type - One of token, userpass, app-id, github, cubby
  • token - Vault Authorization token if type is token, Github Personal token if type is github, temp token with {"token":"perm_token"} in cubby_path if type is cubby.
  • cubby_path - The path in v1/cubbyhole/ when using cubby authorization. Default will be /vault-token.
  • username - Username for userpass authenication.
  • password - Password for userpass authenication.
  • app_id - See APP_ID in Vault Startup Authorization Methods
  • user_id_method - See USER_ID_METHOD in Vault Startup Authorization Methods
  • user_id_interface - See USER_ID_INTERFACE in Vault Startup Authorization Methods
  • user_id_path - See USER_ID_PATH in Vault Startup Authorization Methods
  • user_id_hash - See USER_ID_HASH in Vault Startup Authorization Methods
  • user_id_salt - See USER_ID_SALT in Vault Startup Authorization Methods

Response -

{
	"ok":true,
	"status":"Either Sealed or Unsealed",
	"error":"error if any"
}

POST /policies/reload

Reload the gatekeeper policies from the Vault secret path (GATE_POLICIES | -policies). Do this after updating the policy secret in the Vault.

Response -

{
	"ok":true,
	"status":"Either Sealed or Unsealed",
	"error":"error if any"
}

POST /token

Request a token.

Parameters (application/json) -

  • task_id - The Mesos Task ID of the service.

Response -

{
	"ok":true,
	"status":"Either Sealed or Unsealed",
	"token":"temp cubbyhole token",
	"error":"error if any"
}

Sample

Here is a simple program that gets a vault token.

package main

import (
	"fmt"
	"github.com/channelmeter/vault-gatekeeper-mesos/gatekeeper"
)

func main() {
	token, err := gatekeeper.RequestVaultToken("geard.3d151450-1092-11e6-8d2c-00163e105043")
	fmt.Printf("%v %v\n", token, err)
}

About

A small service for securely delivering Vault authorization keys.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Go 96.5%
  • Shell 2.9%
  • HCL 0.6%