Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 1 | import os |
| 2 | import base64 |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 3 | import jinja2 |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 4 | import string |
| 5 | import sys |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 6 | import urllib2 |
| 7 | import urlparse |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 8 | import xmlrpclib |
| 9 | |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 10 | from xos.config import Config |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 11 | from core.models import Service, ServiceController, ServiceControllerResource, LoadableModule, LoadableModuleResource, XOS |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 12 | from xos.logger import Logger, logging |
| 13 | |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 14 | from django.utils import timezone |
| 15 | |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 16 | logger = Logger(level=logging.INFO) |
| 17 | |
Scott Baker | fb4c98c | 2016-08-08 17:45:45 -0700 | [diff] [blame] | 18 | def add_unique(list, item): |
| 19 | if not item in list: |
| 20 | list.append(item) |
| 21 | |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 22 | class XOSBuilder(object): |
Scott Baker | 9b49133 | 2016-10-24 10:25:41 -0700 | [diff] [blame^] | 23 | UI_KINDS=["models", "admin", "admin_template", "django_library", "rest_service", "rest_tenant", "tosca_custom_types", "tosca_resource","public_key","vendor_js"] |
Scott Baker | 18c8917 | 2016-06-02 16:40:25 -0700 | [diff] [blame] | 24 | SYNC_CONTROLLER_KINDS=["synchronizer", "private_key", "public_key"] |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 25 | SYNC_ALLCONTROLLER_KINDS=["models", "django_library"] |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 26 | |
Scott Baker | 7581c25 | 2016-05-27 13:12:47 -0700 | [diff] [blame] | 27 | def __init__(self): |
Scott Baker | 2979cae | 2016-10-20 14:48:32 -0700 | [diff] [blame] | 28 | self.source_sync_image = "xosproject/xos" # "xosproject/xos-synchronizer-openstack" |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 29 | self.build_dir = "/opt/xos/BUILD/" |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 30 | self.build_tainted = False |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 31 | |
| 32 | # stuff that has to do with downloading |
| 33 | |
Scott Baker | fb4c98c | 2016-08-08 17:45:45 -0700 | [diff] [blame] | 34 | def get_base_dest_dir(self, scr): |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 35 | xos_base = "opt/xos" |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 36 | service_name = scr.loadable_module.name |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 37 | base_dirs = {"models": "%s/services/%s/" % (xos_base, service_name), |
| 38 | "admin": "%s/services/%s/" % (xos_base, service_name), |
Scott Baker | e360ff8 | 2016-06-13 10:55:23 -0700 | [diff] [blame] | 39 | "admin_template": "%s/services/%s/templates/" % (xos_base, service_name), |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 40 | "django_library": "%s/services/%s/" % (xos_base, service_name), |
| 41 | "synchronizer": "%s/synchronizers/%s/" % (xos_base, service_name), |
| 42 | "tosca_custom_types": "%s/tosca/custom_types/" % (xos_base), |
Scott Baker | 18c8917 | 2016-06-02 16:40:25 -0700 | [diff] [blame] | 43 | "tosca_resource": "%s/tosca/resources/" % (xos_base), |
Scott Baker | 16cfb9c | 2016-06-07 15:37:03 -0700 | [diff] [blame] | 44 | "rest_service": "%s/api/service/" % (xos_base), |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 45 | "rest_tenant": "%s/api/tenant/" % (xos_base), |
Scott Baker | cfd5372 | 2016-06-13 19:02:43 -0700 | [diff] [blame] | 46 | "private_key": "%s/services/%s/keys/" % (xos_base, service_name), |
Scott Baker | 9b49133 | 2016-10-24 10:25:41 -0700 | [diff] [blame^] | 47 | "public_key": "%s/services/%s/keys/" % (xos_base, service_name), |
| 48 | "vendor_js": "%s/core/xoslib/static/vendor/" % (xos_base)} |
Scott Baker | 260a21c | 2016-06-13 10:42:49 -0700 | [diff] [blame] | 49 | dest_dir = base_dirs[scr.kind] |
| 50 | |
Scott Baker | fb4c98c | 2016-08-08 17:45:45 -0700 | [diff] [blame] | 51 | return dest_dir |
| 52 | |
| 53 | def get_dest_dir(self, scr): |
| 54 | dest_dir = self.get_base_dest_dir(scr) |
| 55 | |
Scott Baker | 260a21c | 2016-06-13 10:42:49 -0700 | [diff] [blame] | 56 | if scr.subdirectory: |
| 57 | dest_dir = os.path.join(dest_dir, scr.subdirectory) |
| 58 | |
| 59 | return dest_dir |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 60 | |
| 61 | def get_build_fn(self, scr): |
| 62 | dest_dir = self.get_dest_dir(scr) |
| 63 | dest_fn = os.path.split(urlparse.urlsplit(scr.full_url).path)[-1] |
| 64 | return os.path.join(dest_dir, dest_fn) |
| 65 | |
| 66 | def get_download_fn(self, scr): |
| 67 | dest_fn = self.get_build_fn(scr) |
| 68 | return os.path.join(self.build_dir, dest_fn) |
| 69 | |
| 70 | def read_manifest(self, scr, fn): |
| 71 | manifest = [] |
| 72 | manifest_lines = file(fn).readlines() |
| 73 | manifest_lines = [x.strip() for x in manifest_lines] |
Scott Baker | c2a2fe9 | 2016-06-02 13:19:10 -0700 | [diff] [blame] | 74 | manifest_lines = [x for x in manifest_lines if x] |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 75 | for line in manifest_lines: |
Scott Baker | c2a2fe9 | 2016-06-02 13:19:10 -0700 | [diff] [blame] | 76 | url_parts = urlparse.urlsplit(scr.full_url) |
| 77 | new_path = os.path.join(os.path.join(*os.path.split(url_parts.path)[:-1]),line) |
| 78 | url = urlparse.urlunsplit( (url_parts.scheme, url_parts.netloc, new_path, url_parts.query, url_parts.fragment) ) |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 79 | |
| 80 | build_fn = os.path.join(self.get_dest_dir(scr), line) |
Scott Baker | c2a2fe9 | 2016-06-02 13:19:10 -0700 | [diff] [blame] | 81 | download_fn = os.path.join(self.build_dir, build_fn) |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 82 | |
| 83 | manifest.append( (url, download_fn, build_fn) ) |
| 84 | return manifest |
| 85 | |
| 86 | def download_file(self, url, dest_fn): |
| 87 | logger.info("Download %s to %s" % (url, dest_fn)) |
| 88 | if not os.path.exists(os.path.dirname(dest_fn)): |
| 89 | os.makedirs(os.path.dirname(dest_fn)) |
| 90 | obj = urllib2.urlopen(url) |
| 91 | file(dest_fn,"w").write(obj.read()) |
| 92 | |
Scott Baker | c2a2fe9 | 2016-06-02 13:19:10 -0700 | [diff] [blame] | 93 | # make python files executable |
| 94 | if dest_fn.endswith(".py"): # and contents.startswith("#!"): |
| 95 | os.chmod(dest_fn, 0755) |
| 96 | |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 97 | def download_resource(self, scr): |
| 98 | if scr.format == "manifest": |
| 99 | manifest_fn = self.get_download_fn(scr) |
| 100 | self.download_file(scr.full_url, manifest_fn) |
Scott Baker | c2a2fe9 | 2016-06-02 13:19:10 -0700 | [diff] [blame] | 101 | manifest = self.read_manifest(scr, manifest_fn) |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 102 | for (url, download_fn, build_fn) in manifest: |
| 103 | self.download_file(url, download_fn) |
| 104 | else: |
| 105 | self.download_file(scr.full_url, self.get_download_fn(scr)) |
| 106 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 107 | # XXX docker creates a new container and commits it for every single COPY |
| 108 | # line in the dockerfile. This causes services with many files (for example, |
| 109 | # vsg) to take ~ 10-15 minutes to build the docker file. So instead we'll copy |
| 110 | # the whole build directory, and then run a script that copies the files |
| 111 | # we want. |
| 112 | |
| 113 | # def get_docker_lines(self, scr): |
| 114 | # if scr.format == "manifest": |
| 115 | # manifest_fn = self.get_download_fn(scr) |
| 116 | # manifest = self.read_manifest(scr, manifest_fn) |
| 117 | # lines = [] |
| 118 | # for (url, download_fn, build_fn) in manifest: |
| 119 | # script.append("mkdir -p |
| 120 | # #lines.append("COPY %s /%s" % (build_fn, build_fn)) |
| 121 | # return lines |
| 122 | # else: |
| 123 | # build_fn = self.get_build_fn(scr) |
| 124 | # #return ["COPY %s /%s" % (build_fn, build_fn)] |
| 125 | |
| 126 | # def get_controller_docker_lines(self, controller, kinds): |
| 127 | # need_service_init_py = False |
| 128 | # dockerfile=[] |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 129 | # for scr in controller.loadable_module_resources.all(): |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 130 | # if scr.kind in kinds: |
| 131 | # lines = self.get_docker_lines(scr) |
| 132 | # dockerfile = dockerfile + lines |
| 133 | # if scr.kind in ["admin", "models"]: |
| 134 | # need_service_init_py = True |
| 135 | # |
| 136 | # if need_service_init_py: |
| 137 | # file(os.path.join(self.build_dir, "opt/xos/empty__init__.py"),"w").write("") |
| 138 | # dockerfile.append("COPY opt/xos/empty__init__.py /opt/xos/services/%s/__init__.py" % controller.name) |
| 139 | # |
| 140 | # return dockerfile |
| 141 | |
| 142 | def get_script_lines(self, scr): |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 143 | if scr.format == "manifest": |
| 144 | manifest_fn = self.get_download_fn(scr) |
| 145 | manifest = self.read_manifest(scr, manifest_fn) |
| 146 | lines = [] |
| 147 | for (url, download_fn, build_fn) in manifest: |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 148 | lines.append("mkdir -p /%s" % os.path.dirname(build_fn)) |
| 149 | lines.append("cp /build/%s /%s" % (build_fn, build_fn)) |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 150 | return lines |
| 151 | else: |
| 152 | build_fn = self.get_build_fn(scr) |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 153 | return ["mkdir -p /%s" % os.path.dirname(build_fn), |
| 154 | "cp /build/%s /%s" % (build_fn, build_fn)] |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 155 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 156 | def get_controller_script_lines(self, controller, kinds): |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 157 | need_service_init_py = False |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 158 | script=[] |
Scott Baker | fb4c98c | 2016-08-08 17:45:45 -0700 | [diff] [blame] | 159 | inits=[] |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 160 | for scr in list(controller.loadable_module_resources.all()): |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 161 | if not (scr.kind in kinds): |
| 162 | continue |
| 163 | |
| 164 | # Check and see if the resource we're trying to install has |
| 165 | # disappeared. This may happen if the onboarding synchronizer |
| 166 | # container has been destroyed and restarted. In this case, flag |
| 167 | # the resource for re-download, and set the build_tainted bit |
| 168 | # so we can throw an exception after we've evaluated all |
| 169 | # resources. |
| 170 | |
| 171 | download_fn = self.get_download_fn(scr) |
| 172 | if not os.path.exists(download_fn): |
| 173 | logger.info("File %s is missing; dirtying the resource" % download_fn) |
| 174 | scr.backend_status = "2 - download_fn is missing" |
| 175 | scr.updated = timezone.now() |
| 176 | scr.save(update_fields=['backend_status', 'updated']) |
| 177 | self.build_tainted = True |
| 178 | continue |
| 179 | |
| 180 | lines = self.get_script_lines(scr) |
| 181 | script = script + lines |
| 182 | |
Scott Baker | fb4c98c | 2016-08-08 17:45:45 -0700 | [diff] [blame] | 183 | # compute the set of __init__.py files that we will need |
| 184 | if scr.kind in ["admin", "models", "rest_service", "rest_tenant"]: |
| 185 | dir = self.get_base_dest_dir(scr) |
| 186 | add_unique(inits, dir) |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 187 | |
Scott Baker | fb4c98c | 2016-08-08 17:45:45 -0700 | [diff] [blame] | 188 | if scr.subdirectory: |
| 189 | for part in scr.subdirectory.split("/"): |
| 190 | dir = os.path.join(dir, part) |
| 191 | add_unique(inits, dir) |
| 192 | |
| 193 | for init in inits: |
| 194 | script.append("echo > %s" % os.path.join("/",init,"__init__.py")) |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 195 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 196 | return script |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 197 | |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 198 | def check_controller_unready(self, controller): |
| 199 | unready_resources=[] |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 200 | for scr in controller.loadable_module_resources.all(): |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 201 | if (not scr.backend_status) or (not scr.backend_status.startswith("1")): |
| 202 | unready_resources.append(scr) |
| 203 | |
| 204 | return unready_resources |
| 205 | |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 206 | # stuff that has to do with building |
| 207 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 208 | def create_xos_app_data(self, name, script, app_list, migration_list): |
Scott Baker | ae196c3 | 2016-06-01 23:29:22 -0700 | [diff] [blame] | 209 | if not os.path.exists(os.path.join(self.build_dir,"opt/xos/xos")): |
| 210 | os.makedirs(os.path.join(self.build_dir,"opt/xos/xos")) |
| 211 | |
| 212 | if app_list: |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 213 | script.append("mkdir -p /opt/xos/xos") |
| 214 | script.append("cp /build/opt/xos/xos/%s_xosbuilder_app_list /opt/xos/xos/xosbuilder_app_list" % name) |
| 215 | #dockerfile.append("COPY opt/xos/xos/%s_xosbuilder_app_list /opt/xos/xos/xosbuilder_app_list" % name) |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 216 | file(os.path.join(self.build_dir, "opt/xos/xos/%s_xosbuilder_app_list") % name, "w").write("\n".join(app_list)+"\n") |
Scott Baker | ae196c3 | 2016-06-01 23:29:22 -0700 | [diff] [blame] | 217 | |
| 218 | if migration_list: |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 219 | script.append("mkdir -p /opt/xos/xos") |
| 220 | script.append("cp /build/opt/xos/xos/%s_xosbuilder_migration_list /opt/xos/xos/xosbuilder_migration_list" % name) |
| 221 | #dockerfile.append("COPY opt/xos/xos/%s_xosbuilder_migration_list /opt/xos/xos/xosbuilder_migration_list" % name) |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 222 | file(os.path.join(self.build_dir, "opt/xos/xos/%s_xosbuilder_migration_list") % name, "w").write("\n".join(migration_list)+"\n") |
Scott Baker | ae196c3 | 2016-06-01 23:29:22 -0700 | [diff] [blame] | 223 | |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 224 | def create_ui_dockerfile(self): |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 225 | self.build_tainted = False |
Scott Baker | d2ba749 | 2016-06-15 14:33:34 -0700 | [diff] [blame] | 226 | xos = XOS.objects.all()[0] |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 227 | dockerfile_fn = "Dockerfile.UI" |
| 228 | |
Scott Baker | ae196c3 | 2016-06-01 23:29:22 -0700 | [diff] [blame] | 229 | app_list = [] |
| 230 | migration_list = [] |
| 231 | |
Scott Baker | d2ba749 | 2016-06-15 14:33:34 -0700 | [diff] [blame] | 232 | dockerfile = ["FROM %s" % xos.source_ui_image] |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 233 | script = [] |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 234 | for controller in LoadableModule.objects.all(): |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 235 | if self.check_controller_unready(controller): |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 236 | logger.warning("Loadable Module %s has unready resources" % str(controller)) |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 237 | continue |
| 238 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 239 | #dockerfile = dockerfile + self.get_controller_docker_lines(controller, self.UI_KINDS) |
| 240 | script = script + self.get_controller_script_lines(controller, self.UI_KINDS) |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 241 | if controller.loadable_module_resources.filter(kind="models").exists(): |
Scott Baker | ae196c3 | 2016-06-01 23:29:22 -0700 | [diff] [blame] | 242 | app_list.append("services." + controller.name) |
| 243 | migration_list.append(controller.name) |
| 244 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 245 | self.create_xos_app_data("ui", script, app_list, migration_list) |
| 246 | |
| 247 | file(os.path.join(self.build_dir, "install-xos.sh"), "w").write("\n".join(script)+"\n") |
| 248 | dockerfile.append("COPY . /build/") |
| 249 | dockerfile.append("RUN bash /build/install-xos.sh") |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 250 | |
| 251 | file(os.path.join(self.build_dir, dockerfile_fn), "w").write("\n".join(dockerfile)+"\n") |
| 252 | |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 253 | if self.build_tainted: |
| 254 | raise Exception("Build was tainted due to errors") |
| 255 | |
Scott Baker | 8d252b6 | 2016-06-01 23:08:04 -0700 | [diff] [blame] | 256 | return {"dockerfile_fn": dockerfile_fn, |
| 257 | "docker_image_name": "xosproject/xos-ui"} |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 258 | |
| 259 | def create_synchronizer_dockerfile(self, controller): |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 260 | self.build_tainted = False |
| 261 | |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 262 | if not controller.loadable_module_resources.filter(kind="synchronizer").exists(): |
| 263 | # it doesn't have a synchronizer, therefore it doesn't need a dockerfile |
| 264 | return None |
| 265 | |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 266 | # bake in the synchronizer from this controller |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 267 | sync_lines = self.get_controller_script_lines(controller, self.SYNC_CONTROLLER_KINDS) |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 268 | |
| 269 | if self.build_tainted: |
| 270 | raise Exception("Build was tainted due to errors") |
| 271 | |
| 272 | # If there's no sync_lines for this ServiceController, then it must not |
| 273 | # have a synchronizer. |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 274 | if not sync_lines: |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 275 | return None |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 276 | |
| 277 | dockerfile_fn = "Dockerfile.%s" % controller.name |
| 278 | dockerfile = ["FROM %s" % self.source_sync_image] |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 279 | script = [] |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 280 | |
| 281 | # Now bake in models from this controller as well as the others |
| 282 | # It's important to bake all services in, because some services' |
| 283 | # synchronizers may depend on models from another service. |
| 284 | app_list = [] |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 285 | for c in LoadableModule.objects.all(): |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 286 | script = script + self.get_controller_script_lines(c, self.SYNC_ALLCONTROLLER_KINDS) |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 287 | if c.loadable_module_resources.filter(kind="models").exists(): |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 288 | app_list.append("services." + c.name) |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 289 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 290 | self.create_xos_app_data(controller.name, script, app_list, None) |
Scott Baker | 513ea45 | 2016-06-02 16:03:26 -0700 | [diff] [blame] | 291 | |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 292 | script = script + sync_lines |
| 293 | |
| 294 | file(os.path.join(self.build_dir, "install-%s.sh" % controller.name), "w").write("\n".join(script)+"\n") |
| 295 | dockerfile.append("COPY . /build/") |
| 296 | dockerfile.append("RUN bash /build/install-%s.sh" % controller.name) |
| 297 | |
Scott Baker | c2a2fe9 | 2016-06-02 13:19:10 -0700 | [diff] [blame] | 298 | file(os.path.join(self.build_dir, dockerfile_fn), "w").write("\n".join(dockerfile)+"\n") |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 299 | |
Scott Baker | 2b3b1ad | 2016-08-02 22:29:18 -0700 | [diff] [blame] | 300 | if self.build_tainted: |
| 301 | raise Exception("Build was tainted due to errors") |
| 302 | |
Scott Baker | 8d252b6 | 2016-06-01 23:08:04 -0700 | [diff] [blame] | 303 | return {"dockerfile_fn": dockerfile_fn, |
| 304 | "docker_image_name": "xosproject/xos-synchronizer-%s" % controller.name} |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 305 | |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 306 | def create_docker_compose(self): |
| 307 | xos = XOS.objects.all()[0] |
| 308 | |
Scott Baker | 29b677b | 2016-06-07 10:20:00 -0700 | [diff] [blame] | 309 | volume_list = [] |
| 310 | for volume in xos.volumes.all(): |
| 311 | volume_list.append({"host_path": volume.host_path, |
| 312 | "container_path": volume.container_path, |
| 313 | "read_only": volume.read_only}) |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 314 | |
Scott Baker | b1f466c | 2016-08-22 10:33:09 -0700 | [diff] [blame] | 315 | if xos.extra_hosts: |
| 316 | extra_hosts = [x.strip() for x in xos.extra_hosts.split(",")] |
| 317 | else: |
| 318 | extra_hosts = [] |
| 319 | |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 320 | containers = {} |
| 321 | |
Scott Baker | f68418f | 2016-06-16 15:30:09 -0700 | [diff] [blame] | 322 | # containers["xos_db"] = \ |
| 323 | # {"image": "xosproject/xos-postgres", |
| 324 | # "expose": [5432]} |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 325 | |
| 326 | containers["xos_ui"] = \ |
| 327 | {"image": "xosproject/xos-ui", |
| 328 | "command": "python /opt/xos/manage.py runserver 0.0.0.0:%d --insecure --makemigrations" % xos.ui_port, |
| 329 | "ports": {"%d"%xos.ui_port : "%d"%xos.ui_port}, |
Scott Baker | f68418f | 2016-06-16 15:30:09 -0700 | [diff] [blame] | 330 | #"links": ["xos_db"], |
Scott Baker | 0c0139a | 2016-06-22 14:54:23 -0700 | [diff] [blame] | 331 | "external_links": ["%s:%s" % (xos.db_container_name, "xos_db")], |
Scott Baker | b1f466c | 2016-08-22 10:33:09 -0700 | [diff] [blame] | 332 | "extra_hosts": extra_hosts, |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 333 | "volumes": volume_list} |
| 334 | |
Scott Baker | 885894a | 2016-08-24 22:31:10 -0700 | [diff] [blame] | 335 | if xos.no_start: |
| 336 | containers["xos_ui"]["command"] = "sleep 864000" |
| 337 | |
Scott Baker | 584cd89 | 2016-06-09 16:10:31 -0700 | [diff] [blame] | 338 | # containers["xos_bootstrap_ui"] = {"image": "xosproject/xos", |
| 339 | # "command": "python /opt/xos/manage.py runserver 0.0.0.0:%d --insecure --makemigrations" % xos.bootstrap_ui_port, |
| 340 | # "ports": {"%d"%xos.bootstrap_ui_port : "%d"%xos.bootstrap_ui_port}, |
Scott Baker | 0c0139a | 2016-06-22 14:54:23 -0700 | [diff] [blame] | 341 | # #"external_links": ["%s:%s" % (xos.db_container_name, "xos_db")], |
Scott Baker | 584cd89 | 2016-06-09 16:10:31 -0700 | [diff] [blame] | 342 | # "links": ["xos_db"], |
| 343 | # "volumes": volume_list} |
Scott Baker | be41a12 | 2016-06-06 10:40:40 -0700 | [diff] [blame] | 344 | |
Scott Baker | 057a6a1 | 2016-06-13 15:45:55 -0700 | [diff] [blame] | 345 | if not xos.frontend_only: |
| 346 | for c in ServiceController.objects.all(): |
| 347 | if self.check_controller_unready(c): |
| 348 | logger.warning("Controller %s has unready resources" % str(c)) |
| 349 | continue |
Scott Baker | 7806f62 | 2016-06-07 17:45:04 -0700 | [diff] [blame] | 350 | |
Scott Baker | c01ce94 | 2016-10-03 11:55:12 -0700 | [diff] [blame] | 351 | if c.loadable_module_resources.filter(kind="synchronizer").exists(): |
Scott Baker | 0761590 | 2016-06-13 21:41:52 -0700 | [diff] [blame] | 352 | if c.synchronizer_run and c.synchronizer_config: |
Scott Baker | cf3586f | 2016-06-14 00:22:44 -0700 | [diff] [blame] | 353 | command = 'bash -c "sleep 120; cd /opt/xos/synchronizers/%s; python ./%s -C %s"' % (c.name, c.synchronizer_run, c.synchronizer_config) |
Scott Baker | 0761590 | 2016-06-13 21:41:52 -0700 | [diff] [blame] | 354 | else: |
| 355 | command = 'bash -c "sleep 120; cd /opt/xos/synchronizers/%s; bash ./run.sh"' % c.name |
| 356 | |
| 357 | containers["xos_synchronizer_%s" % c.name] = \ |
| 358 | {"image": "xosproject/xos-synchronizer-%s" % c.name, |
| 359 | "command": command, |
Scott Baker | 0c0139a | 2016-06-22 14:54:23 -0700 | [diff] [blame] | 360 | "external_links": ["%s:%s" % (xos.db_container_name, "xos_db")], |
Scott Baker | b1f466c | 2016-08-22 10:33:09 -0700 | [diff] [blame] | 361 | "extra_hosts": extra_hosts, |
Scott Baker | 0761590 | 2016-06-13 21:41:52 -0700 | [diff] [blame] | 362 | "volumes": volume_list} |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 363 | |
Scott Baker | 0df54fa | 2016-08-31 15:25:06 -0700 | [diff] [blame] | 364 | if c.no_start: |
| 365 | containers["xos_synchronizer_%s" % c.name]["command"] = "sleep 864000" |
| 366 | |
Scott Baker | aab8a29 | 2016-06-03 16:32:45 -0700 | [diff] [blame] | 367 | vars = { "containers": containers } |
| 368 | |
| 369 | template_loader = jinja2.FileSystemLoader( "/opt/xos/synchronizers/onboarding/templates/" ) |
| 370 | template_env = jinja2.Environment(loader=template_loader) |
| 371 | template = template_env.get_template("docker-compose.yml.j2") |
| 372 | buffer = template.render(vars) |
| 373 | |
| 374 | if not os.path.exists("/opt/xos/synchronizers/onboarding/docker-compose"): |
| 375 | os.makedirs("/opt/xos/synchronizers/onboarding/docker-compose") |
| 376 | file("/opt/xos/synchronizers/onboarding/docker-compose/docker-compose.yml", "w").write(buffer) |
| 377 | |
Scott Baker | 8d252b6 | 2016-06-01 23:08:04 -0700 | [diff] [blame] | 378 | # def build_xos(self): |
| 379 | # dockerfiles=[] |
| 380 | # dockerfiles.append(self.create_ui_dockerfile()) |
| 381 | # |
| 382 | # for controller in ServiceController.objects.all(): |
| 383 | # dockerfiles.append(self.create_synchronizer_dockerfile(controller)) |
Scott Baker | b0eb23e | 2016-06-01 16:08:04 -0700 | [diff] [blame] | 384 | |
| 385 | |
| 386 | |