diff --git a/.gitignore b/.gitignore index 84e345ad8..b5cc23268 100644 --- a/.gitignore +++ b/.gitignore @@ -9,3 +9,7 @@ ChangeLog devenv/* .coverage coverage.xml +imagebuild/coreos/build +imagebuild/coreos/dist +imagebuild/coreos/oem/authorized_keys +imagebuild/coreos/UPLOAD diff --git a/imagebuild/README.md b/imagebuild/README.md new file mode 100644 index 000000000..d073c9aad --- /dev/null +++ b/imagebuild/README.md @@ -0,0 +1,4 @@ +teeth-agent images +================== + +coreos - Builds a CoreOS Ramdisk and Kernel suitable for running teeth-agent diff --git a/imagebuild/coreos/Makefile b/imagebuild/coreos/Makefile new file mode 100644 index 000000000..26a1cc2b8 --- /dev/null +++ b/imagebuild/coreos/Makefile @@ -0,0 +1,17 @@ +default: docker coreos + +docker: + ./docker_build.bash + +coreos: + mkdir UPLOAD + ./coreos-oem-inject.py oem UPLOAD + +clean: + rm -rf teeth-agent + rm -f oem/container.tar.gz + rm -f UPLOAD/coreos_production_pxe_image-oem.cpio.gz + rm -f UPLOAD/coreos_production_pxe.vmlinuz + +docker_clean: + ./docker_clean.bash diff --git a/imagebuild/coreos/README.md b/imagebuild/coreos/README.md new file mode 100644 index 000000000..4798fe810 --- /dev/null +++ b/imagebuild/coreos/README.md @@ -0,0 +1,33 @@ +# teeth-agent CoreOS Image builder. + +Builds a CoreOS image suitable for running the teeth-agent on a server. + +# Requirements + +Must be run from a linux machine with a working docker installation and python-pip + +Run the following locally or from a virtualenv to install the python requirements +``` +pip install -r requirements.txt +``` + +# Instructions + +To create a docker repository and embed it into a CoreOS pxe image: +``` +make +``` + +To just create the docker repository in oem/container.tar.gz: +``` +make docker +``` + +To embed the oem/ directory into a CoreOS pxe image: + +Note: In order to have the ability to ssh into the created image, you need to +pass ssh keys in via the kernel command line for CoreOS, or create +oem/authorized_keys with the keys you need added before building the image. +``` +make coreos +``` diff --git a/imagebuild/coreos/coreos-oem-inject.py b/imagebuild/coreos/coreos-oem-inject.py new file mode 100755 index 000000000..e471b9a95 --- /dev/null +++ b/imagebuild/coreos/coreos-oem-inject.py @@ -0,0 +1,246 @@ +#!/usr/bin/env python + +import os +import sys +import time +import requests +import tempfile +import shutil +from plumbum import local, cmd + +COREOS_VERSION="dev-channel" + +COREOS_ARCH="amd64-generic" +COREOS_BASE_URL="http://storage.core-os.net/coreos/{}/{}".format(COREOS_ARCH, COREOS_VERSION) +COREOS_PXE_DIGESTS="coreos_production_pxe_image.cpio.gz.DIGESTS.asc" +COREOS_PXE_KERNEL="coreos_production_pxe.vmlinuz" +COREOS_PXE_IMAGE="coreos_production_pxe_image.cpio.gz" +COREOS_PXE_IMAGE_URL = "{}/{}".format(COREOS_BASE_URL, COREOS_PXE_IMAGE) +COREOS_PXE_KERNEL_URL = "{}/{}".format(COREOS_BASE_URL, COREOS_PXE_KERNEL) +COREOS_PXE_DIGESTS_URL = "{}/{}".format(COREOS_BASE_URL, COREOS_PXE_DIGESTS) + + + +def get_etag(cache_name): + etag_file = "{}.etag".format(cache_name) + if not os.path.exists(etag_file): + return None + with open(etag_file, 'rb') as fp: + etag = fp.read() + etag.strip() + return etag + +def save_etag(cache_name, etag): + etag_file = "{}.etag".format(cache_name) + with open(etag_file, 'w+b') as fp: + fp.write(etag) + +def cache_file(cache_name, remote_url): + print("{} <- {}".format(cache_name, remote_url)) + etag = get_etag(cache_name) + headers = {} + if etag: + headers['If-None-Match'] = etag + + start = time.time() + r = requests.get(remote_url, headers=headers) + + if r.status_code == 304: + print("[etag-match]") + return + + if r.status_code != 200: + raise RuntimeError('Failed to download {}, got HTTP {} Status Code.'.format(remote_url, r.status_code)) + + with open(cache_name, 'w+b') as fp: + fp.write(r.content) + + print("{} bytes in {} seconds".format(len(r.content), time.time() - start)) + save_etag(cache_name, r.headers['etag']) + +def inject_oem(archive, oem_dir, output_file): + d = tempfile.mkdtemp(prefix="oem-inject") + try: + with local.cwd(d): + dest_oem_dir = os.path.join(d, 'usr', 'share', 'oem') + uz = cmd.gunzip["-c", archive] + extract = cmd.cpio["-iv"] + chain = uz | extract + print chain + chain() + + shutil.copytree(oem_dir, dest_oem_dir) + + find = cmd.find['.', '-depth', '-print'] + cpio = cmd.cpio['-o', '-H', 'newc'] + gz = cmd.gzip + chain = find | cmd.sort | cpio | gz > output_file + print chain + chain() + finally: + shutil.rmtree(d) + return output_file + +def validate_digests(digests, target, hash_type='sha1'): + with local.cwd(os.path.dirname(digests)): + gethashes = cmd.grep['-i', '-A1', '^# {} HASH$'.format(hash_type), digests] + forthis = cmd.grep[os.path.basename(target)] + viasum = local[hash_type + "sum"]['-c', '/dev/stdin'] + chain = gethashes | forthis | viasum + print chain + chain() + +def main(): + if len(sys.argv) != 3: + print("usage: {} [oem-directory-to-inject] [output-directory]".format(os.path.basename(__file__))) + return + + oem_dir = os.path.abspath(os.path.expanduser(sys.argv[1])) + output_dir = os.path.abspath(os.path.expanduser(sys.argv[2])) + + if not os.path.exists(oem_dir): + print("Error: {} doesn't exist.".format(oem_dir)) + return + + if not os.path.exists(os.path.join(oem_dir, 'run.sh')): + print("Error: {} is missing oem.sh".format(oem_dir)) + return + + here = os.path.abspath(os.path.dirname(__file__)) + + top_cache_dir = os.path.join(os.path.dirname(here), ".image_cache") + cache_dir = os.path.join(top_cache_dir, COREOS_ARCH, COREOS_VERSION) + + if not os.path.exists(cache_dir): + os.makedirs(cache_dir) + + orig_cpio = os.path.join(cache_dir, COREOS_PXE_IMAGE) + digests = os.path.join(cache_dir, COREOS_PXE_DIGESTS) + kernel = os.path.join(cache_dir, COREOS_PXE_KERNEL) + + cache_file(digests, COREOS_PXE_DIGESTS_URL) + gpg_verify_file(digests) + cache_file(kernel, COREOS_PXE_KERNEL_URL) + validate_digests(digests, kernel) + cache_file(orig_cpio, COREOS_PXE_IMAGE_URL) + validate_digests(digests, orig_cpio) + + if not os.path.exists(output_dir): + os.makedirs(output_dir) + + output_kernel = os.path.join(output_dir, os.path.basename(kernel)) + output_cpio = os.path.join(output_dir, os.path.basename(orig_cpio).replace('.cpio.gz', '-oem.cpio.gz')) + inject_oem(orig_cpio, oem_dir, output_cpio) + shutil.copy(kernel, output_kernel) + +def gpg_verify_file(ascfile): + d = tempfile.mkdtemp(prefix="oem-gpg-validate") + try: + tmpring = os.path.join(d, 'tmp.gpg') + key = os.path.join(d, 'coreos.key') + with open(key, 'w+b') as fp: + fp.write(gpg_key()) + + i = cmd.gpg['--batch', + '--no-default-keyring', + '--keyring', + tmpring, + '--import', + key] + print(i) + i() + + r = cmd.gpg['--batch', + '--no-default-keyring', + '--keyring', + tmpring, + '--verify', + ascfile] + print(r) + r() + + finally: + shutil.rmtree(d) + +def gpg_key(): + GPG_LONG_ID="50E0885593D2DCB4" + GPG_KEY="""-----BEGIN PGP PUBLIC KEY BLOCK----- + Version: GnuPG v2.0.20 (GNU/Linux) + + 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 + BeiJ6tEeDHDzdA== + =4Qn0 + -----END PGP PUBLIC KEY BLOCK----- + """ + return GPG_KEY + +if __name__ == "__main__": + main() diff --git a/imagebuild/coreos/docker_build.bash b/imagebuild/coreos/docker_build.bash new file mode 100755 index 000000000..d8b6715dc --- /dev/null +++ b/imagebuild/coreos/docker_build.bash @@ -0,0 +1,26 @@ +#!/bin/bash +# +# docker_build.bash - Prepares and outputs a tarball'd docker repository +# suitable for injection into a coreos pxe image +# + +set -e + +OUTPUT_FILE="oem/container.tar.gz" + +# If there's already a container.tar.gz, don't overwrite it -- instead, bail +if [[ -e "${OUTPUT_FILE}" ]]; then + echo "${OUTPUT_FILE} already exists. Will not overwrite. Exiting." + exit 1 +fi + +# Build the docker image +cd ../../teeth-agent +docker build -t oemdocker . +cd - + +# Export the oemdocker repository to a tarball so it can be embedded in CoreOS +# TODO: Investigate running a container and using "export" to flatten the +# image to shrink the CoreOS fs size. This will also require run.sh to +# use docker import instead of docker load as well. +docker save oemdocker | gzip > ${OUTPUT_FILE} diff --git a/imagebuild/coreos/docker_clean.bash b/imagebuild/coreos/docker_clean.bash new file mode 100755 index 000000000..091c8998a --- /dev/null +++ b/imagebuild/coreos/docker_clean.bash @@ -0,0 +1,16 @@ +#!/bin/bash +# +# Cleans up docker images and containers + +containers=$(docker ps -a -q) +images=$(docker images -q) + +# All the docker commands followed by || true because occassionally docker +# will fail to remove an image or container, & I want make to keep going anyway +if [[ ! -z "$containers" ]]; then + docker rm $containers || true +fi + +if [[ ! -z "$images" ]]; then + docker rmi $images || true +fi diff --git a/imagebuild/coreos/oem/run.sh b/imagebuild/coreos/oem/run.sh new file mode 100755 index 000000000..ff09be45d --- /dev/null +++ b/imagebuild/coreos/oem/run.sh @@ -0,0 +1,31 @@ +#!/bin/bash + +set -e + +# CoreOS by default only has an OEM partition of 2GB. This isn't large enough +# for some images. Remount it with a larger size. Note: When CoreOS changes to +# r/w /, instead of remounting here, we'll use rootflags= to set the size. +mount -o remount,size=20G /media/state + +cd /usr/share/oem/ + +mkdir -pm 0700 /home/core/.ssh + +# TODO: Use proper https://github.com/coreos/init/blob/master/bin/update-ssh-keys script +if [[ -e authorized_keys ]]; then + cat authorized_keys >> /home/core/.ssh/authorized_keys +fi + +chown -R core:core /home/core/.ssh/ + +# We have to wait until docker is started to proceed +# In a perfect world I'd use inotifywait, but that doesn't exist on coreos +while [ ! -e /var/run/docker.sock ]; do + sleep 1; +done + +# TODO: Use docker import (and export the image) to shrink image size +docker load < container.tar.gz + +systemctl enable --runtime /usr/share/oem/system/* +systemctl start teeth-agent.service diff --git a/imagebuild/coreos/oem/system/teeth-agent.service b/imagebuild/coreos/oem/system/teeth-agent.service new file mode 100644 index 000000000..a25c2ba7d --- /dev/null +++ b/imagebuild/coreos/oem/system/teeth-agent.service @@ -0,0 +1,6 @@ +[Service] +ExecStart=/usr/bin/docker run -p 9999:9999 -privileged=true -v=/sys:/mnt/sys oemdocker /usr/local/bin/teeth-agent --ipaddr="`ip a | grep '10\.' | sed -e 's/inet \(10\.[0-9\.]\+\).*/\1/'`" +Restart=always + +[Install] +WantedBy=oem.target diff --git a/teeth_agent/agent.py b/teeth_agent/agent.py index 9083c989e..62292fdde 100644 --- a/teeth_agent/agent.py +++ b/teeth_agent/agent.py @@ -107,9 +107,10 @@ class TeethAgentHeartbeater(threading.Thread): class TeethAgent(object): - def __init__(self, api_url, listen_address): + def __init__(self, api_url, listen_address, ipaddr): self.api_url = api_url self.listen_address = listen_address + self.ipaddr = ipaddr self.mode_implementation = None self.version = pkg_resources.get_distribution('teeth-agent').version self.api = app.VersionSelectorApplication(self) @@ -223,5 +224,5 @@ def _load_mode_implementation(mode_name): return mgr.driver -def build_agent(api_url, listen_host, listen_port): - return TeethAgent(api_url, (listen_host, listen_port)) +def build_agent(api_url, listen_host, listen_port, ipaddr): + return TeethAgent(api_url, (listen_host, listen_port), ipaddr) diff --git a/teeth_agent/cmd/agent.py b/teeth_agent/cmd/agent.py index bdfa8d41d..92f4ef386 100644 --- a/teeth_agent/cmd/agent.py +++ b/teeth_agent/cmd/agent.py @@ -39,8 +39,13 @@ def run(): type=int, help='The port to listen on') + parser.add_argument('--ipaddr', + required=True, + help='The external IP address to advertise to ironic') + args = parser.parse_args() logging.configure() agent.build_agent(args.api_url, args.listen_host, - args.listen_port).run() + args.listen_port, + args.ipaddr).run() diff --git a/teeth_agent/tests/agent.py b/teeth_agent/tests/agent.py index ce5995396..33b011267 100644 --- a/teeth_agent/tests/agent.py +++ b/teeth_agent/tests/agent.py @@ -122,7 +122,8 @@ class TestBaseAgent(unittest.TestCase): encoding.SerializationViews.PUBLIC, indent=4) self.agent = agent.TeethAgent('https://fake_api.example.org:8081/', - ('localhost', 9999)) + ('localhost', 9999), + '192.168.1.1') def assertEqualEncoded(self, a, b): # Evidently JSONEncoder.default() can't handle None (??) so we have to