mirror of
https://github.com/ceph/ceph
synced 2025-01-04 18:19:48 +00:00
232 lines
6.8 KiB
Python
232 lines
6.8 KiB
Python
|
"""
|
||
|
Deploy and configure Vault for Teuthology
|
||
|
"""
|
||
|
|
||
|
import argparse
|
||
|
import contextlib
|
||
|
import logging
|
||
|
import time
|
||
|
|
||
|
import httplib
|
||
|
import json
|
||
|
|
||
|
from teuthology import misc as teuthology
|
||
|
from teuthology import contextutil
|
||
|
from teuthology.orchestra import run
|
||
|
from teuthology.exceptions import ConfigError
|
||
|
|
||
|
|
||
|
log = logging.getLogger(__name__)
|
||
|
|
||
|
|
||
|
def assign_ports(ctx, config, initial_port):
|
||
|
"""
|
||
|
Assign port numbers starting from @initial_port
|
||
|
"""
|
||
|
port = initial_port
|
||
|
role_endpoints = {}
|
||
|
for remote, roles_for_host in ctx.cluster.remotes.iteritems():
|
||
|
for role in roles_for_host:
|
||
|
if role in config:
|
||
|
role_endpoints[role] = (remote.name.split('@')[1], port)
|
||
|
port += 1
|
||
|
|
||
|
return role_endpoints
|
||
|
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def download(ctx, config):
|
||
|
"""
|
||
|
Download Vault Release from Hashicopr website.
|
||
|
Remove downloaded file upon exit.
|
||
|
"""
|
||
|
assert isinstance(config, dict)
|
||
|
log.info('Downloading Vault...')
|
||
|
testdir = teuthology.get_testdir(ctx)
|
||
|
|
||
|
for (client, cconf) in config.items():
|
||
|
vault_version = cconf.get('version', '1.2.2')
|
||
|
|
||
|
ctx.cluster.only(client).run(
|
||
|
args=['mkdir', '-p', '{tdir}/vault'.format(tdir=testdir)])
|
||
|
|
||
|
cmd = [
|
||
|
'curl', '-L',
|
||
|
'https://releases.hashicorp.com/vault/{version}/vault_{version}_linux_amd64.zip'.format(version=vault_version), '-o',
|
||
|
'{tdir}/vault_{version}.zip'.format(tdir=testdir, version=vault_version)
|
||
|
]
|
||
|
ctx.cluster.only(client).run(args=cmd)
|
||
|
|
||
|
log.info('Extracting vault...')
|
||
|
# Using python in case unzip is not installed on hosts
|
||
|
cmd = ['python', '-m', 'zipfile', '-e',
|
||
|
'{tdir}/vault_{version}.zip'.format(tdir=testdir, version=vault_version),
|
||
|
'{tdir}/vault'.format(tdir=testdir)]
|
||
|
ctx.cluster.only(client).run(args=cmd)
|
||
|
|
||
|
try:
|
||
|
yield
|
||
|
finally:
|
||
|
log.info('Removing Vault...')
|
||
|
testdir = teuthology.get_testdir(ctx)
|
||
|
for client in config:
|
||
|
ctx.cluster.only(client).run(
|
||
|
args=[
|
||
|
'rm',
|
||
|
'-rf',
|
||
|
'{tdir}/vault_{version}.zip'.format(tdir=testdir, version=vault_version),
|
||
|
'{tdir}/vault'.format(tdir=testdir),
|
||
|
],
|
||
|
)
|
||
|
|
||
|
|
||
|
def get_vault_dir(ctx):
|
||
|
return '{tdir}/vault'.format(tdir=teuthology.get_testdir(ctx))
|
||
|
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def run_vault(ctx, config):
|
||
|
assert isinstance(config, dict)
|
||
|
|
||
|
for (client, cconf) in config.items():
|
||
|
(remote,) = ctx.cluster.only(client).remotes.iterkeys()
|
||
|
cluster_name, _, client_id = teuthology.split_role(client)
|
||
|
|
||
|
_, port = ctx.vault.endpoints[client]
|
||
|
listen_addr = "0.0.0.0:{}".format(port)
|
||
|
|
||
|
root_token = ctx.vault.root_token = cconf.get('root_token', 'root')
|
||
|
|
||
|
log.info("Starting Vault listening on %s ...", listen_addr)
|
||
|
v_params = [
|
||
|
'-dev',
|
||
|
'-dev-listen-address={}'.format(listen_addr),
|
||
|
'-dev-no-store-token',
|
||
|
'-dev-root-token-id={}'.format(root_token)
|
||
|
]
|
||
|
|
||
|
cmd = "chmod +x {vdir}/vault && {vdir}/vault server {vargs}".format(vdir=get_vault_dir(ctx), vargs=" ".join(v_params))
|
||
|
|
||
|
ctx.daemons.add_daemon(
|
||
|
remote, 'vault', client_id,
|
||
|
cluster=cluster_name,
|
||
|
args=['bash', '-c', cmd, run.Raw('& { read; kill %1; }')],
|
||
|
logger=log.getChild(client),
|
||
|
stdin=run.PIPE,
|
||
|
cwd=get_vault_dir(ctx),
|
||
|
wait=False,
|
||
|
check_status=False,
|
||
|
)
|
||
|
time.sleep(10)
|
||
|
try:
|
||
|
yield
|
||
|
finally:
|
||
|
log.info('Stopping Vault instance')
|
||
|
ctx.daemons.get_daemon('vault', client_id, cluster_name).stop()
|
||
|
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def setup_vault(ctx, config):
|
||
|
"""
|
||
|
Mount simple kv Secret Engine
|
||
|
Note: this will be extended to support transit secret engine
|
||
|
"""
|
||
|
data = {
|
||
|
"type": "kv",
|
||
|
"options": {
|
||
|
"version": "2"
|
||
|
}
|
||
|
}
|
||
|
|
||
|
(cclient, cconfig) = config.items()[0]
|
||
|
log.info('Mount kv version 2 secret engine')
|
||
|
send_req(ctx, cconfig, cclient, '/v1/sys/mounts/kv', json.dumps(data))
|
||
|
yield
|
||
|
|
||
|
|
||
|
def send_req(ctx, cconfig, client, path, body, method='POST'):
|
||
|
host, port = ctx.vault.endpoints[client]
|
||
|
req = httplib.HTTPConnection(host, port, timeout=30)
|
||
|
token = cconfig.get('root_token', 'atoken')
|
||
|
log.info("Send request to Vault: %s:%s with token: %s", host, port, token)
|
||
|
headers = {'X-Vault-Token': token}
|
||
|
req.request(method, path, headers=headers, body=body)
|
||
|
resp = req.getresponse()
|
||
|
log.info(resp.read())
|
||
|
if not (resp.status >= 200 and resp.status < 300):
|
||
|
raise Exception("Error Contacting Vault Server")
|
||
|
return resp
|
||
|
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def create_secrets(ctx, config):
|
||
|
(cclient, cconfig) = config.items()[0]
|
||
|
secrets = cconfig.get('secrets')
|
||
|
if secrets is None:
|
||
|
raise ConfigError("No secrets specified, please specify some.")
|
||
|
|
||
|
for secret in secrets:
|
||
|
try:
|
||
|
data = {
|
||
|
"data": {
|
||
|
"key": secret['secret']
|
||
|
}
|
||
|
}
|
||
|
except KeyError:
|
||
|
raise ConfigError('vault.secrets must have "secret" field')
|
||
|
try:
|
||
|
send_req(ctx, cconfig, cclient, secret['path'], json.dumps(data))
|
||
|
except KeyError:
|
||
|
raise ConfigError('vault.secrets must have "path" field')
|
||
|
|
||
|
log.info("secrets created")
|
||
|
yield
|
||
|
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def task(ctx, config):
|
||
|
"""
|
||
|
Deploy and configure Vault
|
||
|
|
||
|
Example of configuration:
|
||
|
|
||
|
tasks:
|
||
|
- vault:
|
||
|
client.0:
|
||
|
version: 1.2.2
|
||
|
root_token: test_root_token
|
||
|
secrets:
|
||
|
- path: kv/teuthology/key_a
|
||
|
secret: YmluCmJvb3N0CmJvb3N0LWJ1aWxkCmNlcGguY29uZgo=
|
||
|
- path: kv/teuthology/key_b
|
||
|
secret: aWIKTWFrZWZpbGUKbWFuCm91dApzcmMKVGVzdGluZwo=
|
||
|
"""
|
||
|
all_clients = ['client.{id}'.format(id=id_)
|
||
|
for id_ in teuthology.all_roles_of_type(ctx.cluster, 'client')]
|
||
|
if config is None:
|
||
|
config = all_clients
|
||
|
if isinstance(config, list):
|
||
|
config = dict.fromkeys(config)
|
||
|
|
||
|
overrides = ctx.config.get('overrides', {})
|
||
|
# merge each client section, not the top level.
|
||
|
for client in config.iterkeys():
|
||
|
if not config[client]:
|
||
|
config[client] = {}
|
||
|
teuthology.deep_merge(config[client], overrides.get('vault', {}))
|
||
|
|
||
|
log.debug('Vault config is %s', config)
|
||
|
|
||
|
ctx.vault = argparse.Namespace()
|
||
|
ctx.vault.endpoints = assign_ports(ctx, config, 8200)
|
||
|
ctx.vault.root_token = None
|
||
|
|
||
|
with contextutil.nested(
|
||
|
lambda: download(ctx=ctx, config=config),
|
||
|
lambda: run_vault(ctx=ctx, config=config),
|
||
|
lambda: setup_vault(ctx=ctx, config=config),
|
||
|
lambda: create_secrets(ctx=ctx, config=config)
|
||
|
):
|
||
|
yield
|
||
|
|