Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2012 the original author or authors. |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 16 | import org.yaml.snakeyaml.Yaml |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 17 | |
| 18 | ext { |
| 19 | |
| 20 | // Target registry to be used to publish docker images needed for deployment |
| 21 | targetReg = project.hasProperty('targetReg') ? project.getProperty('targetReg') : 'localhost:5000' |
| 22 | |
| 23 | // The tag used to tag the docker images push to the target registry |
| 24 | targetTag = project.hasProperty('targetTag') ? project.getProperty('targetTag') : 'candidate' |
| 25 | |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 26 | // Deployment target config file (yaml format); this can be overwritten from the command line |
| 27 | // using the -PdeployConfig=<file-path> syntax. |
| 28 | deployConfig = project.hasProperty('deployConfig') ? project.getProperty('deployConfig') : './config/default.yml' |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 29 | |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 30 | dockerPath = project.hasProperty('dockerPath') ? project.getProperty('dockerPath') : '/usr/bin' |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 31 | } |
| 32 | |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 33 | // Switch Configuration Image |
| 34 | |
| 35 | task buildSwitchqImage(type: Exec) { |
| 36 | commandLine "$dockerPath/docker", 'build', '-t', 'cord-maas-switchq', './switchq' |
| 37 | } |
| 38 | |
| 39 | task tagSwitchqImage(type: Exec) { |
| 40 | dependsOn buildSwitchqImage |
| 41 | commandLine "$dockerPath/docker", 'tag', 'cord-maas-switchq', "$targetReg/cord-maas-switchq:$targetTag" |
| 42 | } |
| 43 | |
| 44 | task publishSwitchqImage(type: Exec) { |
| 45 | dependsOn tagSwitchqImage |
| 46 | commandLine "$dockerPath/docker", 'push', "$targetReg/cord-maas-switchq:$targetTag" |
| 47 | } |
| 48 | |
| 49 | // Bootstrap Image |
| 50 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 51 | task buildBootstrapImage(type: Exec) { |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 52 | commandLine "$dockerPath/docker", 'build', '-t', 'cord-maas-bootstrap', './bootstrap' |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 53 | } |
| 54 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 55 | task tagBootstrapImage(type: Exec) { |
| 56 | dependsOn buildBootstrapImage |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 57 | commandLine "$dockerPath/docker", 'tag', 'cord-maas-bootstrap', "$targetReg/cord-maas-bootstrap:$targetTag" |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 58 | } |
| 59 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 60 | task publishBootstrapImage(type: Exec) { |
| 61 | dependsOn tagBootstrapImage |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 62 | commandLine "$dockerPath/docker", 'push', "$targetReg/cord-maas-bootstrap:$targetTag" |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 63 | } |
| 64 | |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 65 | // IP Allocator Image |
| 66 | |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 67 | task buildAllocationImage(type: Exec) { |
David K. Bainbridge | f22dc06 | 2016-05-31 15:35:39 -0700 | [diff] [blame] | 68 | commandLine "$dockerPath/docker", 'build', '-t', 'cord-ip-allocator', './ip-allocator' |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 69 | } |
| 70 | |
| 71 | task tagAllocationImage(type: Exec) { |
David K. Bainbridge | f22dc06 | 2016-05-31 15:35:39 -0700 | [diff] [blame] | 72 | dependsOn buildAllocationImage |
| 73 | commandLine "$dockerPath/docker", 'tag', 'cord-ip-allocator', "$targetReg/cord-ip-allocator:$targetTag" |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 74 | } |
| 75 | |
| 76 | task publishAllocationImage(type: Exec) { |
David K. Bainbridge | f22dc06 | 2016-05-31 15:35:39 -0700 | [diff] [blame] | 77 | dependsOn tagAllocationImage |
| 78 | commandLine "$dockerPath/docker", 'push', "$targetReg/cord-ip-allocator:$targetTag" |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 79 | } |
| 80 | |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 81 | // Provisioner Image |
| 82 | |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 83 | task buildProvisionerImage(type: Exec) { |
David K. Bainbridge | d86d96d | 2016-06-01 17:28:46 -0700 | [diff] [blame] | 84 | commandLine "$dockerPath/docker", 'build', '-t', 'cord-provisioner', './provisioner' |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 85 | } |
| 86 | |
| 87 | task tagProvisionerImage(type: Exec) { |
| 88 | dependsOn buildProvisionerImage |
| 89 | commandLine "$dockerPath/docker", 'tag', 'cord-provisioner', "$targetReg/cord-provisioner:$targetTag" |
| 90 | } |
| 91 | |
| 92 | task publishProvisionerImage(type: Exec) { |
| 93 | dependsOn tagProvisionerImage |
| 94 | commandLine "$dockerPath/docker", 'push', "$targetReg/cord-provisioner:$targetTag" |
| 95 | } |
| 96 | |
David K. Bainbridge | 9d1e02d | 2016-06-22 09:22:16 -0700 | [diff] [blame] | 97 | // Automation Image |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 98 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 99 | task buildAutomationImage(type: Exec) { |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 100 | commandLine "$dockerPath/docker", 'build', '-t', "cord-maas-automation", "-f", "./automation/Dockerfile", "./automation" |
David K. Bainbridge | efa951d | 2016-05-26 10:54:25 -0700 | [diff] [blame] | 101 | } |
| 102 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 103 | task tagAutomationImage(type: Exec) { |
| 104 | dependsOn buildAutomationImage |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 105 | commandLine "$dockerPath/docker", 'tag', 'cord-maas-automation', "$targetReg/cord-maas-automation:$targetTag" |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 106 | } |
| 107 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 108 | task publishAutomationImage(type: Exec) { |
| 109 | dependsOn tagAutomationImage |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 110 | commandLine "$dockerPath/docker", 'push', "$targetReg/cord-maas-automation:$targetTag" |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 111 | } |
| 112 | |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 113 | // DHCP Harvester Images |
| 114 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 115 | task buildHarvesterImage(type: Exec) { |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 116 | commandLine "$dockerPath/docker", 'build', '-t', "cord-dhcp-harvester", "./harvester" |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 117 | } |
| 118 | |
| 119 | task tagHarvesterImage(type: Exec) { |
| 120 | dependsOn buildHarvesterImage |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 121 | commandLine "$dockerPath/docker", 'tag', 'cord-dhcp-harvester', "$targetReg/cord-dhcp-harvester:$targetTag" |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 122 | } |
| 123 | |
| 124 | task publishHarvesterImage(type: Exec) { |
| 125 | dependsOn tagHarvesterImage |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 126 | commandLine "$dockerPath/docker", 'push', "$targetReg/cord-dhcp-harvester:$targetTag" |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 127 | } |
| 128 | |
| 129 | // ~~~~~~~~~~~~~~~~~~~ Global tasks ~~~~~~~~~~~~~~~~~~~~~~~ |
| 130 | |
| 131 | // To be used to fetch upstream binaries, clone repos, etc. |
| 132 | task fetch(type: Exec) { |
| 133 | // this is where we fetch upstream artifacts that we do not need internet for the build phase" |
| 134 | // Placeholdr example: |
David K. Bainbridge | 19b8d27 | 2016-05-26 21:20:43 -0700 | [diff] [blame] | 135 | commandLine "$dockerPath/docker", "pull", "golang:alpine" |
| 136 | commandLine "$dockerPath/docker", "pull", "python:2.7-alpine" |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 137 | } |
| 138 | |
| 139 | // To be used to generate all needed binaries that need to be present on the target |
| 140 | // as docker images in the local docker runner. |
| 141 | task buildImages { |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 142 | dependsOn buildBootstrapImage |
| 143 | dependsOn buildHarvesterImage |
David K. Bainbridge | 9d1e02d | 2016-06-22 09:22:16 -0700 | [diff] [blame] | 144 | dependsOn buildAutomationImage |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 145 | dependsOn buildAllocationImage |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 146 | dependsOn buildProvisionerImage |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 147 | dependsOn buildSwitchqImage |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 148 | } |
| 149 | |
| 150 | task tagImages { |
| 151 | dependsOn tagBootstrapImage |
| 152 | dependsOn tagHarvesterImage |
David K. Bainbridge | 9d1e02d | 2016-06-22 09:22:16 -0700 | [diff] [blame] | 153 | dependsOn tagAutomationImage |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 154 | dependsOn tagAllocationImage |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 155 | dependsOn tagProvisionerImage |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 156 | dependsOn tagSwitchqImage |
Zsolt Haraszti | 2a792f6 | 2016-05-12 17:49:02 -0700 | [diff] [blame] | 157 | } |
| 158 | |
| 159 | task publish { |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 160 | dependsOn publishBootstrapImage |
| 161 | dependsOn publishHarvesterImage |
David K. Bainbridge | 9d1e02d | 2016-06-22 09:22:16 -0700 | [diff] [blame] | 162 | dependsOn publishAutomationImage |
David K. Bainbridge | 8bc905c | 2016-05-31 14:07:10 -0700 | [diff] [blame] | 163 | dependsOn publishAllocationImage |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 164 | dependsOn publishProvisionerImage |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 165 | dependsOn publishSwitchqImage |
David K. Bainbridge | efa951d | 2016-05-26 10:54:25 -0700 | [diff] [blame] | 166 | } |
| 167 | |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 168 | // ~~~~~~~~~~~~~~~~~~~ Deployment / Test Tasks ~~~~~~~~~~~~~~~~~~~~~~~ |
| 169 | |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 170 | List.metaClass.asParam = { prefix, sep -> |
| 171 | if (delegate.size() == 0) { |
| 172 | "" |
| 173 | } |
| 174 | String result = "--" + prefix + "=" |
| 175 | String p = "" |
| 176 | delegate.each { |
| 177 | result += p + "${it}" |
| 178 | p = sep |
| 179 | } |
| 180 | result |
David K. Bainbridge | b541504 | 2016-05-13 17:06:10 -0700 | [diff] [blame] | 181 | } |
| 182 | |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 183 | List.metaClass.p = { value, name -> |
| 184 | if (value != null && value != "") { |
| 185 | delegate << name + "=" + value |
| 186 | } else { |
| 187 | delegate |
| 188 | } |
| 189 | } |
| 190 | |
| 191 | List.metaClass.p = { spec -> |
| 192 | if (spec != null && spec != "") { |
| 193 | delegate += spec |
| 194 | } else { |
| 195 | delegate |
| 196 | } |
| 197 | } |
| 198 | |
David K. Bainbridge | f418170 | 2016-06-17 14:44:03 -0700 | [diff] [blame] | 199 | task prime (type: Exec) { |
| 200 | println "Using deployment config: $deployConfig" |
| 201 | File configFile = new File(deployConfig) |
| 202 | def yaml = new Yaml() |
| 203 | def config = yaml.load(configFile.newReader()) |
| 204 | |
| 205 | executable = "ansible-playbook" |
| 206 | args = ["-i", config.seedServer.ip + ','] |
| 207 | |
| 208 | if ( config.seedServer.user != null && config.seedServer.user != "" ) { |
| 209 | args = args << "--user=$config.seedServer.user" |
| 210 | } |
| 211 | |
| 212 | def extraVars = [] |
| 213 | if (config.seedServer) { |
| 214 | extraVars = extraVars.p(config.seedServer.extraVars) |
| 215 | .p(config.seedServer.password, "ansible_ssh_pass") |
| 216 | .p(config.seedServer.sudoPassword, "ansible_sudo_pass") |
| 217 | .p(config.seedServer.fabric_ip, "fabric_ip") |
| 218 | .p(config.seedServer.management_ip, "management_ip") |
| 219 | .p(config.seedServer.management_network, "management_network") |
| 220 | .p(config.seedServer.management_iface, "management_iface") |
| 221 | .p(config.seedServer.external_ip, "external_ip") |
| 222 | .p(config.seedServer.external_network, "external_network") |
| 223 | .p(config.seedServer.external_iface, "external_iface") |
| 224 | .p(config.seedServer.fabric_ip, "fabric_ip") |
| 225 | .p(config.seedServer.fabric_network, "fabric_network") |
| 226 | .p(config.seedServer.fabric_iface, "fabric_iface") |
| 227 | .p(config.seedServer.domain, "domain") |
David K. Bainbridge | be58a0d | 2016-06-22 15:43:02 -0700 | [diff] [blame] | 228 | .p(config.seedServer.virtualbox_support, "virtualbox_support") |
| 229 | .p(config.seedServer.power_helper_user, "power_helper_user") |
| 230 | .p(config.seedServer.power_helper_host, "power_helper_host") |
| 231 | .p(config.seedServer.port, "ansible_ssh_port") |
David K. Bainbridge | f418170 | 2016-06-17 14:44:03 -0700 | [diff] [blame] | 232 | } |
| 233 | |
| 234 | if (config.otherServers) { |
| 235 | extraVars = extraVars.p(config.otherServers.location, "prov_location") |
| 236 | .p(config.otherServers.rolesPath, "prov_role_path") |
| 237 | .p(config.otherServers.role, "prov_role") |
| 238 | } |
| 239 | |
| 240 | if (config.docker) { |
| 241 | extraVars = extraVars.p(config.docker.registry, "docker_registry") |
| 242 | .p(config.docker.imageVersion, "docker_image_version") |
| 243 | } |
| 244 | |
| 245 | def skipTags = [].p(config.seedServer.skipTags) |
| 246 | |
| 247 | args = args.p(skipTags.asParam("skip-tags", ",")).p(extraVars.asParam("extra-vars", " ")) << "prime-node.yml" |
| 248 | } |
| 249 | |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 250 | task deploy (type: Exec) { |
| 251 | println "Using deployment config: $deployConfig" |
| 252 | File configFile = new File(deployConfig) |
| 253 | def yaml = new Yaml() |
| 254 | def config = yaml.load(configFile.newReader()) |
| 255 | |
| 256 | executable = "ansible-playbook" |
David K. Bainbridge | 13c765c | 2016-05-26 11:24:22 -0700 | [diff] [blame] | 257 | args = ["-i", config.seedServer.ip + ','] |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 258 | |
David K. Bainbridge | 13c765c | 2016-05-26 11:24:22 -0700 | [diff] [blame] | 259 | if ( config.seedServer.user != null && config.seedServer.user != "" ) { |
| 260 | args = args << "--user=$config.seedServer.user" |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 261 | } |
| 262 | |
| 263 | def extraVars = [] |
David K. Bainbridge | 13c765c | 2016-05-26 11:24:22 -0700 | [diff] [blame] | 264 | if (config.seedServer) { |
| 265 | extraVars = extraVars.p(config.seedServer.extraVars) |
| 266 | .p(config.seedServer.password, "ansible_ssh_pass") |
| 267 | .p(config.seedServer.sudoPassword, "ansible_sudo_pass") |
| 268 | .p(config.seedServer.fabric_ip, "fabric_ip") |
| 269 | .p(config.seedServer.management_ip, "management_ip") |
David K. Bainbridge | c82a446 | 2016-06-14 12:39:01 -0700 | [diff] [blame] | 270 | .p(config.seedServer.management_network, "management_network") |
| 271 | .p(config.seedServer.management_iface, "management_iface") |
David K. Bainbridge | 13c765c | 2016-05-26 11:24:22 -0700 | [diff] [blame] | 272 | .p(config.seedServer.external_ip, "external_ip") |
David K. Bainbridge | c82a446 | 2016-06-14 12:39:01 -0700 | [diff] [blame] | 273 | .p(config.seedServer.external_network, "external_network") |
| 274 | .p(config.seedServer.external_iface, "external_iface") |
| 275 | .p(config.seedServer.fabric_ip, "fabric_ip") |
| 276 | .p(config.seedServer.fabric_network, "fabric_network") |
| 277 | .p(config.seedServer.fabric_iface, "fabric_iface") |
| 278 | .p(config.seedServer.domain, "domain") |
David K. Bainbridge | be58a0d | 2016-06-22 15:43:02 -0700 | [diff] [blame] | 279 | .p(config.seedServer.virtualbox_support, "virtualbox_support") |
David K. Bainbridge | c82a446 | 2016-06-14 12:39:01 -0700 | [diff] [blame] | 280 | .p(config.seedServer.power_helper_user, "power_helper_user") |
| 281 | .p(config.seedServer.power_helper_host, "power_helper_host") |
David K. Bainbridge | be58a0d | 2016-06-22 15:43:02 -0700 | [diff] [blame] | 282 | .p(config.seedServer.port, "ansible_ssh_port") |
David K. Bainbridge | 6ea57c1 | 2016-06-06 23:29:12 -0700 | [diff] [blame] | 283 | } |
| 284 | |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 285 | if (config.otherServers) { |
| 286 | extraVars = extraVars.p(config.otherServers.location, "prov_location") |
| 287 | .p(config.otherServers.rolesPath, "prov_role_path") |
| 288 | .p(config.otherServers.role, "prov_role") |
| 289 | } |
| 290 | |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 291 | if (config.docker) { |
| 292 | extraVars = extraVars.p(config.docker.registry, "docker_registry") |
| 293 | .p(config.docker.imageVersion, "docker_image_version") |
| 294 | } |
| 295 | |
David K. Bainbridge | 13c765c | 2016-05-26 11:24:22 -0700 | [diff] [blame] | 296 | def skipTags = [].p(config.seedServer.skipTags) |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 297 | |
| 298 | args = args.p(skipTags.asParam("skip-tags", ",")).p(extraVars.asParam("extra-vars", " ")) << "head-node.yml" |
David K. Bainbridge | 10b0c11 | 2016-05-24 13:17:23 -0700 | [diff] [blame] | 299 | } |