diff options
| author | Monty Taylor <mordred@inaugust.com> | 2015-08-01 02:53:18 +1000 |
|---|---|---|
| committer | Monty Taylor <mordred@inaugust.com> | 2015-08-01 03:21:11 +1000 |
| commit | 22ab0f1a87a61a84dc78721ecbdde43ec2b1c556 (patch) | |
| tree | ec912b14a07cf7e1e046e4b9e957feb727e18800 /src | |
| parent | 71db6e4ca1962d708f2da6cf6e532ace38eabfba (diff) | |
Add talks
Diffstat (limited to 'src')
| -rw-r--r-- | src/talks/a-vision-for-the-future/index.html | 2 | ||||
| -rw-r--r-- | src/talks/ansible-cloud/index.html | 792 | ||||
| -rw-r--r-- | src/talks/glean/index.html | 209 | ||||
| -rw-r--r-- | src/talks/index/index.html | 74 | ||||
| -rw-r--r-- | src/talks/lemmings/index.html | 331 | ||||
| -rw-r--r-- | src/talks/now-what/index.html | 436 | ||||
| -rw-r--r-- | src/talks/os-client-config/index.html | 242 | ||||
| -rw-r--r-- | src/talks/product-management/index.html | 454 | ||||
| -rw-r--r-- | src/talks/tripleo-ansible/index.html | 794 |
9 files changed, 3333 insertions, 1 deletions
diff --git a/src/talks/a-vision-for-the-future/index.html b/src/talks/a-vision-for-the-future/index.html index 88fdbe5..abd3851 100644 --- a/src/talks/a-vision-for-the-future/index.html +++ b/src/talks/a-vision-for-the-future/index.html | |||
| @@ -483,7 +483,7 @@ | |||
| 483 | Creative Commons Attribution 4.0 International License | 483 | Creative Commons Attribution 4.0 International License |
| 484 | </a>. | 484 | </a>. |
| 485 | <br /> | 485 | <br /> |
| 486 | Source code available at <a href='http://git.inaugust.com/cgit/mordred-talks'>http://git.inaugust.com/cgit/mordred-talks</a> | 486 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> |
| 487 | </div> | 487 | </div> |
| 488 | 488 | ||
| 489 | </div> | 489 | </div> |
diff --git a/src/talks/ansible-cloud/index.html b/src/talks/ansible-cloud/index.html new file mode 100644 index 0000000..37e66c6 --- /dev/null +++ b/src/talks/ansible-cloud/index.html | |||
| @@ -0,0 +1,792 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>NoOps with Ansible and Puppet</title> | ||
| 8 | |||
| 9 | <meta name="author" content="Monty Taylor" /> | ||
| 10 | |||
| 11 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 12 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 13 | |||
| 14 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 15 | |||
| 16 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 17 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 18 | |||
| 19 | <!-- For syntax highlighting --> | ||
| 20 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 21 | |||
| 22 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 23 | <script> | ||
| 24 | var link = document.createElement('link'); | ||
| 25 | link.rel = 'stylesheet'; | ||
| 26 | link.type = 'text/css'; | ||
| 27 | link.href = | ||
| 28 | window.location.search.match(/print-pdf/gi) ? '/css/print/pdf.css' : | ||
| 29 | '/css/print/paper.css'; | ||
| 30 | document.getElementsByTagName('head')[0].appendChild(link); | ||
| 31 | </script> | ||
| 32 | </head> | ||
| 33 | <body> | ||
| 34 | <body> | ||
| 35 | |||
| 36 | <div class="background"> | ||
| 37 | <img alt="" id="head-icon" width="218" height="67" | ||
| 38 | src="/images/openstack-cloud-software-horizontal-small.png" /></div> | ||
| 39 | |||
| 40 | |||
| 41 | <div class="reveal"> | ||
| 42 | <div class="slides"> | ||
| 43 | |||
| 44 | <section data-state="cover"> | ||
| 45 | <img src="/images/openstack-cloud-software-vertical-large.png" | ||
| 46 | alt="OpenStack" id="cover"/> | ||
| 47 | |||
| 48 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 49 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 50 | property="dct:title" | ||
| 51 | rel="dct:type"> | ||
| 52 | NoOps with Ansible and Puppet | ||
| 53 | </span></h1> | ||
| 54 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 55 | property="cc:attributionName">Monty Taylor</h3> | ||
| 56 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 57 | rel="cc:attributionURL" | ||
| 58 | href='http://inaugust.com/talks/ansible-cloud.html'>http://inaugust.com/talks/ansible-cloud.html</a> </h4> | ||
| 59 | <h3> twitter: @e_monty </h3> | ||
| 60 | </section> | ||
| 61 | |||
| 62 | <section id="who-am-i" class="slide level2"> | ||
| 63 | <h1>Who am I?</h1> | ||
| 64 | <ul> | ||
| 65 | <li>Distinguished Technologist at HP</li> | ||
| 66 | <li>OpenStack Technical Committee</li> | ||
| 67 | <li>OpenStack Foundation Board of Directors</li> | ||
| 68 | <li>OpenStack Developer Infrastructure Core Team</li> | ||
| 69 | <li>Former Consultant for MySQL, Inc</li> | ||
| 70 | <li>Former Drizzle Core Developer</li> | ||
| 71 | </ul> | ||
| 72 | </section> | ||
| 73 | |||
| 74 | <section id="what-are-we-going-to-talk-about" class="slide level2"> | ||
| 75 | <h1>What are we going to talk about?</h1> | ||
| 76 | <ul> | ||
| 77 | <li>NoOps</li> | ||
| 78 | <li>Cloud Applications</li> | ||
| 79 | <li>Puppet</li> | ||
| 80 | <li>Ansible</li> | ||
| 81 | </ul> | ||
| 82 | </section> | ||
| 83 | |||
| 84 | <section id="no-ops" class="slide level2"> | ||
| 85 | <h1>NoOps</h1> | ||
| 86 | <p class="fragment">NoOps means developers can code and let a service deploy, manage and scale their code</p> | ||
| 87 | <p class="fragment">I don't want to "do ops"</p> | ||
| 88 | <p class="fragment">I want to change the system by landing commits</p> | ||
| 89 | <p class="fragment">If I have to use my root access, it's a bug</p> | ||
| 90 | </section> | ||
| 91 | |||
| 92 | <section id="cloud-native" class="slide level2"> | ||
| 93 | <h1>Cloud Native</h1> | ||
| 94 | <ul> | ||
| 95 | <li>Ephemeral Compute</li> | ||
| 96 | <li>Data services</li> | ||
| 97 | <li>Design your applications to be resilient via scale out</li> | ||
| 98 | </ul> | ||
| 99 | </section> | ||
| 100 | |||
| 101 | <section id="cloud-scale-out" class="slide level2"> | ||
| 102 | <h1>Cloud Scale Out</h1> | ||
| 103 | <ul> | ||
| 104 | <li>Forget HA of one system</li> | ||
| 105 | <li>Forget long-lived systems</li> | ||
| 106 | <li>Shared-nothing for EVERYTHING</li> | ||
| 107 | </ul> | ||
| 108 | </section> | ||
| 109 | |||
| 110 | <section id="cloud-scale-out-is-great-for-new-applications" class="slide level2"> | ||
| 111 | <h1>Cloud Scale Out is great for new applications</h1> | ||
| 112 | </section> | ||
| 113 | |||
| 114 | <section id="what-about-existing-applications" class="slide level2"> | ||
| 115 | <h1>What about existing applications?</h1> | ||
| 116 | </section> | ||
| 117 | |||
| 118 | <section> | ||
| 119 | <section id="openstack-infra" data-transition='zoom'> | ||
| 120 | <h1>OpenStack Infra</h1> | ||
| 121 | </section> | ||
| 122 | |||
| 123 | <section id="tooling-automation-and-ci-for-openstack-project" class="slide level2" data-transition='zoom'> | ||
| 124 | <h1>Tooling, Automation and CI for OpenStack Project</h1> | ||
| 125 | </section> | ||
| 126 | |||
| 127 | <section id="developers" class="slide level2" data-transition='zoom'> | ||
| 128 | <h1>2000 Developers</h1> | ||
| 129 | </section> | ||
| 130 | |||
| 131 | <section id="gated-commits" class="slide level2" data-transition='zoom'> | ||
| 132 | <h1>Gated Commits</h1> | ||
| 133 | <p>Every commit is fully integration tested (twice) before landing</p> | ||
| 134 | </section> | ||
| 135 | |||
| 136 | <section id="each-test-runs-on-a-single-use-cloud-slave" class="slide level2" data-transition='zoom'> | ||
| 137 | <h1>Each Test Runs on a Single Use Cloud Slave</h1> | ||
| 138 | <p>This is that "cloud scale out" part</p> | ||
| 139 | </section> | ||
| 140 | |||
| 141 | <section id="million-test-jobs-in-the-last-6-months" class="slide level2" data-transition='zoom'> | ||
| 142 | <h1>1.7 Million Test Jobs in the last 6 Months</h1> | ||
| 143 | </section> | ||
| 144 | |||
| 145 | <section id="15-million-tests-in-december" class="slide level2" data-transition='zoom'> | ||
| 146 | <h1>15 Million Tests in December</h1> | ||
| 147 | </section> | ||
| 148 | |||
| 149 | <section id="terabytes-of-log-data-in-six-months" class="slide level2" data-transition='zoom'> | ||
| 150 | <h1>18 Terabytes of Log Data in six months</h1> | ||
| 151 | </section> | ||
| 152 | |||
| 153 | <section id="we-have-no-servers" class="slide level2" data-transition='zoom'> | ||
| 154 | <h1>We have no servers</h1> | ||
| 155 | <p>It all runs across HP and Rackspace Public Clouds.</p> | ||
| 156 | </section> | ||
| 157 | |||
| 158 | </section> | ||
| 159 | |||
| 160 | <section id="architecture" class="slide level2"> | ||
| 161 | <h1>Architecture</h1> | ||
| 162 | <p><img src="/images/infra_architecture.jpg" alt="image" /></p> | ||
| 163 | </section> | ||
| 164 | |||
| 165 | <section id="it-didnt-start-this-way" class="slide level2"> | ||
| 166 | <h1>It didn't start this way</h1> | ||
| 167 | <p><img src="/images/original.jpg" alt="image" /></p> | ||
| 168 | </section> | ||
| 169 | |||
| 170 | <section id="step-one-puppet" class="titleslide slide level1"> | ||
| 171 | <h1>Step One: Puppet</h1> | ||
| 172 | </section> | ||
| 173 | |||
| 174 | <section id="overview" class="slide level2"> | ||
| 175 | <h1>Overview</h1> | ||
| 176 | <ul> | ||
| 177 | <li>Open Source Config Management System</li> | ||
| 178 | <li>Written in Ruby</li> | ||
| 179 | <li>Models intended state</li> | ||
| 180 | <li>Wants to own entire system</li> | ||
| 181 | </ul> | ||
| 182 | </section> | ||
| 183 | |||
| 184 | <section id="config-management-is-great" class="slide level2"> | ||
| 185 | <h1>Config Management is Great</h1> | ||
| 186 | <ul> | ||
| 187 | <li>Repeatable and consistent machines</li> | ||
| 188 | <li>Code Review</li> | ||
| 189 | <li>Collaboration from non-root users</li> | ||
| 190 | <li>Less repetition for me</li> | ||
| 191 | <li>Open Source Infrastructure</li> | ||
| 192 | <li><a href="http://git.openstack.org/cgit/openstack-infra/system-config/">http://git.openstack.org/cgit/openstack-infra/system-config/</a></li> | ||
| 193 | <li><a href="http://puppetdb.openstack.org/">http://puppetdb.openstack.org/</a></li> | ||
| 194 | </ul> | ||
| 195 | </section> | ||
| 196 | |||
| 197 | <section id="ruby-dsl" class="slide level2"> | ||
| 198 | <h1>Ruby DSL</h1> | ||
| 199 | <pre><code> | ||
| 200 | package { 'git': | ||
| 201 | ensure => 'present', | ||
| 202 | } | ||
| 203 | </code></pre> | ||
| 204 | </section> | ||
| 205 | |||
| 206 | <section id="leaky-abstraction" class="slide level2"> | ||
| 207 | <h1>Leaky Abstraction</h1> | ||
| 208 | <pre><code> | ||
| 209 | if !defined(Package['git']) { | ||
| 210 | package { 'git': | ||
| 211 | ensure => 'present', | ||
| 212 | } | ||
| 213 | } | ||
| 214 | </code></pre> | ||
| 215 | </section> | ||
| 216 | |||
| 217 | <section id="three-ways-to-run" class="slide level2"> | ||
| 218 | <h1>Three ways to run</h1> | ||
| 219 | <ul> | ||
| 220 | <li>puppet apply</li> | ||
| 221 | <li>puppetmaster + puppet agent daemons</li> | ||
| 222 | <li>puppetmaster + puppet agent non-daemon</li> | ||
| 223 | </ul> | ||
| 224 | </section> | ||
| 225 | |||
| 226 | <section id="managing-users-and-ssh-keys" class="slide level2"> | ||
| 227 | <h1>Managing users and ssh keys</h1> | ||
| 228 | <pre><code> | ||
| 229 | define user::virtual::localuser( | ||
| 230 | $realname, | ||
| 231 | $groups = [ 'sudo', 'admin', ], | ||
| 232 | $sshkeys = '', | ||
| 233 | $key_id = '', | ||
| 234 | $old_keys = [], | ||
| 235 | $shell = '/bin/bash', | ||
| 236 | $home = "/home/${title}", | ||
| 237 | $managehome = true | ||
| 238 | ) { | ||
| 239 | |||
| 240 | group { $title: | ||
| 241 | ensure => present, | ||
| 242 | } | ||
| 243 | </code></pre> | ||
| 244 | </section> | ||
| 245 | |||
| 246 | <section id="managing-users-and-ssh-keys-2" class="slide level2"> | ||
| 247 | <h1>Managing users and ssh keys (cont.)</h1> | ||
| 248 | <pre><code> | ||
| 249 | user { $title: | ||
| 250 | ensure => present, | ||
| 251 | comment => $realname, | ||
| 252 | gid => $title, | ||
| 253 | groups => $groups, | ||
| 254 | home => $home, | ||
| 255 | managehome => $managehome, | ||
| 256 | membership => 'minimum', | ||
| 257 | shell => $shell, | ||
| 258 | require => Group[$title], | ||
| 259 | } | ||
| 260 | </code></pre> | ||
| 261 | </section> | ||
| 262 | |||
| 263 | <section id="managing-users-and-ssh-keys-3" class="slide level2"> | ||
| 264 | <h1>Managing users and ssh keys (cont.)</h1> | ||
| 265 | <pre><code> | ||
| 266 | ssh_authorized_key { $key_id: | ||
| 267 | ensure => present, | ||
| 268 | key => $sshkeys, | ||
| 269 | user => $title, | ||
| 270 | type => 'ssh-rsa', | ||
| 271 | } | ||
| 272 | |||
| 273 | if ( $old_keys != [] ) { | ||
| 274 | ssh_authorized_key { $old_keys: | ||
| 275 | ensure => absent, | ||
| 276 | user => $title, | ||
| 277 | } | ||
| 278 | } | ||
| 279 | } | ||
| 280 | </code></pre> | ||
| 281 | </section> | ||
| 282 | |||
| 283 | <section id="our-code-is-open-source-right-what-about-passwords-and-keys-..." class="slide level2"> | ||
| 284 | <h1>Our code is Open Source, right? What about passwords and keys ...</h1> | ||
| 285 | </section> | ||
| 286 | |||
| 287 | <section id="hiera" class="slide level2"> | ||
| 288 | <h1>hiera</h1> | ||
| 289 | <ul> | ||
| 290 | <li>Simple YAML database, sits on puppetmaster</li> | ||
| 291 | <li>Use it for secret data</li> | ||
| 292 | <li>puppet code is still complete</li> | ||
| 293 | </ul> | ||
| 294 | <pre><code> | ||
| 295 | node default { | ||
| 296 | class { 'openstack_project::server': | ||
| 297 | sysadmins => hiera('sysadmins', []), | ||
| 298 | } | ||
| 299 | } | ||
| 300 | </code></pre> | ||
| 301 | <p>Breaks ability to use simple puppet apply</p> | ||
| 302 | </section> | ||
| 303 | |||
| 304 | <section id="step-two-ansible-for-orchestration" class="titleslide slide level1"> | ||
| 305 | <h1>Step Two: Ansible for Orchestration</h1> | ||
| 306 | </section> | ||
| 307 | |||
| 308 | <section id="about-ansible" class="slide level2"> | ||
| 309 | <h1>About Ansible</h1> | ||
| 310 | <ul> | ||
| 311 | <li>Open Source System Management tool</li> | ||
| 312 | <li>Written in Python</li> | ||
| 313 | <li>Sequence of steps to perform</li> | ||
| 314 | <li>Works over SSH</li> | ||
| 315 | <li>Incremental Adoption</li> | ||
| 316 | </ul> | ||
| 317 | </section> | ||
| 318 | |||
| 319 | <section> | ||
| 320 | <h1>ad-hoc operation</h1> | ||
| 321 | <pre> | ||
| 322 | ansible '*' -m shell -p uptime | ||
| 323 | </pre> | ||
| 324 | </section> | ||
| 325 | |||
| 326 | <section id="yaml-syntax" class="slide level2"> | ||
| 327 | <h1>YAML Syntax</h1> | ||
| 328 | <pre><code> | ||
| 329 | - hosts: '*.slave.openstack.org' | ||
| 330 | tasks: | ||
| 331 | - shell: 'rm -rf ~jenkins/workspace/*{{ project }}*' | ||
| 332 | </code></pre> | ||
| 333 | <p>That's executed:</p> | ||
| 334 | <pre> | ||
| 335 | ansible-playbook -f 10 /etc/ansible/clean_workspaces.yaml --extra-vars "project=$PROJECTNAME" | ||
| 336 | </pre> | ||
| 337 | </section> | ||
| 338 | |||
| 339 | <section id="ansible-organization" class="slide level2"> | ||
| 340 | <h1>Ansible Organization</h1> | ||
| 341 | <ul> | ||
| 342 | <li>modules</li> | ||
| 343 | <li>plays</li> | ||
| 344 | <li>playbooks</li> | ||
| 345 | <li>roles</li> | ||
| 346 | </ul> | ||
| 347 | </section> | ||
| 348 | |||
| 349 | <section id="use-ansible-to-run-puppet" class="slide level2"> | ||
| 350 | <h1>Use Ansible to Run Puppet!</h1> | ||
| 351 | </section> | ||
| 352 | |||
| 353 | <section id="puppet-module" class="slide level2"> | ||
| 354 | <h1>puppet module</h1> | ||
| 355 | <pre><code>def main(): | ||
| 356 | module = AnsibleModule(argument_spec=dict( | ||
| 357 | timeout=dict(default="30m"), | ||
| 358 | puppetmaster=dict(required=True), | ||
| 359 | show_diff=dict(default=False, aliases=['show-diff'], type='bool'), | ||
| 360 | )) | ||
| 361 | p = module.params | ||
| 362 | |||
| 363 | puppet_cmd = module.get_bin_path("puppet", False) | ||
| 364 | if not puppet_cmd: | ||
| 365 | module.fail_json(msg="Could not find puppet. Please ensure it is installed.") | ||
| 366 | </code></pre> | ||
| 367 | </section> | ||
| 368 | |||
| 369 | <section id="puppet-module-2" class="slide level2"> | ||
| 370 | <h1>puppet module (cont)</h1> | ||
| 371 | <pre><code class="python"> | ||
| 372 | cmd = ("timeout -s 9 %(timeout)s %(puppet_cmd)s agent --onetime" | ||
| 373 | " --server %(puppetmaster)s" | ||
| 374 | " --ignorecache --no-daemonize --no-usecacheonfailure --no-splay" | ||
| 375 | " --detailed-exitcodes --verbose") % dict( | ||
| 376 | timeout=pipes.quote(p['timeout']), puppet_cmd=PUPPET_CMD, | ||
| 377 | puppetmaster=pipes.quote(p['puppetmaster'])) | ||
| 378 | if p['show_diff']: | ||
| 379 | cmd += " --show-diff" | ||
| 380 | rc, stdout, stderr = module.run_command(cmd) | ||
| 381 | </code></pre> | ||
| 382 | </section> | ||
| 383 | |||
| 384 | <section id="puppet-module-3" class="slide level2"> | ||
| 385 | Please. Everyone. Marvel at the following logic | ||
| 386 | <pre><code> | ||
| 387 | if rc == 0: # success | ||
| 388 | module.exit_json(rc=rc, changed=False, stdout=stdout) | ||
| 389 | elif rc == 1: | ||
| 390 | # rc==1 could be because it's disabled OR there was a compilation failure | ||
| 391 | disabled = "administratively disabled" in stdout | ||
| 392 | if disabled: | ||
| 393 | msg = "puppet is disabled" | ||
| 394 | else: | ||
| 395 | msg = "puppet compilation failed" | ||
| 396 | module.fail_json(rc=rc, disabled=disabled, msg=msg, stdout=stdout, stderr=stderr) | ||
| 397 | elif rc == 2: # success with changes | ||
| 398 | module.exit_json(changed=True) | ||
| 399 | elif rc == 124: # timeout | ||
| 400 | module.exit_json(rc=rc, msg="%s timed out" % cmd, stdout=stdout, stderr=stderr) | ||
| 401 | else: # failure | ||
| 402 | module.fail_json(rc=rc, msg="%s failed" % (cmd), stdout=stdout, stderr=stderr) | ||
| 403 | </code></pre> | ||
| 404 | </section> | ||
| 405 | |||
| 406 | <section id="puppet-play" class="slide level2"> | ||
| 407 | <h1>puppet play</h1> | ||
| 408 | <pre><code> | ||
| 409 | - name: run puppet | ||
| 410 | puppet: | ||
| 411 | puppetmaster: "{{puppetmaster}}" | ||
| 412 | </code></pre> | ||
| 413 | </section> | ||
| 414 | |||
| 415 | <section id="puppet-role" class="slide level2"> | ||
| 416 | <h1>puppet role</h1> | ||
| 417 | <p>roles/puppet/tasks/main.yml</p> | ||
| 418 | </section> | ||
| 419 | |||
| 420 | <section id="remote-puppet-playbook" class="slide level2"> | ||
| 421 | <h1>remote puppet playbook</h1> | ||
| 422 | <pre><code> | ||
| 423 | - hosts: git0* | ||
| 424 | gather_facts: false | ||
| 425 | max_fail_percentage: 1 | ||
| 426 | roles: | ||
| 427 | - { role: puppet, puppetmaster: puppetmaster.openstack.org } | ||
| 428 | - hosts: review.openstack.org | ||
| 429 | gather_facts: false | ||
| 430 | roles: | ||
| 431 | - { role: puppet, puppetmaster: puppetmaster.openstack.org } | ||
| 432 | - hosts: "!review.openstack.org:!git0*:!afs*" | ||
| 433 | gather_facts: false | ||
| 434 | roles: | ||
| 435 | - { role: puppet, puppetmaster: puppetmaster.openstack.org } | ||
| 436 | </pre></code> | ||
| 437 | </section> | ||
| 438 | |||
| 439 | <section id="ansible-inventory" class="slide level2"> | ||
| 440 | <h1>ansible inventory</h1> | ||
| 441 | <ul> | ||
| 442 | <li>List of servers to operate on</li> | ||
| 443 | <li>Optionally variables associated with each server</li> | ||
| 444 | <li>Optional groups of servers</li> | ||
| 445 | <li>Simple yaml file in /etc/ansible/hosts</li> | ||
| 446 | <li>Dynamic executable that returns JSON</li> | ||
| 447 | </ul> | ||
| 448 | </section> | ||
| 449 | |||
| 450 | <section id="ansible-inventory-from-file" class="slide level2"> | ||
| 451 | <h1>Simple inventory</h1> | ||
| 452 | <pre> | ||
| 453 | review.openstack.org | ||
| 454 | git01.openstack.org | ||
| 455 | git02.openstack.org | ||
| 456 | pypi.dfw.openstack.org | ||
| 457 | pypi.iad.openstack.org | ||
| 458 | |||
| 459 | [pypi] | ||
| 460 | pypi.dfw.openstack.org | ||
| 461 | pypi.iad.openstack.org | ||
| 462 | |||
| 463 | [git] | ||
| 464 | git01.openstack.org | ||
| 465 | git02.openstack.org | ||
| 466 | </pre> | ||
| 467 | </section> | ||
| 468 | |||
| 469 | <section id="ansible-inventory-from-puppet" class="slide level2"> | ||
| 470 | <h1>ansible inventory from puppet certs</h1> | ||
| 471 | <pre><code> | ||
| 472 | import json | ||
| 473 | import subprocess | ||
| 474 | |||
| 475 | output = [ | ||
| 476 | x.split()[1][1:-1] for x in subprocess.check_output( | ||
| 477 | ["puppet","cert","list","-a"]).split('\n') | ||
| 478 | if x.startswith('+') | ||
| 479 | ] | ||
| 480 | |||
| 481 | data = { | ||
| 482 | '_meta': {'hostvars': dict()}, | ||
| 483 | 'ungrouped': output, | ||
| 484 | } | ||
| 485 | print json.dumps(data, sort_keys=True, indent=2) | ||
| 486 | </code></pre> | ||
| 487 | </section> | ||
| 488 | |||
| 489 | <section> | ||
| 490 | <h1>Step Three: Ansible for Cloud Management</h1> | ||
| 491 | </section> | ||
| 492 | |||
| 493 | <section> | ||
| 494 | <h1>ansible and OpenStack</h1> | ||
| 495 | <ul> | ||
| 496 | <li>Ansible modules are just python</li> | ||
| 497 | <li>playbooks are lists of steps to take</li> | ||
| 498 | <li>Have plays/roles that provision servers</li> | ||
| 499 | <li>Infrastructure as code - for real!</li> | ||
| 500 | </ul> | ||
| 501 | </section> | ||
| 502 | |||
| 503 | <section> | ||
| 504 | <h1>Consider this data</h1> | ||
| 505 | <pre><code> | ||
| 506 | pypi.dfw.openstack.org: | ||
| 507 | image_name: Ubuntu 12.04.4 | ||
| 508 | flavor_ram: 2048 | ||
| 509 | region: DFW | ||
| 510 | cloud: rackspace | ||
| 511 | volumes: | ||
| 512 | - size: 200 | ||
| 513 | mount: /srv | ||
| 514 | pypi.region-b.geo-1.openstack.org: | ||
| 515 | image_name: Ubuntu 12.04.4 | ||
| 516 | flavor_ram: 2048 | ||
| 517 | region: region-b.geo-1 | ||
| 518 | cloud: hp | ||
| 519 | volumes: | ||
| 520 | - size: 200 | ||
| 521 | mount: /srv | ||
| 522 | </code></pre> | ||
| 523 | </section> | ||
| 524 | |||
| 525 | <section> | ||
| 526 | <h1>Further consider this data</h1> | ||
| 527 | <pre><code> | ||
| 528 | pypi: | ||
| 529 | image_name: Ubuntu 12.04.4 | ||
| 530 | flavor_ram: 2048 | ||
| 531 | volumes: | ||
| 532 | - size: 200 | ||
| 533 | mount: /srv | ||
| 534 | hosts: | ||
| 535 | pypi.dfw: | ||
| 536 | region: DFW | ||
| 537 | pypi.iad: | ||
| 538 | region: IAD | ||
| 539 | pypi.ord: | ||
| 540 | region: ORD | ||
| 541 | pypi.region-b.geo-1: | ||
| 542 | cloud: hp | ||
| 543 | </code></pre> | ||
| 544 | </section> | ||
| 545 | |||
| 546 | <section> | ||
| 547 | <h1>Steps to launch a node</h1> | ||
| 548 | <ol> | ||
| 549 | <li>Create a compute instance</li> | ||
| 550 | <li>Wait for instance to exist</li> | ||
| 551 | <li>Create a floating IP</li> | ||
| 552 | <li>Attach floating IP to instance</li> | ||
| 553 | <li>Create one or more volumes</li> | ||
| 554 | <li>Attach volumes to instance</li> | ||
| 555 | <li>Wait for SSH to work</li> | ||
| 556 | <li>On host, format each volume</li> | ||
| 557 | <li>On host, mount each volume</li> | ||
| 558 | <li>On host, install config management software</li> | ||
| 559 | <li>On host, run config management software</li> | ||
| 560 | </ol> | ||
| 561 | </section> | ||
| 562 | |||
| 563 | <section> | ||
| 564 | <h1>Launch a node</h1> | ||
| 565 | <pre><code> | ||
| 566 | --- | ||
| 567 | - name: Launch Node | ||
| 568 | os_compute: | ||
| 569 | cloud: "{{ cloud }}" | ||
| 570 | region_name: "{{ region_name }}" | ||
| 571 | name: "{{ name }}" | ||
| 572 | image_name: "{{ image_name }}" | ||
| 573 | flavor_ram: "{{ flavor_ram }}" | ||
| 574 | flavor_include: "{{ flavor_include }}" | ||
| 575 | meta: | ||
| 576 | group: "{{ group }}" | ||
| 577 | key_name: "{{ launch_keypair }}" | ||
| 578 | register: node | ||
| 579 | - name: Create volumes | ||
| 580 | os_volume: | ||
| 581 | cloud: "{{ cloud }}" | ||
| 582 | size: "{{ item.size }}" | ||
| 583 | display_name: "{{ item.display_name }}" | ||
| 584 | with_items: volumes | ||
| 585 | - name: Attach volumes | ||
| 586 | os_compute_volume: | ||
| 587 | cloud: "{{ cloud }}" | ||
| 588 | server_id: "{{ node.id }}" | ||
| 589 | volume_name: "{{ item.display_name }}" | ||
| 590 | with_items: volumes | ||
| 591 | register: attached_volumes | ||
| 592 | - debug: var=attached_volumes | ||
| 593 | - name: Re-request server to get up to date metadata after the volume loop | ||
| 594 | os_compute_facts: | ||
| 595 | cloud: "{{ cloud }}" | ||
| 596 | name: "{{ name }}" | ||
| 597 | when: attached_volumes.changed | ||
| 598 | - name: Wait for SSH to work | ||
| 599 | wait_for: host={{ node.openstack.interface_ip }} port=22 | ||
| 600 | when: node.changed == True | ||
| 601 | - name: Add SSH host key to known hosts | ||
| 602 | shell: ssh-keyscan "{{ node.openstack.interface_ip|quote }}" >> ~/.ssh/known_hosts | ||
| 603 | when: node.changed == True | ||
| 604 | - name: Add all instance public IPs to host group | ||
| 605 | add_host: | ||
| 606 | name: "{{ node.openstack.interface_ip }}" | ||
| 607 | groups: "{{ provision_group }}" | ||
| 608 | openstack: "{{ node.openstack }}" | ||
| 609 | when: attached_volumes|length == 0 | ||
| 610 | - name: Add all instance public IPs to host and volumes group | ||
| 611 | add_host: | ||
| 612 | name: "{{ node.openstack.interface_ip }}" | ||
| 613 | groups: "{{ provision_group }},hasvolumes" | ||
| 614 | openstack: "{{ node.openstack }}" | ||
| 615 | when: attached_volumes|length != 0 | ||
| 616 | </code></pre> | ||
| 617 | </section> | ||
| 618 | |||
| 619 | <section> | ||
| 620 | <h1> Cloud based inventory </h1> | ||
| 621 | <ul> | ||
| 622 | <li> Just ask the cloud for the inventory </li> | ||
| 623 | <li> All of the meta-data the cloud knows is available </li> | ||
| 624 | <li> No need for puppetmaster now </li> | ||
| 625 | </ul> | ||
| 626 | </section> | ||
| 627 | |||
| 628 | <section> | ||
| 629 | <pre><code> | ||
| 630 | "pypi.dfw.openstack.org": { | ||
| 631 | "ansible_ssh_host": "23.253.237.8", | ||
| 632 | "openstack": { | ||
| 633 | "HUMAN_ID": true, | ||
| 634 | "NAME_ATTR": "name", | ||
| 635 | "OS-DCF:diskConfig": "MANUAL", | ||
| 636 | "OS-EXT-STS:power_state": 1, | ||
| 637 | "OS-EXT-STS:task_state": null, | ||
| 638 | "OS-EXT-STS:vm_state": "active", | ||
| 639 | "accessIPv4": "23.253.237.8", | ||
| 640 | "accessIPv6": "2001:4800:7817:104:d256:7a33:5187:7e1b", | ||
| 641 | "addresses": { | ||
| 642 | "private": [ | ||
| 643 | { | ||
| 644 | "addr": "10.208.195.50", | ||
| 645 | "version": 4 | ||
| 646 | } | ||
| 647 | ], | ||
| 648 | "public": [ | ||
| 649 | { | ||
| 650 | "addr": "23.253.237.8", | ||
| 651 | "version": 4 | ||
| 652 | }, | ||
| 653 | { | ||
| 654 | "addr": "2001:4800:7817:104:d256:7a33:5187:7e1b", | ||
| 655 | "version": 6 | ||
| 656 | } | ||
| 657 | ] | ||
| 658 | }, | ||
| 659 | "cloud": "rax", | ||
| 660 | "config_drive": "", | ||
| 661 | "created": "2014-09-05T15:32:14Z", | ||
| 662 | "flavor": { | ||
| 663 | "id": "performance1-4", | ||
| 664 | "links": [ | ||
| 665 | { | ||
| 666 | "href": "https://dfw.servers.api.rackspacecloud.com/610275/flavors/performance1-4", | ||
| 667 | "rel": "bookmark" | ||
| 668 | } | ||
| 669 | ], | ||
| 670 | "name": "4 GB Performance" | ||
| 671 | }, | ||
| 672 | "hostId": "adb603d4566efe0392756c76dab38ffcba22099368837c7973321e77", | ||
| 673 | "human_id": "pypidfwopenstackorg", | ||
| 674 | "id": "de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 675 | "image": { | ||
| 676 | "id": "928e709d-35f0-47eb-b296-d18e1b0a76b7", | ||
| 677 | "links": [ | ||
| 678 | { | ||
| 679 | "href": "https://dfw.servers.api.rackspacecloud.com/610275/images/928e709d-35f0-47eb-b296-d18e1b0a76b7", | ||
| 680 | "rel": "bookmark" | ||
| 681 | } | ||
| 682 | ] | ||
| 683 | }, | ||
| 684 | "interface_ip": "23.253.237.8", | ||
| 685 | "key_name": "launch-node-root", | ||
| 686 | "links": [ | ||
| 687 | { | ||
| 688 | "href": "https://dfw.servers.api.rackspacecloud.com/v2/610275/servers/de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 689 | "rel": "self" | ||
| 690 | }, | ||
| 691 | { | ||
| 692 | "href": "https://dfw.servers.api.rackspacecloud.com/610275/servers/de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 693 | "rel": "bookmark" | ||
| 694 | } | ||
| 695 | ], | ||
| 696 | "metadata": {}, | ||
| 697 | "name": "pypi.dfw.openstack.org", | ||
| 698 | "networks": { | ||
| 699 | "private": [ | ||
| 700 | "10.208.195.50" | ||
| 701 | ], | ||
| 702 | "public": [ | ||
| 703 | "23.253.237.8", | ||
| 704 | "2001:4800:7817:104:d256:7a33:5187:7e1b" | ||
| 705 | ] | ||
| 706 | }, | ||
| 707 | "progress": 100, | ||
| 708 | "region": "DFW", | ||
| 709 | "status": "ACTIVE", | ||
| 710 | "tenant_id": "610275", | ||
| 711 | "updated": "2014-09-05T15:32:49Z", | ||
| 712 | "user_id": "156284", | ||
| 713 | "volumes": [ | ||
| 714 | { | ||
| 715 | "HUMAN_ID": false, | ||
| 716 | "NAME_ATTR": "name", | ||
| 717 | "attachments": [ | ||
| 718 | { | ||
| 719 | "device": "/dev/xvdb", | ||
| 720 | "host_name": null, | ||
| 721 | "id": "c6f5229c-1cc0-47c4-aab7-60db1f6cf8e8", | ||
| 722 | "server_id": "de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 723 | "volume_id": "c6f5229c-1cc0-47c4-aab7-60db1f6cf8e8" | ||
| 724 | } | ||
| 725 | ], | ||
| 726 | "availability_zone": "nova", | ||
| 727 | "bootable": "false", | ||
| 728 | "created_at": "2014-09-05T14:37:42.000000", | ||
| 729 | "device": "/dev/xvdb", | ||
| 730 | "display_description": null, | ||
| 731 | "display_name": "pypi.dfw.openstack.org/main01", | ||
| 732 | "human_id": null, | ||
| 733 | "id": "c6f5229c-1cc0-47c4-aab7-60db1f6cf8e8", | ||
| 734 | "metadata": { | ||
| 735 | "readonly": "False", | ||
| 736 | "storage-node": "1845027a-5e07-47a1-9572-3eea4716f726" | ||
| 737 | }, | ||
| 738 | "os-vol-tenant-attr:tenant_id": "610275", | ||
| 739 | "size": 200, | ||
| 740 | "snapshot_id": null, | ||
| 741 | "source_volid": null, | ||
| 742 | "status": "in-use", | ||
| 743 | "volume_type": "SATA" | ||
| 744 | } | ||
| 745 | ] | ||
| 746 | } | ||
| 747 | }, | ||
| 748 | </code></pre> | ||
| 749 | </section> | ||
| 750 | |||
| 751 | <section> | ||
| 752 | <h1> Wait - what about secrets? </h1> | ||
| 753 | <p class="fragment">ansible can just pass secrets to puppet apply as parameters </p> | ||
| 754 | </section> | ||
| 755 | |||
| 756 | <section> | ||
| 757 | <h1> Step Four: Just get rid of puppet ... </h1> | ||
| 758 | <p class="fragment">but that's another talk</p> | ||
| 759 | </section> | ||
| 760 | |||
| 761 | <section> | ||
| 762 | <h1> Thank you! </h1> | ||
| 763 | <p> http://inaugust.com/talks/ansible-cloud.html </p> | ||
| 764 | <h3>Monty Taylor</h3> | ||
| 765 | <h3> twitter: @e_monty </h3> | ||
| 766 | <h4> <a href='http://inaugust.com/talks/ansible-cloud.html'>http://inaugust.com/talks/ansible-cloud.html</a> </h4> | ||
| 767 | </section> | ||
| 768 | |||
| 769 | </div> | ||
| 770 | <div class="footer"> | ||
| 771 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 772 | <img alt="Creative Commons License" | ||
| 773 | style="border-width:0" | ||
| 774 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 775 | </a><br /> | ||
| 776 | Licensed under a | ||
| 777 | <a rel="license" | ||
| 778 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 779 | Creative Commons Attribution 4.0 International License | ||
| 780 | </a>. | ||
| 781 | <br /> | ||
| 782 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 783 | </div> | ||
| 784 | </div> | ||
| 785 | |||
| 786 | <script src="/lib/js/head.min.js"></script> | ||
| 787 | <script src="/js/reveal.js"></script> | ||
| 788 | |||
| 789 | <script src="/js/this.js"></script> | ||
| 790 | |||
| 791 | </body> | ||
| 792 | </html> | ||
diff --git a/src/talks/glean/index.html b/src/talks/glean/index.html new file mode 100644 index 0000000..bbf943a --- /dev/null +++ b/src/talks/glean/index.html | |||
| @@ -0,0 +1,209 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>glean: a minimal non-cloud-init cloud-init</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = '/css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | </head> | ||
| 32 | <body> | ||
| 33 | |||
| 34 | <div class="background"></div> | ||
| 35 | |||
| 36 | <div class="reveal"> | ||
| 37 | <div class="slides"> | ||
| 38 | |||
| 39 | <section data-state="cover"> | ||
| 40 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 41 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 42 | property="dct:title" | ||
| 43 | rel="dct:type"> | ||
| 44 | glean: a minimal non-cloud-init cloud-init | ||
| 45 | </span></h1> | ||
| 46 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 47 | property="cc:attributionName">Monty Taylor</h3> | ||
| 48 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 49 | rel="cc:attributionURL" | ||
| 50 | href='http://inaugust.com/talks/os-client-config.html'>http://inaugust.com/talks/glean.html</a> </h4> | ||
| 51 | <h3> twitter: @e_monty </h3> | ||
| 52 | </section> | ||
| 53 | |||
| 54 | <section> | ||
| 55 | <h1>When you boot a VM, it needs boot time data or bootstrapping</h1> | ||
| 56 | </section> | ||
| 57 | |||
| 58 | <section> | ||
| 59 | <h1>I use ansible for orchestration</h1> | ||
| 60 | </section> | ||
| 61 | |||
| 62 | <section> | ||
| 63 | <h1>I don't need boot time fancy, | ||
| 64 | I <em>JUST</em> need ssh bootstrap</h1> | ||
| 65 | <p class='fragment'>Ok. And maybe network</p> | ||
| 66 | </section> | ||
| 67 | |||
| 68 | <section> | ||
| 69 | <h1>Things I must consume at boot</h1> | ||
| 70 | <ul> | ||
| 71 | <li class='fragment'> | ||
| 72 | Network Configuration | ||
| 73 | </li> | ||
| 74 | <li class='fragment'> | ||
| 75 | <em>maybe</em> SSH Keys | ||
| 76 | </li> | ||
| 77 | </ul> | ||
| 78 | </section> | ||
| 79 | |||
| 80 | <section class="slide level2"> | ||
| 81 | <h1>Network information should be easy</h1> | ||
| 82 | <p>There is this thing called DHCP</p> | ||
| 83 | <p>Dynamic Host Configuration Protocol</p> | ||
| 84 | </section> | ||
| 85 | |||
| 86 | <section class="slide level2"> | ||
| 87 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 88 | </section> | ||
| 89 | |||
| 90 | <section class="slide level2"> | ||
| 91 | <h1>Some providers don't support DHCP - because bong</h1> | ||
| 92 | </section> | ||
| 93 | |||
| 94 | <section class="slide level2"> | ||
| 95 | <h1>cloud-init</h1> | ||
| 96 | <p>Handles many cases, but not all</p> | ||
| 97 | <p>Has lots of dependencies that conflict with gate depends</p> | ||
| 98 | <p>'frozen' pending 2.0 rewrite</p> | ||
| 99 | </section> | ||
| 100 | |||
| 101 | <section class="slide level2"> | ||
| 102 | <h1>glean</h1> | ||
| 103 | <h3>http://git.openstack.org/cgit/openstack-infra/glean</h3> | ||
| 104 | <ul> | ||
| 105 | <p>No dependencies</p> | ||
| 106 | <p>Handles future static network config in config drive</p> | ||
| 107 | <p>Optionally also reads ssh keys from config drive</p> | ||
| 108 | <p>Nothing else</p> | ||
| 109 | </ul> | ||
| 110 | </section> | ||
| 111 | |||
| 112 | <section> | ||
| 113 | <h1>network_info.json</h1> | ||
| 114 | <p>Hopefully added in liberty</p> | ||
| 115 | <pre> | ||
| 116 | services: | ||
| 117 | - type: dns | ||
| 118 | address: 72.3.128.241 | ||
| 119 | - type: dns | ||
| 120 | address: 72.3.128.240 | ||
| 121 | networks: | ||
| 122 | - network_id: 00000000-0000-0000-0000-000000000000 | ||
| 123 | type: ipv4 | ||
| 124 | netmask: 255.255.255.0 | ||
| 125 | link: tapfafb5c05-a6 | ||
| 126 | routes: | ||
| 127 | - netmask: 0.0.0.0 | ||
| 128 | network: 0.0.0.0 | ||
| 129 | gateway: 23.253.229.1 | ||
| 130 | ip_address: 23.253.229.154 | ||
| 131 | id: network0 | ||
| 132 | - network_id: 11111111-1111-1111-1111-111111111111 | ||
| 133 | type: ipv4 | ||
| 134 | netmask: 255.255.224.0 | ||
| 135 | link: tape501e1cd-10 | ||
| 136 | routes: | ||
| 137 | - netmask: 255.240.0.0 | ||
| 138 | network: 10.176.0.0 | ||
| 139 | gateway: 10.208.160.1 | ||
| 140 | - netmask: 255.240.0.0 | ||
| 141 | network: 10.208.0.0 | ||
| 142 | gateway: 10.208.160.1 | ||
| 143 | ip_address: 10.208.169.118 | ||
| 144 | id: network1 | ||
| 145 | links: | ||
| 146 | - ethernet_mac_address: BC:76:4E:01:62:86 | ||
| 147 | mtu: 1500 | ||
| 148 | type: null | ||
| 149 | id: tapfafb5c05-a6 | ||
| 150 | vif_id: fafb5c05-a661-48ae-9810-46601c7e22d1 | ||
| 151 | - ethernet_mac_address: BC:76:4E:05:7B:06 | ||
| 152 | mtu: 1500 | ||
| 153 | type: null | ||
| 154 | id: tape501e1cd-10 | ||
| 155 | vif_id: e501e1cd-10d0-4e63-b0c2-6542989ccbb2 | ||
| 156 | </pre> | ||
| 157 | </section> | ||
| 158 | |||
| 159 | <section class="slide level2"> | ||
| 160 | <h1>Integrated with diskimage-builder</h1> | ||
| 161 | <pre> | ||
| 162 | disk-image-create -o debian.qcow2 debian-minimal vm simple-init | ||
| 163 | </pre> | ||
| 164 | </section> | ||
| 165 | |||
| 166 | <section class="slide level2"> | ||
| 167 | <h1>You know what has less depends than minimal python?</h1> | ||
| 168 | </section> | ||
| 169 | |||
| 170 | <section class="slide level2"> | ||
| 171 | <h1>Rust</h1> | ||
| 172 | </section> | ||
| 173 | |||
| 174 | <section class="slide level2"> | ||
| 175 | <h1>glean.rs</h1> | ||
| 176 | <h3>https://gitlab.com/mordred/glean.rs</h3> | ||
| 177 | </section> | ||
| 178 | |||
| 179 | <section> | ||
| 180 | <h1> Thank you! </h1> | ||
| 181 | <h4> <a href='http://inaugust.com/talks/glean.html'>http://inaugust.com/talks/glean.html</a> </h4> | ||
| 182 | <h3> twitter: @e_monty </h3> | ||
| 183 | </section> | ||
| 184 | |||
| 185 | </div> | ||
| 186 | |||
| 187 | <div class="footer"> | ||
| 188 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 189 | <img alt="Creative Commons License" | ||
| 190 | style="border-width:0" | ||
| 191 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 192 | </a><br /> | ||
| 193 | Licensed under a | ||
| 194 | <a rel="license" | ||
| 195 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 196 | Creative Commons Attribution 4.0 International License | ||
| 197 | </a>. | ||
| 198 | <br /> | ||
| 199 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 200 | </div> | ||
| 201 | </div> | ||
| 202 | |||
| 203 | <script src="/lib/js/head.min.js"></script> | ||
| 204 | <script src="/js/reveal.js"></script> | ||
| 205 | |||
| 206 | <script src="/js/this.js"></script> | ||
| 207 | |||
| 208 | </body> | ||
| 209 | </html> | ||
diff --git a/src/talks/index/index.html b/src/talks/index/index.html new file mode 100644 index 0000000..47b9774 --- /dev/null +++ b/src/talks/index/index.html | |||
| @@ -0,0 +1,74 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>Monty Taylor Conference Talks</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = 'css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | |||
| 32 | <script src="js/jquery-1.7.2.min.js" charset="utf-8" type="text/javascript"></script> | ||
| 33 | <script src="js/raphael-min.js" type="text/javascript" charset="utf-8"></script> | ||
| 34 | |||
| 35 | </head> | ||
| 36 | <body> | ||
| 37 | |||
| 38 | <div class="reveal"> | ||
| 39 | <div class="slides"> | ||
| 40 | <section data-state="cover"> | ||
| 41 | <h1> Monty Taylor Conference Talks </h1> | ||
| 42 | |||
| 43 | <p><a href='http://inaugust.com/talks/os-client-config.html'>os-client-config lightning talk</a></p> | ||
| 44 | <p><a href='http://inaugust.com/talks/ansible-cloud.html'>Using Ansible to Manage Cloud Applications</a></p> | ||
| 45 | <p><a href='http://inaugust.com/talks/tripleo-ansible.html'>Deploying OpenStack using TripleO and Ansible</a></p> | ||
| 46 | <p><a href='http://inaugust.com/talks/lemmings.html'>Lemmings, Think Different and Why Free Software is Important</a></p> | ||
| 47 | <p><a href='http://inaugust.com/talks/product-management.html'>Liberty, Product Management and OpenStack Technology</a></p> | ||
| 48 | <p><a href='http://inaugust.com/talks/now-what.html'>OpenStack works ... so now what?</a></p> | ||
| 49 | <p><a href='http://inaugust.com/talks/glean.html'>glean: a minimal non-cloud-init cloud-init</a></p> | ||
| 50 | <p><a href='http://inaugust.com/talks/a-vision-for-the-future.html'>OpenStack: a vision for the future</a> | ||
| 51 | |||
| 52 | </section> | ||
| 53 | </div> | ||
| 54 | <div class="footer"> | ||
| 55 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 56 | <img alt="Creative Commons License" | ||
| 57 | style="border-width:0" | ||
| 58 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 59 | </a><br /> | ||
| 60 | Licensed under a | ||
| 61 | <a rel="license" | ||
| 62 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 63 | Creative Commons Attribution 4.0 International License | ||
| 64 | </a>. | ||
| 65 | <br /> | ||
| 66 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 67 | </div> | ||
| 68 | </div> | ||
| 69 | <script src="lib/js/head.min.js"></script> | ||
| 70 | <script src="js/reveal.js"></script> | ||
| 71 | |||
| 72 | <script src="js/this.js"></script> | ||
| 73 | |||
| 74 | </html> | ||
diff --git a/src/talks/lemmings/index.html b/src/talks/lemmings/index.html new file mode 100644 index 0000000..bbaa982 --- /dev/null +++ b/src/talks/lemmings/index.html | |||
| @@ -0,0 +1,331 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>Lemmings, Think Different and Why Free Software is Important</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = '/css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | </head> | ||
| 32 | <body> | ||
| 33 | |||
| 34 | <div class="reveal"><div class="slides"> | ||
| 35 | |||
| 36 | <section data-state="cover"> | ||
| 37 | |||
| 38 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 39 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 40 | property="dct:title" | ||
| 41 | rel="dct:type"> | ||
| 42 | Lemmings, Think Different and Why Free Software is Important | ||
| 43 | </span></h1> | ||
| 44 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 45 | property="cc:attributionName">Monty Taylor</h3> | ||
| 46 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 47 | rel="cc:attributionURL" | ||
| 48 | href='http://inaugust.com/talks/lemmings.html'>http://inaugust.com/talks/lemmings.html</a> </h4> | ||
| 49 | <h3> twitter: @e_monty </h3> | ||
| 50 | </section> | ||
| 51 | |||
| 52 | <section class="slide level2"> | ||
| 53 | <img src="/images/openstack-cloud-software-vertical-large.png" | ||
| 54 | alt="OpenStack" /> | ||
| 55 | </section> | ||
| 56 | |||
| 57 | <section class="slide level2"> | ||
| 58 | <img src="/images/openstack-software-diagram.png" alt="OpenStack" /> | ||
| 59 | </section> | ||
| 60 | |||
| 61 | <section class="slide level2"> | ||
| 62 | <h1>Mission</h1> | ||
| 63 | <p> | ||
| 64 | to produce the ubiquitous Open Source Cloud Computing platform that will meet the needs of public and private clouds regardless of size, by being simple to implement and massively scalable. | ||
| 65 | </p> | ||
| 66 | </section> | ||
| 67 | |||
| 68 | <section class="slide level2"> | ||
| 69 | <img src="/images/openstack-size.jpg" /> | ||
| 70 | </section> | ||
| 71 | |||
| 72 | <section class="slide level2"> | ||
| 73 | <h1>Borrowed from Ubuntu</h1> | ||
| 74 | <p>Time Based Releases</p> | ||
| 75 | <p>Six Month Cadance</p> | ||
| 76 | <p>Design summits each cycle</p> | ||
| 77 | <p>Release codenames in alphabetical order</p> | ||
| 78 | </section> | ||
| 79 | |||
| 80 | <section class="slide level2"> | ||
| 81 | <h1>Different from Ubuntu</h1> | ||
| 82 | <p>No BDFL</p> | ||
| 83 | <p>All decisions are democratic</p> | ||
| 84 | </section> | ||
| 85 | |||
| 86 | <section class="slide level2"> | ||
| 87 | <h1>Naming Process</h1> | ||
| 88 | Pick a name from names of cities or geographical features near the | ||
| 89 | summit location | ||
| 90 | </section> | ||
| 91 | |||
| 92 | <section class="slide level2"> | ||
| 93 | <h1>Grizzly</h1> | ||
| 94 | <p class="fragment">San Diego</p> | ||
| 95 | <p class="fragment">Grizzly bear on the flag</p> | ||
| 96 | </section> | ||
| 97 | |||
| 98 | <section class="slide level2"> | ||
| 99 | <h1>Havana</h1> | ||
| 100 | <p class="fragment">Portland, OR</p> | ||
| 101 | <p class="fragment">not Cuba</p> | ||
| 102 | </section> | ||
| 103 | |||
| 104 | <section class="slide level2"> | ||
| 105 | <h1>Icehouse</h1> | ||
| 106 | <p class="fragment">Hong Kong</p> | ||
| 107 | <p class="fragment">Street in HK</p> | ||
| 108 | </section> | ||
| 109 | |||
| 110 | <section class="slide level2"> | ||
| 111 | <h1>Juno</h1> | ||
| 112 | <p class="fragment">Atlanta, GA</p> | ||
| 113 | <p class="fragment">Town in Georgia</p> | ||
| 114 | </section> | ||
| 115 | |||
| 116 | <section class="slide level2"> | ||
| 117 | <h1>Kilo</h1> | ||
| 118 | <p class="fragment">Paris</p> | ||
| 119 | <p class="fragment">location of the actual kilogramme</p> | ||
| 120 | </section> | ||
| 121 | |||
| 122 | <section class="slide level2"> | ||
| 123 | <h1>The L Word</h1> | ||
| 124 | <p class="fragment">Vancouver</p> | ||
| 125 | <p class="fragment">Lemming?</p> | ||
| 126 | <p class="fragment">Lemming Peak</p> | ||
| 127 | </section> | ||
| 128 | |||
| 129 | <section class="slide level2"> | ||
| 130 | <h1>The Marketing Community was not thrilled</h1> | ||
| 131 | </section> | ||
| 132 | |||
| 133 | <section class="slide level2"> | ||
| 134 | <h1>Lemmings commit mass suicide by following each other off cliffs</h1> | ||
| 135 | </section> | ||
| 136 | |||
| 137 | <section class="slide level2"> | ||
| 138 | <h1>Except they don't</h1> | ||
| 139 | </section> | ||
| 140 | |||
| 141 | <section class="slide level2"> | ||
| 142 | <h1>White Wilderness</h1> | ||
| 143 | <p>1958 Disney Documentary</p> | ||
| 144 | <p>Staged footage</p> | ||
| 145 | <p class="fragment">Camera crew forced them off the cliff</p> | ||
| 146 | <p class="fragment">Won an Oscar, of course</p> | ||
| 147 | </section> | ||
| 148 | |||
| 149 | <section class="slide level2"> | ||
| 150 | <h1>Definition of Irony</h1> | ||
| 151 | <p>The actual meaning is the opposite of the literal meaning</p> | ||
| 152 | </section> | ||
| 153 | |||
| 154 | <section class="slide level2"> | ||
| 155 | <h1>Recursive Definition of Irony</h1> | ||
| 156 | <p>The lemmings metaphor only works because people are lemmings in their adherence to the myth that lemmings are like lemmings.</p> | ||
| 157 | </section> | ||
| 158 | |||
| 159 | <section class="slide level2"> | ||
| 160 | <h1>This could never happen to us, right?</h1> | ||
| 161 | <p class="fragment">MySQL doesn't have transactions</p> | ||
| 162 | <p class="fragment">MongoDB is Web Scale</p> | ||
| 163 | <p class="fragment">PHP is insecure</p> | ||
| 164 | <p class="fragment">Frequent password rotation is secure</p> | ||
| 165 | <p class="fragment">sysvinit needs to be replaced</p> | ||
| 166 | </section> | ||
| 167 | |||
| 168 | <section class="slide level2"> | ||
| 169 | <h1>Results</h1> | ||
| 170 | <p>We didn't pick Lemming - we picked Liberty</p> | ||
| 171 | <p class="fragment">Liberty is a town in Saskatchewan</p> | ||
| 172 | </section> | ||
| 173 | |||
| 174 | <section class="slide level2"> | ||
| 175 | <h1>Four Freedoms</h1> | ||
| 176 | <p>0 - to run the program as you wish, for any purpose</p> | ||
| 177 | <p>1 - to study how the program works, and change it so it does your computing as you wish</p> | ||
| 178 | <p>2 - to redistribute copies so you can help your neighbor</p> | ||
| 179 | <p>3 - to distribute copies of your modified version to others</p> | ||
| 180 | </section> | ||
| 181 | |||
| 182 | <section class="slide level2"> | ||
| 183 | <h1> GPL is anti-business </h1> | ||
| 184 | </section> | ||
| 185 | |||
| 186 | <section class="slide level2"> | ||
| 187 | <h1>Except it isn't</h1> | ||
| 188 | </section> | ||
| 189 | |||
| 190 | <section class="slide level2"> | ||
| 191 | <h1>Definition of Eating Crow</h1> | ||
| 192 | <p>humiliation by admitting wrongness or having been proven wrong after taking a strong position</p> | ||
| 193 | </section> | ||
| 194 | |||
| 195 | <section class="slide level2"> | ||
| 196 | <h1>Enjoyable Definition of Eating Crow</h1> | ||
| 197 | <p>Microsoft purchases Revolution Analytics</p> | ||
| 198 | <p>R language is GPL</p> | ||
| 199 | </section> | ||
| 200 | |||
| 201 | <section class="slide level2"> | ||
| 202 | <h1>Sun/MySQL</h1> | ||
| 203 | <p>Sun bought MySQL for $1 Billion</p> | ||
| 204 | <p>Oracle bought Sun</p> | ||
| 205 | <p>Oracle now develops and ships MySQL under the GPL</p> | ||
| 206 | </section> | ||
| 207 | |||
| 208 | <section class="slide level2"> | ||
| 209 | <h1>Linux</h1> | ||
| 210 | <p class="fragment">anybody here heard of it?</p> | ||
| 211 | </section> | ||
| 212 | |||
| 213 | <section class="slide level2"> | ||
| 214 | <h1>Android is the market leading Mobile OS</p> | ||
| 215 | </section> | ||
| 216 | |||
| 217 | <section class="slide level2"> | ||
| 218 | <h1>Linux runs:</h1> | ||
| 219 | <p>Google</p> | ||
| 220 | <p>Facebook</p> | ||
| 221 | <p>Twitter</p> | ||
| 222 | <p class="fragment">They run MySQL too</p> | ||
| 223 | </section> | ||
| 224 | |||
| 225 | <section class="slide level2"> | ||
| 226 | <h1>Linux is not just for big companies - it's startup friendly too!</h1> | ||
| 227 | </section> | ||
| 228 | |||
| 229 | <section class="slide level2"> | ||
| 230 | <img src="/images/hp-logo.png" /> | ||
| 231 | </section> | ||
| 232 | |||
| 233 | <section class="slide level2"> | ||
| 234 | <h1>HP ships a Linux server every minute</h1> | ||
| 235 | <p class="fragment">business unit that does is a $28B business</p> | ||
| 236 | </section> | ||
| 237 | |||
| 238 | <section class="slide level2"> | ||
| 239 | <h1>Why is Free Software Important?</h1> | ||
| 240 | </section> | ||
| 241 | |||
| 242 | <section class="slide level2"> | ||
| 243 | <p>1984 - Apple's 1984 Super Bowl Commercial</p> | ||
| 244 | <p class="fragment">1985 - Apple's Lemmings Super Bowl Commercial</p> | ||
| 245 | <p class="fragment">1997 - Apple's Think Different Campaign</p> | ||
| 246 | </section> | ||
| 247 | |||
| 248 | <section class="slide level2"> | ||
| 249 | <p>Don't be a lemming</p> | ||
| 250 | </section> | ||
| 251 | |||
| 252 | <section class="slide level2"> | ||
| 253 | <h1>Everybody in this room is wearing a uniform and don't kid yourself</h1> | ||
| 254 | <p>Frank Zappa</p> | ||
| 255 | </section> | ||
| 256 | |||
| 257 | <section class="slide level2"> | ||
| 258 | <h1>"Think Different" isn't about allowing for individualism, it's about defining an alternate tribe</h1> | ||
| 259 | <p class="fragment">And don't kid yourself, it's about selling product</p> | ||
| 260 | </section> | ||
| 261 | |||
| 262 | <section class="slide level2"> | ||
| 263 | <h1>What if you're this poor fellow?</h1> | ||
| 264 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 265 | <p class="fragment">http://worstcats.tumblr.com</p> | ||
| 266 | </section> | ||
| 267 | |||
| 268 | <section class="slide level2"> | ||
| 269 | <p>How much of your life is computerized?</p> | ||
| 270 | <p>How many of the four freedoms do you have with each of those things?</p> | ||
| 271 | </section> | ||
| 272 | |||
| 273 | <section class="slide level2"> | ||
| 274 | <h1>superfish</p> | ||
| 275 | <p class="fragment">My laptop did not come with any software pre-installed that is designed to "improve my shopping experience"</p> | ||
| 276 | <p class="fragment">If it had, I am empowered to remove it</p> | ||
| 277 | </section> | ||
| 278 | |||
| 279 | <section class="slide level2"> | ||
| 280 | <p>Why are we here? I think many people assume, wrongly, that a company exists solely to make money. Money is an important part of a company's existence, if the company is any good. But a result is not a cause. We have to go deeper and find the real reason for our being.</p> | ||
| 281 | <p>David Packard</p> | ||
| 282 | </section> | ||
| 283 | |||
| 284 | <section class="slide level2"> | ||
| 285 | <h1>Free Software is a way that I can make the world a better place</h1> | ||
| 286 | </section> | ||
| 287 | |||
| 288 | <section class="slide level2"> | ||
| 289 | <h1>Free Software is a way to reconnect with our humanity in a sea of white privilege</h1> | ||
| 290 | </section> | ||
| 291 | |||
| 292 | <section class="slide level2"> | ||
| 293 | <h1>Free Software drives an unprecedented amount of innovation</h1> | ||
| 294 | </section> | ||
| 295 | |||
| 296 | <section class="slide level2"> | ||
| 297 | <h1>Believe you can change the world</h1> | ||
| 298 | <p>David Packard</p> | ||
| 299 | </section> | ||
| 300 | |||
| 301 | <section> | ||
| 302 | <h1> Thank you! </h1> | ||
| 303 | <h4> <a href='http://inaugust.com/talks/lemmings.html'>http://inaugust.com/talks/lemmings.html</a> </h4> | ||
| 304 | <h3> twitter: @e_monty </h3> | ||
| 305 | </section> | ||
| 306 | |||
| 307 | </div> | ||
| 308 | <div class="footer"> | ||
| 309 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 310 | <img alt="Creative Commons License" | ||
| 311 | style="border-width:0" | ||
| 312 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 313 | </a><br /> | ||
| 314 | Licensed under a | ||
| 315 | <a rel="license" | ||
| 316 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 317 | Creative Commons Attribution 4.0 International License | ||
| 318 | </a>. | ||
| 319 | <br /> | ||
| 320 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 321 | </div> | ||
| 322 | |||
| 323 | </div> | ||
| 324 | |||
| 325 | <script src="/lib/js/head.min.js"></script> | ||
| 326 | <script src="/js/reveal.js"></script> | ||
| 327 | |||
| 328 | <script src="/js/this.js"></script> | ||
| 329 | |||
| 330 | </body> | ||
| 331 | </html> | ||
diff --git a/src/talks/now-what/index.html b/src/talks/now-what/index.html new file mode 100644 index 0000000..ca21fa8 --- /dev/null +++ b/src/talks/now-what/index.html | |||
| @@ -0,0 +1,436 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>OpenStack works ... so now what?</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = '/css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | </head> | ||
| 32 | <body> | ||
| 33 | |||
| 34 | <div class="reveal"><div class="slides"> | ||
| 35 | |||
| 36 | <section data-state="cover"> | ||
| 37 | |||
| 38 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 39 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 40 | property="dct:title" | ||
| 41 | rel="dct:type"> | ||
| 42 | OpenStack works ... so now what? | ||
| 43 | </span></h1> | ||
| 44 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 45 | property="cc:attributionName">Monty Taylor</h3> | ||
| 46 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 47 | rel="cc:attributionURL" | ||
| 48 | href='http://inaugust.com/talks/now-what.html'>http://inaugust.com/talks/now-what.html</a> </h4> | ||
| 49 | <h3> twitter: @e_monty </h3> | ||
| 50 | </section> | ||
| 51 | |||
| 52 | <section id="who-am-i-hp" class="slide level2"> | ||
| 53 | <h1>Who am I?</h1> | ||
| 54 | <img style="float:right; margin-right:24pt" src="/images/hp-logo.png"/> | ||
| 55 | <p> Distinguished Technologist </p> | ||
| 56 | <p> HP Cloud </p> | ||
| 57 | </section> | ||
| 58 | |||
| 59 | <section id="who-am-i-openstack" class="slide level2"> | ||
| 60 | <h1>Who am I?</h1> | ||
| 61 | <img style="float:right; margin-right:24pt" src="/images/openstack-cloud-software-vertical-large.png" /> | ||
| 62 | <p>Technical Committee</p> | ||
| 63 | <p>Foundation Board of Directors</p> | ||
| 64 | <p>Developer Infrastructure Core Team</p> | ||
| 65 | </section> | ||
| 66 | |||
| 67 | <section id="what-are-we-going-to-talk-about" class="slide level2"> | ||
| 68 | <h1>What are we going to talk about?</h1> | ||
| 69 | <ul> | ||
| 70 | <li>OpenStack</li> | ||
| 71 | <li>My application</li> | ||
| 72 | <li>Your applications</li> | ||
| 73 | </ul> | ||
| 74 | </section> | ||
| 75 | |||
| 76 | <section class="slide level2"> | ||
| 77 | <img src="/images/openstack-software-diagram.png" | ||
| 78 | alt="OpenStack" /> | ||
| 79 | </section> | ||
| 80 | |||
| 81 | <section class="slide level2"> | ||
| 82 | <p>As an application developer,<br /> | ||
| 83 | I want to deploy and run an application on the internet<br /> | ||
| 84 | so that my customers all over the world can consume it.</p> | ||
| 85 | </section> | ||
| 86 | |||
| 87 | <section class="slide level2"> | ||
| 88 | <p>As an application developer,<br /> | ||
| 89 | I want to deploy the application across multiple clouds<br /> | ||
| 90 | so that my service survives issues in any one of them.</p> | ||
| 91 | </section> | ||
| 92 | <section class="slide level2"> | ||
| 93 | <h1>THIS WORKS</h1> | ||
| 94 | <h3>I'm doing it myself as we speak</h3> | ||
| 95 | <img style="float:right; margin-right:24pt" src="/images/graphite.openstack.org.png"/> | ||
| 96 | |||
| 97 | |||
| 98 | <p class='fragment'>10-20k VMs per-day</p> | ||
| 99 | <p class='fragment'>3 (soon to be 6) clouds</p> | ||
| 100 | <p class='fragment'>Only using OpenStack APIs</p> | ||
| 101 | |||
| 102 | </section> | ||
| 103 | |||
| 104 | <section id="openstack-infra" data-transition='zoom'> | ||
| 105 | <h1>OpenStack Infra</h1> | ||
| 106 | </section> | ||
| 107 | |||
| 108 | <section id="tooling-automation-and-ci-for-openstack-project" class="slide level2" data-transition='zoom'> | ||
| 109 | <h1>Tooling, Automation and CI for OpenStack Project</h1> | ||
| 110 | </section> | ||
| 111 | |||
| 112 | <section id="developers" class="slide level2" data-transition='zoom'> | ||
| 113 | <h1>2000 Developers</h1> | ||
| 114 | </section> | ||
| 115 | |||
| 116 | <section id="gated-commits" class="slide level2" data-transition='zoom'> | ||
| 117 | <h1>Gated Commits</h1> | ||
| 118 | <p>Every commit is fully integration tested (twice) before landing</p> | ||
| 119 | </section> | ||
| 120 | |||
| 121 | <section id="each-test-runs-on-a-single-use-cloud-slave" class="slide level2" data-transition='zoom'> | ||
| 122 | <h1>Each Test Runs on a Single Use Cloud Slave</h1> | ||
| 123 | <p>This is that "cloud scale out" part</p> | ||
| 124 | </section> | ||
| 125 | |||
| 126 | <section id="million-test-jobs-in-the-last-6-months" class="slide level2" data-transition='zoom'> | ||
| 127 | <h1>1.7 Million Test Jobs in the last 6 Months</h1> | ||
| 128 | </section> | ||
| 129 | |||
| 130 | <section id="million-tests-in-a-month" class="slide level2" data-transition='zoom'> | ||
| 131 | <h1>15 Million Tests in a month</h1> | ||
| 132 | </section> | ||
| 133 | |||
| 134 | <section id="terabytes-of-log-data-in-six-months" class="slide level2" data-transition='zoom'> | ||
| 135 | <h1>18 Terabytes of Log Data in six months</h1> | ||
| 136 | </section> | ||
| 137 | |||
| 138 | <section id="we-have-no-servers" class="slide level2" data-transition='zoom'> | ||
| 139 | <h1>We have no servers</h1> | ||
| 140 | <p>It all runs across HP and Rackspace Public Clouds.</p> | ||
| 141 | </section> | ||
| 142 | |||
| 143 | <section id="architecture" class="slide level2"> | ||
| 144 | <h1>Architecture</h1> | ||
| 145 | <p><img src="/images/infra_architecture.jpg" alt="image" /></p> | ||
| 146 | </section> | ||
| 147 | |||
| 148 | <section class="slide level2"> | ||
| 149 | <h1>Gerrit</h1> | ||
| 150 | <ul> | ||
| 151 | <li class="fragment"> | ||
| 152 | Traditional 'Enterprise' Java Application</li> | ||
| 153 | <li class="fragment">Single Nova VM, Cinder Volume</li> | ||
| 154 | <li class="fragment">Scale out farm of git replicas</li> | ||
| 155 | </ul> | ||
| 156 | </section> | ||
| 157 | |||
| 158 | <section class="slide level2"> | ||
| 159 | <img src="/images/gerrit-graph.png" /> | ||
| 160 | </section> | ||
| 161 | |||
| 162 | <section class="slide level2"> | ||
| 163 | <h1>Fun Numbers</h1> | ||
| 164 | <ul> | ||
| 165 | <li>2500 changes every week</li> | ||
| 166 | <li>15000 change <em>revisions</em> every week</li> | ||
| 167 | <li>10,000 new changes every 42 days</li> | ||
| 168 | </ul> | ||
| 169 | </section> | ||
| 170 | |||
| 171 | <section class="slide level2"> | ||
| 172 | <img src="/images/what-happens.png" /> | ||
| 173 | </section> | ||
| 174 | |||
| 175 | <section class="slide level2"> | ||
| 176 | <h1>nodepool</h1> | ||
| 177 | <ul> | ||
| 178 | <li class="fragment">Cloud Native</li> | ||
| 179 | <li class="fragment"> | ||
| 180 | Purpose built in Python</li> | ||
| 181 | <li class="fragment">Keeps a pool of ready to go nodes</li> | ||
| 182 | <li class="fragment">Multi-cloud</li> | ||
| 183 | <li class="fragment">Fully elastic - responds to demand</li> | ||
| 184 | </ul> | ||
| 185 | </section> | ||
| 186 | |||
| 187 | <section class="slide level2"> | ||
| 188 | <img style="float:left; margin-left:24pt; width:35%;" src="/images/gerrit-graph.png" /> | ||
| 189 | <img style="float:right; margin-right:24pt; width: 50%;" src="/images/nodepool-graph.png" /> | ||
| 190 | </section> | ||
| 191 | |||
| 192 | <section class="slide level2"> | ||
| 193 | <h1>OpenStack Works!</h1> | ||
| 194 | <p class='fragment'>What next?</p> | ||
| 195 | <p class='fragment'>Make it easier</p> | ||
| 196 | </section> | ||
| 197 | |||
| 198 | <section class="slide level2"> | ||
| 199 | <h1>Basic things you want to do</h1> | ||
| 200 | |||
| 201 | <p class='fragment'>Get (make/fetch/find) a base image</p> | ||
| 202 | <p class='fragment'>Upload it to each cloud</p> | ||
| 203 | <p class='fragment'>Boot a VM on one or more of the clouds</p> | ||
| 204 | <p class='fragment'>Ensure it's on the Internet</p> | ||
| 205 | </section> | ||
| 206 | |||
| 207 | <section class="slide level2"> | ||
| 208 | <h1>We've made this harder than it should be</h1> | ||
| 209 | </section> | ||
| 210 | |||
| 211 | <section class="slide level2"> | ||
| 212 | <h1>Get a base image</h1> | ||
| 213 | <p class='fragment'>OpenStack diskimage-builder</p> | ||
| 214 | <p class='fragment'>packer</p> | ||
| 215 | <p class='fragment'>Download pre-built image from Ubuntu/RedHat/SuSE</p> | ||
| 216 | </section> | ||
| 217 | |||
| 218 | <section class="slide level2"> | ||
| 219 | <h1>Problem: hypervisor image file format</h1> | ||
| 220 | <ul> | ||
| 221 | <li class='fragment'> | ||
| 222 | Rackspace uses VHD | ||
| 223 | </li> | ||
| 224 | <li class='fragment'> | ||
| 225 | HP uses qcow2 | ||
| 226 | </li> | ||
| 227 | <li class='fragment'> | ||
| 228 | DreamHost uses RAW | ||
| 229 | </li> | ||
| 230 | </ul> | ||
| 231 | </section> | ||
| 232 | |||
| 233 | <section class="slide level2"> | ||
| 234 | <h1>Problem: image API version</h1> | ||
| 235 | <ul> | ||
| 236 | <li class='fragment'> | ||
| 237 | HP uses v1 | ||
| 238 | </li> | ||
| 239 | <li class='fragment'> | ||
| 240 | vexxhost uses v2 | ||
| 241 | </li> | ||
| 242 | </ul> | ||
| 243 | <p class="fragment">Good news! We made a plan for this at | ||
| 244 | the summit</p> | ||
| 245 | </section> | ||
| 246 | |||
| 247 | <section class="slide level2"> | ||
| 248 | <h1>Problem: image task vs. PUT</h1> | ||
| 249 | <pre> | ||
| 250 | swift upload --object-name local-image-filename images image-name | ||
| 251 | glance task-create | ||
| 252 | --type=import | ||
| 253 | --input='{"import_from": "images/image-name", | ||
| 254 | "image_properties" : {"name": "My Image Name"}}' | ||
| 255 | </pre> | ||
| 256 | <pre> | ||
| 257 | glance image-create --name=image-name --file=local-image-filename | ||
| 258 | </pre> | ||
| 259 | </section> | ||
| 260 | |||
| 261 | <section class="slide level2"> | ||
| 262 | <h1>Problem: Ensure it's on the Internet</h1> | ||
| 263 | <ul> | ||
| 264 | <li class='fragment'>Cloud has externally routable IP from neutron (RunAbove, OVH)</li> | ||
| 265 | <li class='fragment'>Cloud has externally routable IP neutron AND supports optional private tenant networks (vexxhost)</li> | ||
| 266 | <li class='fragment'>Cloud has private tenant network provided by neutron and requires floating IP (HP, Dreamhost)</li> | ||
| 267 | <li class='fragment'>Cloud only has private tenant network provided by nova-network and requires floating-ip for external routing (auro)</li> | ||
| 268 | <li class='fragment'>Cloud has externally routable IP from neutron but no neutron APIs (Rackspace)</li> | ||
| 269 | </ul> | ||
| 270 | </section> | ||
| 271 | |||
| 272 | <section class="slide level2"> | ||
| 273 | <h1>Maybe in code ...</h1> | ||
| 274 | <pre> | ||
| 275 | def get_server_external_ipv4(cloud, server): | ||
| 276 | if cloud.has_service('network'): | ||
| 277 | try: | ||
| 278 | server_ports = cloud.search_ports( | ||
| 279 | filters={'device_id': server.id}) | ||
| 280 | ext_nets = cloud.search_networks(filters={'router:external': True}) | ||
| 281 | except NeutronClientException as e: | ||
| 282 | pass # fall through | ||
| 283 | else: | ||
| 284 | for net in ext_nets: | ||
| 285 | for port in server_ports: | ||
| 286 | if net['id'] == port['network_id']: | ||
| 287 | for ip in port['fixed_ips']: | ||
| 288 | if _utils.is_ipv4(ip['ip_address']): | ||
| 289 | return ip['ip_address'] | ||
| 290 | ext_ip = get_server_ip(server, key_name='public') | ||
| 291 | if ext_ip is not None: | ||
| 292 | return ext_ip | ||
| 293 | for interfaces in server.addresses.values(): | ||
| 294 | for interface in interfaces: | ||
| 295 | if _utils.is_ipv4(interface['addr']) and \ | ||
| 296 | _utils.is_globally_routable_ipv4(interface['addr']): | ||
| 297 | return interface['addr'] | ||
| 298 | return None | ||
| 299 | </pre> | ||
| 300 | </section> | ||
| 301 | |||
| 302 | <section class="slide level2"> | ||
| 303 | <h1>I think we can do better than that</h1> | ||
| 304 | </section> | ||
| 305 | |||
| 306 | <section class="slide level2"> | ||
| 307 | <h1>What am I doing about it?</h1> | ||
| 308 | </section> | ||
| 309 | |||
| 310 | <section class="slide level2"> | ||
| 311 | <h1>os-client-config</h1> | ||
| 312 | <h3>http://git.openstack.org/cgit/openstack/os-client-config</h3> | ||
| 313 | <p>A library to handle config information for openstack clients</p> | ||
| 314 | <p>Tracks differences in vendors that can't be discovered</p> | ||
| 315 | <p>In use in python-openstackclient and ansible</p> | ||
| 316 | </section> | ||
| 317 | |||
| 318 | <section class="slide level2"> | ||
| 319 | <h1>os-client-config</h1> | ||
| 320 | <p>~/.config/openstack/clouds.yaml</p> | ||
| 321 | <pre> | ||
| 322 | clouds: | ||
| 323 | hp-mordred: | ||
| 324 | profile: hp | ||
| 325 | auth: | ||
| 326 | username: mordred@inaugust.com | ||
| 327 | password: XXXXXXXXXXXXX | ||
| 328 | project_name: mordred@inaugust.com | ||
| 329 | region_name: region-b.geo-1 | ||
| 330 | dreamhost: | ||
| 331 | profile: dreamhost | ||
| 332 | auth: | ||
| 333 | username: montay6 | ||
| 334 | project_name: dhc2111978 | ||
| 335 | password: XXXXXXXXXXXXX | ||
| 336 | region_name: RegionOne | ||
| 337 | </pre> | ||
| 338 | </section> | ||
| 339 | |||
| 340 | <section class="slide level2"> | ||
| 341 | <h1>shade</h1> | ||
| 342 | <h3>http://git.openstack.org/cgit/openstack-infra/shade</h3> | ||
| 343 | <p>A library to wrap business logic around client libraries</p> | ||
| 344 | <pre> | ||
| 345 | cloud.create_image('image-name', filename='image-filename.qcow2') | ||
| 346 | cloud.create_server('my-server', image='immage-name', auto_ip=True) | ||
| 347 | </pre> | ||
| 348 | <p>In use in Infra Nodepool and ansible</p> | ||
| 349 | </section> | ||
| 350 | |||
| 351 | <section class="slide level2"> | ||
| 352 | <h1>ansible</h1> | ||
| 353 | <p>Brand new modules, based on shade</p> | ||
| 354 | <p>Coming in 2.0 release</p> | ||
| 355 | <pre> | ||
| 356 | - os_keypair: | ||
| 357 | cloud: hp-mordred | ||
| 358 | name: mordred | ||
| 359 | public_key_file: ~/.ssh/id_rsa.pub | ||
| 360 | - os_image: | ||
| 361 | cloud: hp-mordred | ||
| 362 | name: Monty Ubuntu | ||
| 363 | file: ubuntu.vhd | ||
| 364 | - os_server: | ||
| 365 | cloud: hp-mordred | ||
| 366 | name: my-server | ||
| 367 | flavor_ram: 1024 | ||
| 368 | image: Monty Ubuntu | ||
| 369 | </pre> | ||
| 370 | </section> | ||
| 371 | |||
| 372 | <section class="slide level2"> | ||
| 373 | <h1>ansible</h1> | ||
| 374 | <p>multi-cloud support</p> | ||
| 375 | <pre> | ||
| 376 | - os_keypair: | ||
| 377 | cloud: "{{ item }" | ||
| 378 | name: mordred | ||
| 379 | public_key_file: ~/.ssh/id_rsa.pub | ||
| 380 | with-items: | ||
| 381 | - vexxhost | ||
| 382 | - rackspace | ||
| 383 | - mordred-hp | ||
| 384 | - ovh | ||
| 385 | </pre> | ||
| 386 | </section> | ||
| 387 | |||
| 388 | <section class="slide level2"> | ||
| 389 | <h1>What should we do about it?</h1> | ||
| 390 | <ul> | ||
| 391 | <li> | ||
| 392 | Get back to basics | ||
| 393 | </li><li> | ||
| 394 | shade existence is a bug | ||
| 395 | </li><li> | ||
| 396 | Make some decisions about divergences in the basic levels | ||
| 397 | </li><li> | ||
| 398 | Take a stand even if one of our product managers disagrees | ||
| 399 | </li><li> | ||
| 400 | Ensure that simple things are simple | ||
| 401 | </li> | ||
| 402 | </ul> | ||
| 403 | </section> | ||
| 404 | |||
| 405 | |||
| 406 | <section> | ||
| 407 | <h1> Thank you! </h1> | ||
| 408 | <h4> <a href='http://inaugust.com/talks/now-what.html'>http://inaugust.com/talks/now-what.html</a> </h4> | ||
| 409 | <h3> twitter: @e_monty </h3> | ||
| 410 | </section> | ||
| 411 | |||
| 412 | </div> | ||
| 413 | <div class="footer"> | ||
| 414 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 415 | <img alt="Creative Commons License" | ||
| 416 | style="border-width:0" | ||
| 417 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 418 | </a><br /> | ||
| 419 | Licensed under a | ||
| 420 | <a rel="license" | ||
| 421 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 422 | Creative Commons Attribution 4.0 International License | ||
| 423 | </a>. | ||
| 424 | <br /> | ||
| 425 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 426 | </div> | ||
| 427 | |||
| 428 | </div> | ||
| 429 | |||
| 430 | <script src="/lib/js/head.min.js"></script> | ||
| 431 | <script src="/js/reveal.js"></script> | ||
| 432 | |||
| 433 | <script src="/js/this.js"></script> | ||
| 434 | |||
| 435 | </body> | ||
| 436 | </html> | ||
diff --git a/src/talks/os-client-config/index.html b/src/talks/os-client-config/index.html new file mode 100644 index 0000000..04ef2fd --- /dev/null +++ b/src/talks/os-client-config/index.html | |||
| @@ -0,0 +1,242 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>os-client-config: Making OpenStack usability easier, starting with client configuration</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = '/css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | </head> | ||
| 32 | <body> | ||
| 33 | |||
| 34 | <div class="background"></div> | ||
| 35 | |||
| 36 | <div class="reveal"> | ||
| 37 | <div class="slides"> | ||
| 38 | |||
| 39 | <section data-state="cover"> | ||
| 40 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 41 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 42 | property="dct:title" | ||
| 43 | rel="dct:type"> | ||
| 44 | os-client-config | ||
| 45 | </span></h1> | ||
| 46 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 47 | property="cc:attributionName">Monty Taylor</h3> | ||
| 48 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 49 | rel="cc:attributionURL" | ||
| 50 | href='http://inaugust.com/talks/os-client-config.html'>http://inaugust.com/talks/os-client-config.html</a> </h4> | ||
| 51 | <h3> twitter: @e_monty </h3> | ||
| 52 | </section> | ||
| 53 | |||
| 54 | <section> | ||
| 55 | <h1>I'm awash in cloud accounts</h1> | ||
| 56 | <ul> | ||
| 57 | <li>Infra has 5<ul> | ||
| 58 | <li>control plane and nodepool in 3 Rackspace regions</li> | ||
| 59 | <li>control plane and nodepool in HP</li> | ||
| 60 | </ul></li> | ||
| 61 | <li>two personal on HP and one on Rackspace</li> | ||
| 62 | <li>two internal HP accounts</li> | ||
| 63 | <li>That's 12 OpenStack cloud-regions</li> | ||
| 64 | </ul> | ||
| 65 | </section> | ||
| 66 | |||
| 67 | <section> | ||
| 68 | How do I connect to them? | ||
| 69 | </section> | ||
| 70 | |||
| 71 | <section> | ||
| 72 | List Servers | ||
| 73 | <pre> | ||
| 74 | openstack \ | ||
| 75 | --auth-url=https://region-b.geo-1.identity.hpcloudsvc.com:35357/v2.0 \ | ||
| 76 | --username=mordred@inaugust.com \ | ||
| 77 | --password=XXXXXX \ | ||
| 78 | --project-name=mordred@inaugust.com \ | ||
| 79 | --region-name=region-b.geo-1 \ | ||
| 80 | server list | ||
| 81 | </pre> | ||
| 82 | </section> | ||
| 83 | |||
| 84 | <section> | ||
| 85 | Well that's silly | ||
| 86 | </section> | ||
| 87 | |||
| 88 | <section> | ||
| 89 | Consistent Environment Variables | ||
| 90 | <pre> | ||
| 91 | export OS_AUTH_URL=https://region-b.geo-1.identity.hpcloudsvc.com:35357/v2.0 | ||
| 92 | export OS_USERNAME=mordred@inaugust.com | ||
| 93 | export OS_PASSWORD=XXXXXXXXXXXXX | ||
| 94 | export OS_TENANT_NAME=mordred@inaugust.com | ||
| 95 | export OS_REGION_NAME=region-b.geo-1 | ||
| 96 | </pre> | ||
| 97 | </section> | ||
| 98 | |||
| 99 | <section> | ||
| 100 | 12 shell script snippet files | ||
| 101 | </section> | ||
| 102 | |||
| 103 | <section> | ||
| 104 | ENV processing is in command line tools, not client libraries | ||
| 105 | </section> | ||
| 106 | |||
| 107 | <section> | ||
| 108 | ENV processing is just copied everywhere | ||
| 109 | <pre> | ||
| 110 | $ egrep -r 'add_(option|argument).*os-username' openstack/python*client | wc -l | ||
| 111 | 13 | ||
| 112 | </pre> | ||
| 113 | </section> | ||
| 114 | |||
| 115 | <section> | ||
| 116 | <h1> Pre-existing knowledge </h1> | ||
| 117 | <ul> | ||
| 118 | <li>auth-url</li> | ||
| 119 | <li>glance API version</li> | ||
| 120 | <li>override settings<ul> | ||
| 121 | <li>URLs (swift URL in rackspace keystone catalog is wrong)</li> | ||
| 122 | <li>service types (DNS in HP is hpext:dns)</li> | ||
| 123 | </ul></li> | ||
| 124 | </ul> | ||
| 125 | </section> | ||
| 126 | |||
| 127 | <section> | ||
| 128 | <p>I wrote a library (yay!)</p> | ||
| 129 | <h1>os-client-config</h1> | ||
| 130 | </section> | ||
| 131 | |||
| 132 | <section> | ||
| 133 | <h1>Processes, in this order</h1> | ||
| 134 | <ul> | ||
| 135 | <li>config file: clouds.yaml</li> | ||
| 136 | <li>env vars</li> | ||
| 137 | <li>command line args</li> | ||
| 138 | </ul> | ||
| 139 | </section> | ||
| 140 | |||
| 141 | <section> | ||
| 142 | <h1>Provides vendor defaults</h1> | ||
| 143 | <pre> | ||
| 144 | CLOUD_DEFAULTS = dict( | ||
| 145 | hp=dict( | ||
| 146 | auth_url='https://region-b.geo-1.identity.hpcloudsvc.com:35357/v2.0', | ||
| 147 | dns_service_type='hpext:dns', | ||
| 148 | ), | ||
| 149 | rackspace=dict( | ||
| 150 | auth_url='https://identity.api.rackspacecloud.com/v2.0/', | ||
| 151 | database_service_type='rax:database', | ||
| 152 | image_api_version='2', | ||
| 153 | ) | ||
| 154 | ) | ||
| 155 | </pre> | ||
| 156 | <p> Patches welcome for any cloud </p> | ||
| 157 | </section> | ||
| 158 | |||
| 159 | <section> | ||
| 160 | <h1>clouds.yaml</h1> | ||
| 161 | <pre> | ||
| 162 | clouds: | ||
| 163 | mordred: | ||
| 164 | cloud: hp | ||
| 165 | username: mordred@inaugust.com | ||
| 166 | password: XXXXXXXXXX | ||
| 167 | project_name: mordred@inaugust.com | ||
| 168 | region_name: region-b.geo-1 | ||
| 169 | monty: | ||
| 170 | cloud: https://region-b.geo-1.identity.hpcloudsvc.com:35357/v2.0 | ||
| 171 | username: monty.taylor@hp.com | ||
| 172 | password: XXXXXXXXX | ||
| 173 | project_name: monty.taylor@hp.com-default-tenant | ||
| 174 | region_name: region-b.geo-1 | ||
| 175 | dns_service_type: hpext:dns | ||
| 176 | rax: | ||
| 177 | cloud: rackspace | ||
| 178 | username: openstackci | ||
| 179 | password: XXXXXXX | ||
| 180 | project_id: 610275 | ||
| 181 | region_name: DFW,ORD,IAD | ||
| 182 | </pre> | ||
| 183 | </section> | ||
| 184 | |||
| 185 | <section> | ||
| 186 | <h1> Allows for named clouds </h1> | ||
| 187 | <pre> | ||
| 188 | openstack --cloud=mordred server list | ||
| 189 | </pre> | ||
| 190 | <pre> | ||
| 191 | export OS_CLOUD=mordred | ||
| 192 | openstack server list | ||
| 193 | </pre> | ||
| 194 | </section> | ||
| 195 | |||
| 196 | |||
| 197 | <section> | ||
| 198 | <h1> Where is it in use? </h1> | ||
| 199 | <ul> | ||
| 200 | <li>Shade library</li> | ||
| 201 | <li>python-openstackclient <small>(patch in flight)</small></li> | ||
| 202 | <li>ansible <small>(patch in flight)</small></li> | ||
| 203 | </ul> | ||
| 204 | </section> | ||
| 205 | |||
| 206 | <section> | ||
| 207 | <h1>Where is it?</h1> | ||
| 208 | <p>http://git.openstack.org/stackforge/os-client-config</p> | ||
| 209 | <p>http://pypi.python.org/pypi/os-client-config</p> | ||
| 210 | </section> | ||
| 211 | |||
| 212 | <section> | ||
| 213 | <h1>Where is it?</h1> | ||
| 214 | <p>http://git.openstack.org/stackforge/os-client-config</p> | ||
| 215 | <p>http://pypi.python.org/pypi/os-client-config</p> | ||
| 216 | </section> | ||
| 217 | |||
| 218 | </div> | ||
| 219 | |||
| 220 | <div class="footer"> | ||
| 221 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 222 | <img alt="Creative Commons License" | ||
| 223 | style="border-width:0" | ||
| 224 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 225 | </a><br /> | ||
| 226 | Licensed under a | ||
| 227 | <a rel="license" | ||
| 228 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 229 | Creative Commons Attribution 4.0 International License | ||
| 230 | </a>. | ||
| 231 | <br /> | ||
| 232 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 233 | </div> | ||
| 234 | </div> | ||
| 235 | |||
| 236 | <script src="/lib/js/head.min.js"></script> | ||
| 237 | <script src="/js/reveal.js"></script> | ||
| 238 | |||
| 239 | <script src="/js/this.js"></script> | ||
| 240 | |||
| 241 | </body> | ||
| 242 | </html> | ||
diff --git a/src/talks/product-management/index.html b/src/talks/product-management/index.html new file mode 100644 index 0000000..58f8bd4 --- /dev/null +++ b/src/talks/product-management/index.html | |||
| @@ -0,0 +1,454 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>Liberty, Product Management and OpenStack Technology</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = '/css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | </head> | ||
| 32 | <body> | ||
| 33 | |||
| 34 | <div class="reveal"><div class="slides"> | ||
| 35 | |||
| 36 | <section data-state="cover"> | ||
| 37 | |||
| 38 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 39 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 40 | property="dct:title" | ||
| 41 | rel="dct:type"> | ||
| 42 | Liberty, Product Management and OpenStack Technology | ||
| 43 | </span></h1> | ||
| 44 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 45 | property="cc:attributionName">Monty Taylor</h3> | ||
| 46 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 47 | rel="cc:attributionURL" | ||
| 48 | href='http://inaugust.com/talks/product-management.html'>http://inaugust.com/talks/product-management.html</a> </h4> | ||
| 49 | <h3> twitter: @e_monty </h3> | ||
| 50 | </section> | ||
| 51 | |||
| 52 | <section class="slide level2"> | ||
| 53 | <img src="/images/openstack-cloud-software-vertical-large.png" | ||
| 54 | alt="OpenStack" /> | ||
| 55 | </section> | ||
| 56 | |||
| 57 | <section class="slide level2"> | ||
| 58 | <h1>Mission</h1> | ||
| 59 | <p> | ||
| 60 | to produce the ubiquitous Open Source Cloud Computing platform that will meet the needs of public and private clouds regardless of size, by being simple to implement and massively scalable. | ||
| 61 | </p> | ||
| 62 | </section> | ||
| 63 | |||
| 64 | <section class="slide level2"> | ||
| 65 | <h1>Borrowed from Ubuntu</h1> | ||
| 66 | <p>Time Based Releases</p> | ||
| 67 | <p>Design summits each cycle</p> | ||
| 68 | <p>Release codenames in alphabetical order</p> | ||
| 69 | </section> | ||
| 70 | |||
| 71 | <section class="slide level2"> | ||
| 72 | <h1>Different from Ubuntu</h1> | ||
| 73 | <p>No BDFL</p> | ||
| 74 | <p>All decisions are democratic</p> | ||
| 75 | </section> | ||
| 76 | |||
| 77 | <section class="slide level2"> | ||
| 78 | <img src="/images/openstack-size.jpg" /> | ||
| 79 | </section> | ||
| 80 | |||
| 81 | <section class="slide level2"> | ||
| 82 | <h1>"OpenStack needs product management"</h1> | ||
| 83 | </section> | ||
| 84 | |||
| 85 | <section class="slide level2"> | ||
| 86 | <p>Developers work on whatever they feel like, so there is no cohesion</p> | ||
| 87 | <p class='fragment'><small>not quite</small></p> | ||
| 88 | </section> | ||
| 89 | |||
| 90 | <section class="slide level2"> | ||
| 91 | <img src="/images/openstack-size.jpg" /> | ||
| 92 | </section> | ||
| 93 | |||
| 94 | <section class="slide level2"> | ||
| 95 | <h1>OpenStack needs product management <em>coordination</em></h1> | ||
| 96 | </section> | ||
| 97 | |||
| 98 | <section class="slide level2"> | ||
| 99 | <h1> Product Management Working Group </h1> | ||
| 100 | <p>Today 2:00pm - 3:30pm Room 212</p> | ||
| 101 | </section> | ||
| 102 | |||
| 103 | <section class="slide level2"> | ||
| 104 | <h1>What can the working group do?</h1> | ||
| 105 | <ul> | ||
| 106 | <li class='fragment'> | ||
| 107 | Define problems | ||
| 108 | </li> | ||
| 109 | <li class='fragment'> | ||
| 110 | Coordinate prorities | ||
| 111 | </li> | ||
| 112 | <li class='fragment'> | ||
| 113 | Communicate problems clearly to tech community | ||
| 114 | </li> | ||
| 115 | </section> | ||
| 116 | |||
| 117 | <section class="slide level2"> | ||
| 118 | <h1>I've got 99 Problems ...</h1> | ||
| 119 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 120 | </section> | ||
| 121 | |||
| 122 | <section class="slide level2"> | ||
| 123 | <p>As an application developer,<br /> | ||
| 124 | I want to deploy and run an application on the internet<br /> | ||
| 125 | so that my customers all over the world can consume it.</p> | ||
| 126 | </section> | ||
| 127 | |||
| 128 | <section class="slide level2"> | ||
| 129 | <p>As an application developer,<br /> | ||
| 130 | I want to deploy the application across multiple clouds<br /> | ||
| 131 | so that my service survives issues in any one of them.</p> | ||
| 132 | </section> | ||
| 133 | |||
| 134 | <section class="slide level2"> | ||
| 135 | <h1>THIS WORKS</h1> | ||
| 136 | <h3>I'm doing it myself as we speak</h3> | ||
| 137 | <img style="float:right; margin-right:24pt" src="/images/graphite.openstack.org.png"/> | ||
| 138 | |||
| 139 | |||
| 140 | <p class='fragment'>10-20k VMs per-day</p> | ||
| 141 | <p class='fragment'>3 (soon to be 6) clouds</p> | ||
| 142 | <p class='fragment'>Only using OpenStack APIs</p> | ||
| 143 | |||
| 144 | </section> | ||
| 145 | |||
| 146 | <section class="slide level2"> | ||
| 147 | <h1>To do this, there are some basic steps</h1> | ||
| 148 | |||
| 149 | <p class='fragment'>Get (make/fetch/find) a base image</p> | ||
| 150 | <p class='fragment'>Upload it to each cloud</p> | ||
| 151 | <p class='fragment'>Boot a VM on one or more of the clouds</p> | ||
| 152 | <p class='fragment'>Ensure it's on the Internet</p> | ||
| 153 | |||
| 154 | </section> | ||
| 155 | |||
| 156 | <section class="slide level2"> | ||
| 157 | <h1>Get a base image</h1> | ||
| 158 | <p class='fragment'>OpenStack diskimage-builder</p> | ||
| 159 | <p class='fragment'>packer</p> | ||
| 160 | <p class='fragment'>Download pre-built image from Ubuntu/Fedora</p> | ||
| 161 | </section> | ||
| 162 | |||
| 163 | <section class="slide level2"> | ||
| 164 | <h1>NO</h1> | ||
| 165 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 166 | </section> | ||
| 167 | |||
| 168 | <section class="slide level2"> | ||
| 169 | <h1>Problem: hypervisor image file format</h1> | ||
| 170 | <ul> | ||
| 171 | <li class='fragment'> | ||
| 172 | Rackspace uses VHD | ||
| 173 | </li> | ||
| 174 | <li class='fragment'> | ||
| 175 | HP uses qcow2 | ||
| 176 | </li> | ||
| 177 | <li class='fragment'> | ||
| 178 | DreamHost uses RAW | ||
| 179 | </li> | ||
| 180 | </ul> | ||
| 181 | </section> | ||
| 182 | |||
| 183 | <section class="slide level2"> | ||
| 184 | <h1>Upload it to each cloud</h1> | ||
| 185 | <pre>glance image-create</pre> | ||
| 186 | </section> | ||
| 187 | |||
| 188 | <section class="slide level2"> | ||
| 189 | <h1>NO</h1> | ||
| 190 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 191 | </section> | ||
| 192 | |||
| 193 | <section class="slide level2"> | ||
| 194 | <h1>Problem: image API version</h1> | ||
| 195 | <ul> | ||
| 196 | <li class='fragment'> | ||
| 197 | HP uses v1 | ||
| 198 | </li> | ||
| 199 | <li class='fragment'> | ||
| 200 | vexxhost uses v2 | ||
| 201 | </li> | ||
| 202 | </ul> | ||
| 203 | </section> | ||
| 204 | |||
| 205 | <section class="slide level2"> | ||
| 206 | <h1>Problem: List API versions</h1> | ||
| 207 | <p>Root of the Image API lists versions</p> | ||
| 208 | <p>keystone catalog only lists a versioned endpoint</p> | ||
| 209 | </section> | ||
| 210 | |||
| 211 | <section class="slide level2"> | ||
| 212 | <h1>Problem: API version discovery</h1> | ||
| 213 | <p>Try one - if it doesn't work, try the other</p> | ||
| 214 | <p><em>maybe</em> look at the end of the API endpoint for v1 or v2</p> | ||
| 215 | </section> | ||
| 216 | |||
| 217 | <section class="slide level2"> | ||
| 218 | <h1>Upload it to each cloud</h1> | ||
| 219 | <pre>glance image-create filename</pre> | ||
| 220 | </section> | ||
| 221 | |||
| 222 | <section class="slide level2"> | ||
| 223 | <h1>NO</h1> | ||
| 224 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 225 | </section> | ||
| 226 | |||
| 227 | <section class="slide level2"> | ||
| 228 | <h1>Problem: image task vs. PUT</h1> | ||
| 229 | <pre> | ||
| 230 | swift upload --object-name local-image-filename images image-name | ||
| 231 | glance task-create | ||
| 232 | --type=import | ||
| 233 | --input='{"import_from": "images/image-name", | ||
| 234 | "image_properties" : {"name": "My Image Name"}}' | ||
| 235 | </pre> | ||
| 236 | <pre> | ||
| 237 | glance image-create --name=image-name --file=local-image-filename | ||
| 238 | </pre> | ||
| 239 | </section> | ||
| 240 | |||
| 241 | <section class="slide level2"> | ||
| 242 | <h1>Boot a VM on one or more of the clouds</h1> | ||
| 243 | <pre>nova boot --image=image-name --flavor=something | ||
| 244 | </pre> | ||
| 245 | </section> | ||
| 246 | |||
| 247 | <section class="slide level2"> | ||
| 248 | <h1>NO</h1> | ||
| 249 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 250 | </section> | ||
| 251 | |||
| 252 | <section class="slide level2"> | ||
| 253 | <h1>Problem: the image needs to get on the network</h1> | ||
| 254 | <ul> | ||
| 255 | <li class='fragment'>DHCP</li> | ||
| 256 | <li class='fragment'>Static Network Config in Config Drive</li> | ||
| 257 | <li class='fragment'>Vendor-specific agent doing file injection</li> | ||
| 258 | </ul> | ||
| 259 | </section> | ||
| 260 | |||
| 261 | <section class="slide level2"> | ||
| 262 | <h1> | ||
| 263 | <pre>nova boot --image=image-name --flavor=something | ||
| 264 | </pre> | ||
| 265 | </section> | ||
| 266 | |||
| 267 | <section class="slide level2"> | ||
| 268 | <h1>NO</h1> | ||
| 269 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 270 | </section> | ||
| 271 | |||
| 272 | <section class="slide level2"> | ||
| 273 | <h1>Problem: Ensure it's on the Internet</h1> | ||
| 274 | <ul> | ||
| 275 | <li class='fragment'>My VM may have a public IP</li> | ||
| 276 | <li class='fragment'>My VM may need a floating IP from nova</li> | ||
| 277 | <li class='fragment'>My VM may need a floating IP from neutron</li> | ||
| 278 | </ul> | ||
| 279 | </section> | ||
| 280 | |||
| 281 | <section class="slide level2"> | ||
| 282 | <h1>VM Network with nova-network</h1> | ||
| 283 | <pre> | ||
| 284 | addresses: | ||
| 285 | private: | ||
| 286 | - addr: 10.181.5.77 | ||
| 287 | version: 4 | ||
| 288 | public: | ||
| 289 | - addr: 2001:4800:7810:512:be76:4eff:fe05:8325 | ||
| 290 | version: 6 | ||
| 291 | - addr: 98.129.169.30 | ||
| 292 | version: 4 | ||
| 293 | </pre> | ||
| 294 | </section> | ||
| 295 | |||
| 296 | <section class="slide level2"> | ||
| 297 | <h1>VM Network with nova-network</h1> | ||
| 298 | <pre> | ||
| 299 | addresses: | ||
| 300 | mordred@inaugust.com-network: | ||
| 301 | - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:a6:de:26 | ||
| 302 | OS-EXT-IPS:type: fixed | ||
| 303 | addr: 10.0.0.6 | ||
| 304 | version: 4 | ||
| 305 | - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:a6:de:26 | ||
| 306 | OS-EXT-IPS:type: floating | ||
| 307 | addr: 15.126.239.219 | ||
| 308 | version: 4 | ||
| 309 | </pre> | ||
| 310 | </section> | ||
| 311 | |||
| 312 | <section class="slide level2"> | ||
| 313 | <h1>Boot a VM on one or more of the clouds</h1> | ||
| 314 | <pre> | ||
| 315 | nova boot --image=image-name --flavor=something --name=my-server | ||
| 316 | nova floating-ip-create | ||
| 317 | nova floating-ip-associate my-server {{ value_from_create }} | ||
| 318 | </pre> | ||
| 319 | </section> | ||
| 320 | |||
| 321 | <section class="slide level2"> | ||
| 322 | <h1>NO</h1> | ||
| 323 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 324 | </section> | ||
| 325 | |||
| 326 | <section class="slide level2"> | ||
| 327 | <h1>Problem: My Internet server is behind a NAT</h1> | ||
| 328 | </section> | ||
| 329 | |||
| 330 | <section class="slide level2"> | ||
| 331 | <h1>Problem: Security Groups<h1> | ||
| 332 | <p>I was trying to spin up a web server, but all my ports were blocked</p> | ||
| 333 | <p><small>I'll be using ansible and puppet on this machine, I can configure iptables thanks</small></p> | ||
| 334 | </section> | ||
| 335 | |||
| 336 | <section class="slide level2"> | ||
| 337 | <h1>Boot a VM on one or more of the clouds</h1> | ||
| 338 | <pre> | ||
| 339 | nova secgroup-add-rule default tcp 80 80 0.0.0.0/0 | ||
| 340 | nova boot --image=image-name --flavor=something --name=my-server | ||
| 341 | nova floating-ip-create | ||
| 342 | nova floating-ip-associate my-server {{ value_from_create }} | ||
| 343 | </pre> | ||
| 344 | </section> | ||
| 345 | |||
| 346 | <section class="slide level2"> | ||
| 347 | <h1>Wow. So that's</h1> | ||
| 348 | <ul> | ||
| 349 | <li> | ||
| 350 | Image Format | ||
| 351 | </li><li> | ||
| 352 | Image API version | ||
| 353 | </li><li> | ||
| 354 | Image upload mechanism | ||
| 355 | </li><li> | ||
| 356 | Networking config / public private | ||
| 357 | </li><li> | ||
| 358 | Networking config nova/neutron | ||
| 359 | </li><li> | ||
| 360 | Floating IP? | ||
| 361 | </li><li> | ||
| 362 | Floating IP nova/neutron | ||
| 363 | </li><li> | ||
| 364 | DHCP or Static networking config | ||
| 365 | </li><li> | ||
| 366 | Security Group config | ||
| 367 | </li> | ||
| 368 | </ul> | ||
| 369 | </section> | ||
| 370 | |||
| 371 | <section class="slide level2"> | ||
| 372 | <h1>I think we can do better than that</h1> | ||
| 373 | </section> | ||
| 374 | |||
| 375 | <section class="slide level2"> | ||
| 376 | <h1>What am I doing about it?</h1> | ||
| 377 | </section> | ||
| 378 | |||
| 379 | <section class="slide level2"> | ||
| 380 | <h1>os-client-config</h1> | ||
| 381 | <h3>http://git.openstack.org/cgit/openstack/os-client-config</h3> | ||
| 382 | <p>A library to handle config information for openstack clients</p> | ||
| 383 | <p>Tracks differences in vendors that can't be discovered</p> | ||
| 384 | <p>In use in python-openstackclient and ansible today</p> | ||
| 385 | </section> | ||
| 386 | |||
| 387 | <section class="slide level2"> | ||
| 388 | <h1>shade</h1> | ||
| 389 | <h3>http://git.openstack.org/cgit/openstack-infra/shade</h3> | ||
| 390 | <p>A library to wrap business logic around client libraries</p> | ||
| 391 | <pre> | ||
| 392 | cloud.create_server('my-server', auto_ip=True) | ||
| 393 | </pre> | ||
| 394 | <p>In use in Infra Nodepool and ansible today</p> | ||
| 395 | </section> | ||
| 396 | |||
| 397 | <section class="slide level2"> | ||
| 398 | <h1>Raising Issues</h1> | ||
| 399 | <ul> | ||
| 400 | <li> Product Management Working Group (today, 2pm, room 212) </li> | ||
| 401 | <li> DefCore (Wednesday 10:30 - 12:30, East Building Room 2/3)</li> | ||
| 402 | <li> Direct interaction with the teams and PTLs </li> | ||
| 403 | </ul> | ||
| 404 | </section> | ||
| 405 | |||
| 406 | <section class="slide level2"> | ||
| 407 | <h1>What should we do about it?</h1> | ||
| 408 | <ul> | ||
| 409 | <li> | ||
| 410 | Get back to basics | ||
| 411 | </li><li> | ||
| 412 | shade existence is a bug | ||
| 413 | </li><li> | ||
| 414 | Make some decisions about divergences in the basic levels | ||
| 415 | </li><li> | ||
| 416 | Take a stand even if one of our product managers disagrees | ||
| 417 | </li><li> | ||
| 418 | Ensure that simple things are simple | ||
| 419 | </li> | ||
| 420 | </ul> | ||
| 421 | </section> | ||
| 422 | |||
| 423 | |||
| 424 | <section> | ||
| 425 | <h1> Thank you! </h1> | ||
| 426 | <h4> <a href='http://inaugust.com/talks/product-management.html'>http://inaugust.com/talks/product-management.html</a> </h4> | ||
| 427 | <h3> twitter: @e_monty </h3> | ||
| 428 | </section> | ||
| 429 | |||
| 430 | </div> | ||
| 431 | <div class="footer"> | ||
| 432 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 433 | <img alt="Creative Commons License" | ||
| 434 | style="border-width:0" | ||
| 435 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 436 | </a><br /> | ||
| 437 | Licensed under a | ||
| 438 | <a rel="license" | ||
| 439 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 440 | Creative Commons Attribution 4.0 International License | ||
| 441 | </a>. | ||
| 442 | <br /> | ||
| 443 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 444 | </div> | ||
| 445 | |||
| 446 | </div> | ||
| 447 | |||
| 448 | <script src="/lib/js/head.min.js"></script> | ||
| 449 | <script src="/js/reveal.js"></script> | ||
| 450 | |||
| 451 | <script src="/js/this.js"></script> | ||
| 452 | |||
| 453 | </body> | ||
| 454 | </html> | ||
diff --git a/src/talks/tripleo-ansible/index.html b/src/talks/tripleo-ansible/index.html new file mode 100644 index 0000000..58814df --- /dev/null +++ b/src/talks/tripleo-ansible/index.html | |||
| @@ -0,0 +1,794 @@ | |||
| 1 | <!doctype html> | ||
| 2 | <html lang="en"> | ||
| 3 | |||
| 4 | <head> | ||
| 5 | <meta charset="utf-8"> | ||
| 6 | |||
| 7 | <title>TripleO and Ansible</title> | ||
| 8 | |||
| 9 | <meta name="apple-mobile-web-app-capable" content="yes" /> | ||
| 10 | <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" /> | ||
| 11 | |||
| 12 | <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | ||
| 13 | |||
| 14 | <link rel="stylesheet" href="/css/reveal.css"> | ||
| 15 | <link rel="stylesheet" href="/css/theme/openstack.css" id="theme"> | ||
| 16 | |||
| 17 | <!-- For syntax highlighting --> | ||
| 18 | <link rel="stylesheet" href="/lib/css/zenburn.css"> | ||
| 19 | |||
| 20 | <!-- If the query includes 'print-pdf', include the PDF print sheet --> | ||
| 21 | <script> | ||
| 22 | if( window.location.search.match( /print-pdf/gi ) ) { | ||
| 23 | var link = document.createElement( 'link' ); | ||
| 24 | link.rel = 'stylesheet'; | ||
| 25 | link.type = 'text/css'; | ||
| 26 | link.href = '/css/print/pdf.css'; | ||
| 27 | document.getElementsByTagName( 'head' )[0].appendChild( link ); | ||
| 28 | } | ||
| 29 | </script> | ||
| 30 | |||
| 31 | </head> | ||
| 32 | <body> | ||
| 33 | |||
| 34 | <div class="background"> | ||
| 35 | <img alt="" id="head-icon" width="218" height="67" | ||
| 36 | src="/images/openstack-cloud-software-horizontal-small.png" /></div> | ||
| 37 | |||
| 38 | |||
| 39 | <div class="reveal"> | ||
| 40 | <div class="slides"> | ||
| 41 | |||
| 42 | <section data-state="cover"> | ||
| 43 | <img src="/images/openstack-cloud-software-vertical-large.png" | ||
| 44 | alt="OpenStack" id="cover"/> | ||
| 45 | |||
| 46 | <h1><span xmlns:dct="http://purl.org/dc/terms/" | ||
| 47 | href="http://purl.org/dc/dcmitype/InteractiveResource" | ||
| 48 | property="dct:title" | ||
| 49 | rel="dct:type"> | ||
| 50 | TripleO and Ansible | ||
| 51 | </span></h1> | ||
| 52 | <h3 xmlns:cc="http://creativecommons.org/ns#" | ||
| 53 | property="cc:attributionName">Monty Taylor</h3> | ||
| 54 | <h4><a xmlns:cc="http://creativecommons.org/ns#" | ||
| 55 | rel="cc:attributionURL" | ||
| 56 | href='http://inaugust.com/talks/tripleo-ansible.html'>http://inaugust.com/talks/tripleo-ansible.html</a> </h4> | ||
| 57 | <h3> twitter: @e_monty </h3> | ||
| 58 | </section> | ||
| 59 | |||
| 60 | <section id="who-am-i" class="slide level2"> | ||
| 61 | <h1>Who am I?</h1> | ||
| 62 | <ul> | ||
| 63 | <li>Distinguished Technologist at HP</li> | ||
| 64 | <li>OpenStack Technical Committee</li> | ||
| 65 | <li>OpenStack Foundation Board of Directors</li> | ||
| 66 | <li>OpenStack Infra Core Team</li> | ||
| 67 | </ul> | ||
| 68 | </section> | ||
| 69 | |||
| 70 | <section id="what-are-we-going-to-talk-about" class="slide level2"> | ||
| 71 | <h1>What are we going to talk about?</h1> | ||
| 72 | <ul> | ||
| 73 | <li>TripleO</li> | ||
| 74 | <li>Ironic</li> | ||
| 75 | <li>Ansible</li> | ||
| 76 | </ul> | ||
| 77 | </section> | ||
| 78 | |||
| 79 | <section> | ||
| 80 | <h1>What is TripleO?</h1> | ||
| 81 | <ul> | ||
| 82 | <li class="fragment"> an idea </li> | ||
| 83 | <li class="fragment"> OpenStack On OpenStack </li> | ||
| 84 | <li class="fragment"> use Ironic-based OpenStack to Operate Openstack </li> | ||
| 85 | </ul> | ||
| 86 | </section> | ||
| 87 | |||
| 88 | |||
| 89 | <section> | ||
| 90 | <h1>Velocity</h1> | ||
| 91 | <ul> | ||
| 92 | <li>Cloud Empowers <em>developers</em></li> | ||
| 93 | <li>Cloud Enables Increased Velocity</li> | ||
| 94 | <li>Cloud Drives Agility</li> | ||
| 95 | </ul> | ||
| 96 | </section> | ||
| 97 | |||
| 98 | <section> | ||
| 99 | <a href="#/2" class="image"><img src="/images/openstack-software-diagram.png" width="90%"></a> | ||
| 100 | <aside class="notes"> | ||
| 101 | Perhaps you've seen this before. <br> | ||
| 102 | Even with addition of lots of PaaS projects, the abstraction here still holds. | ||
| 103 | </aside> | ||
| 104 | </section> | ||
| 105 | |||
| 106 | <section> | ||
| 107 | <h1>OpenStack is not a virtualization layer, | ||
| 108 | it's an abstraction layer.</h1> | ||
| 109 | <aside class="notes"> | ||
| 110 | When Danny Sabah @ IBM said this, it hit home for me. I had already been working on Ironic for a year.<br> | ||
| 111 | KVM, NFV ... <br> | ||
| 112 | Virtualization is a powerful tool. Abstraction empowers people.<br> | ||
| 113 | OpenStack community is thriving because of the power of open abstractions layers. | ||
| 114 | </aside> | ||
| 115 | </section> | ||
| 116 | |||
| 117 | <section> | ||
| 118 | <section> | ||
| 119 | <h1>What if your <em>developers</em> aren't writing Angry Birds?</h1> | ||
| 120 | </section> | ||
| 121 | |||
| 122 | <section> | ||
| 123 | <h1>What if your <em>developers</em> aren't developers?</h1> | ||
| 124 | </section> | ||
| 125 | |||
| 126 | <section> | ||
| 127 | <h3>If infrastructure is code ...</h3> | ||
| 128 | <h1>What if your <em>developers</em> are <em>operators</em>?</h1> | ||
| 129 | </section> | ||
| 130 | |||
| 131 | <section> | ||
| 132 | <img src="/images/ugly-openstack.jpg" /> | ||
| 133 | </section> | ||
| 134 | |||
| 135 | <section> | ||
| 136 | <h1>Why wouldn't you give your operators the same power as your | ||
| 137 | developers?</h1> | ||
| 138 | <p>Don't you like them?</p> | ||
| 139 | </section> | ||
| 140 | |||
| 141 | </section> | ||
| 142 | |||
| 143 | <section> | ||
| 144 | <h1><i>Ironic</i> is a service capable of<br> | ||
| 145 | managing and provisioning<br> | ||
| 146 | <i>physical machines</i>.</h1> | ||
| 147 | <aside class="notes"> | ||
| 148 | Do one thing and do it well. Ready-state-GO!<br> | ||
| 149 | Vendor neutral API. Distributed control plane.<br> | ||
| 150 | Deploy images to reduce entropy. Servers are cattle, not pets.<br> | ||
| 151 | Can use stand-alone, but not simple today. | ||
| 152 | </aside> | ||
| 153 | </section> | ||
| 154 | |||
| 155 | <section> | ||
| 156 | <h1>Nova + Ironic</h1> | ||
| 157 | Same but different | ||
| 158 | <aside class="notes"> | ||
| 159 | User gets same Nova API. Abstraction is maintained.<br> | ||
| 160 | What are the benefits using Nova? (sched, flavors, etc)<br> | ||
| 161 | Talk briefly about evolution from nova-baremetal. | ||
| 162 | </aside> | ||
| 163 | </section> | ||
| 164 | |||
| 165 | <section> | ||
| 166 | <a href="#/2" class="image"><img src="/images/ironic-nova-layer.jpg"></a> | ||
| 167 | <aside class="notes"> | ||
| 168 | Talk through the slide<br> | ||
| 169 | Going to show some examples next | ||
| 170 | </aside> | ||
| 171 | </section> | ||
| 172 | <section> | ||
| 173 | <pre><code> | ||
| 174 | $ ironic node-create -d pxe_ipmitool \ | ||
| 175 | -i ipmi_username=admin -i ipmi_password=fake -i ipmi_address=10.1.2.3 \ | ||
| 176 | -p cpus=4 -p memory_mb=8192 -p local_gb=500 \ | ||
| 177 | -e note='spare server' | ||
| 178 | +--------------+------------------------------------------------------------+ | ||
| 179 | | Property | Value | | ||
| 180 | +--------------+------------------------------------------------------------+ | ||
| 181 | | chassis_uuid | None | | ||
| 182 | | driver | pxe_ipmitool | | ||
| 183 | | driver_info | {u'ipmi_address': u'10.1.2.3', u'ipmi_username': u'admin', | | ||
| 184 | | | u'ipmi_password': u'fake'} | | ||
| 185 | | extra | {u'note': u'spare server'} | | ||
| 186 | | properties | {u'memory_mb': u'8192', u'local_gb': u'500', u'cpus': u'4'}| | ||
| 187 | | uuid | 7a1ce8d0-9679-4d87-8f54-b11f6e8adb8f | | ||
| 188 | +--------------+------------------------------------------------------------+ | ||
| 189 | </code></pre> | ||
| 190 | </section> | ||
| 191 | <section> | ||
| 192 | <pre><code> | ||
| 193 | $ tail -f /var/log/nova/n-cpu.log | ||
| 194 | ... | ||
| 195 | 2014-05-01 03:47:05.878 AUDIT nova.compute.resource_tracker [-] | ||
| 196 | Free ram (MB): 8192 | ||
| 197 | 2014-05-01 03:47:05.878 AUDIT nova.compute.resource_tracker [-] | ||
| 198 | Free disk (GB): 500 | ||
| 199 | 2014-05-01 03:47:05.878 AUDIT nova.compute.resource_tracker [-] | ||
| 200 | Free VCPUS: 4 | ||
| 201 | </code></pre> | ||
| 202 | </section> | ||
| 203 | |||
| 204 | <section> | ||
| 205 | <h1>Diversity is good</h1> | ||
| 206 | <ul> | ||
| 207 | <li>Cloud providers: HP, Rackspace, Dreamhost, Cloudwatt</li> | ||
| 208 | <li>Deployments: Public, Managed, Private</li> | ||
| 209 | <li>Compute drivers: virtual, bare metal, container</li> | ||
| 210 | <li>Ironic drivers: pxe/impi, ilo, HP OneView</li> | ||
| 211 | </ul> | ||
| 212 | </section> | ||
| 213 | |||
| 214 | <section> | ||
| 215 | <h1>What can you do?</h1> | ||
| 216 | <h3 class="fragment">anything - it's a cloud</h3> | ||
| 217 | </section> | ||
| 218 | |||
| 219 | <section> | ||
| 220 | <h1>What the heck has the TripleO team been doing then?</h1> | ||
| 221 | <img src="/images/worstcat-lettuce.jpg" /> | ||
| 222 | </section> | ||
| 223 | |||
| 224 | <section> | ||
| 225 | <h1>TripleO</h1> | ||
| 226 | <ul> | ||
| 227 | <li>Community developed</li> | ||
| 228 | <li>Opinionated</li> | ||
| 229 | <li>Avoid distro and config management religion</li> | ||
| 230 | <li>Be a usable <em>real</em> deployment we can gate on</li> | ||
| 231 | </ul> | ||
| 232 | </section> | ||
| 233 | |||
| 234 | <section> | ||
| 235 | <h1> Community Developed </h1> | ||
| 236 | <ul> | ||
| 237 | <li>Exist as part of the OpenStack project</li> | ||
| 238 | <li>Prove the story end to end</li> | ||
| 239 | <li>Subject to TC governance</li> | ||
| 240 | <li>Tighter feedback loop</li> | ||
| 241 | </ul> | ||
| 242 | </section> | ||
| 243 | |||
| 244 | <section> | ||
| 245 | <h1> Avoid distro and config management religion </h1> | ||
| 246 | <ul> | ||
| 247 | <li>rpm vs. deb - in the gate == rpm + deb</li> | ||
| 248 | <li>puppet vs. chef vs. salt vs. ansible == all of them</li> | ||
| 249 | <li>Choosing one excludes other folks from participating</li> | ||
| 250 | </ul> | ||
| 251 | </section> | ||
| 252 | |||
| 253 | <section> | ||
| 254 | <h1> Opinionated </h1> | ||
| 255 | <ul> | ||
| 256 | <li>Golden Images</li> | ||
| 257 | <li>Upgrade tied to HA</li> | ||
| 258 | <li>Target Continual Delivery</li> | ||
| 259 | <li>Drive fixes into OpenStack directly</li> | ||
| 260 | </ul> | ||
| 261 | </section> | ||
| 262 | |||
| 263 | <section> | ||
| 264 | <h1>Major Components</h1> | ||
| 265 | <ul> | ||
| 266 | <li>nova+ironic</li> | ||
| 267 | <li>heat</li> | ||
| 268 | <li>diskimage-builder</li> | ||
| 269 | <li>os-collect-config</li> | ||
| 270 | <li>os-apply-config</li> | ||
| 271 | <li>os-refresh-config</li> | ||
| 272 | </ul> | ||
| 273 | </section> | ||
| 274 | |||
| 275 | <section> | ||
| 276 | <h1>Lesson from os-*-config</h1> | ||
| 277 | <img class="fragment" src="/images/worstcat-dog.jpg" /> | ||
| 278 | </section> | ||
| 279 | |||
| 280 | <section> | ||
| 281 | <h1>Lesson from os-*-config</h1> | ||
| 282 | <img class="fragment" src="/images/standards.png" /> | ||
| 283 | </section> | ||
| 284 | |||
| 285 | <section> | ||
| 286 | <h1>Lesson from os-*-config</h1> | ||
| 287 | <p>If you ever think "oh, that's silly, it would be so much easier | ||
| 288 | if I just ..."<br /> it will almost never actually be easier if you | ||
| 289 | just ...</p> | ||
| 290 | </section> | ||
| 291 | |||
| 292 | <section> | ||
| 293 | <h1>The Deployment Story</h1> | ||
| 294 | <ol> | ||
| 295 | <li>disk-image-builder builds images and uploads to glance</li> | ||
| 296 | <li>Heat drives Nova/Ironic</li> | ||
| 297 | <li>Heat delivers metadata to os-collect-config</li> | ||
| 298 | <li>os-collect-config applies any in-instance changes needed</li> | ||
| 299 | </ol> | ||
| 300 | </section> | ||
| 301 | |||
| 302 | <section> | ||
| 303 | <h1>The Update Story</h1> | ||
| 304 | <p>Heat magically just updates things</li> | ||
| 305 | </section> | ||
| 306 | |||
| 307 | <section> | ||
| 308 | <h1>BUT I ALREADY USE ?????</h1> | ||
| 309 | <h3>(screw you guys, I'm going home)</h3> | ||
| 310 | </section> | ||
| 311 | |||
| 312 | <section> | ||
| 313 | <h1>Yeah, I do to</h1> | ||
| 314 | <h3>(I use puppet and ansible myself)</h3> | ||
| 315 | </section> | ||
| 316 | |||
| 317 | <section> | ||
| 318 | <h1>That's fine - use them - it's a cloud!</h1> | ||
| 319 | <h3 class="fragment">This is supposed to be empowering, not enforcing</h3> | ||
| 320 | </section> | ||
| 321 | |||
| 322 | <section> | ||
| 323 | <h1>Whatever you want!</h1> | ||
| 324 | <ul> | ||
| 325 | <li>Heat to deploy and update images, os-*-*config for config</li> | ||
| 326 | <li>Heat to deploy images, ansible to update images, puppet for config</li> | ||
| 327 | <li>Ansible to deploy base image + packages, salt to update packages, chef for config</li> | ||
| 328 | <li>juju to deploy ... nah, I'm just kidding</li> | ||
| 329 | </ul> | ||
| 330 | </section> | ||
| 331 | |||
| 332 | <section> | ||
| 333 | <h1>The New Update Story</h1> | ||
| 334 | <p>Ansible takes over for upgrades</p> | ||
| 335 | </section> | ||
| 336 | |||
| 337 | <section id="step-two-ansible-for-orchestration" class="titleslide slide level1"> | ||
| 338 | <h1>Ansible for Orchestration</h1> | ||
| 339 | </section> | ||
| 340 | |||
| 341 | <section id="about-ansible" class="slide level2"> | ||
| 342 | <h1>About Ansible</h1> | ||
| 343 | <ul> | ||
| 344 | <li>Open Source System Management tool</li> | ||
| 345 | <li>Written in Python</li> | ||
| 346 | <li>Sequence of steps to perform</li> | ||
| 347 | <li>Works over SSH</li> | ||
| 348 | <li>Incremental Adoption</li> | ||
| 349 | </ul> | ||
| 350 | </section> | ||
| 351 | |||
| 352 | <section> | ||
| 353 | <h1>ad-hoc operation</h1> | ||
| 354 | <pre> | ||
| 355 | ansible '*' -m shell -p uptime | ||
| 356 | </pre> | ||
| 357 | </section> | ||
| 358 | |||
| 359 | <section id="yaml-syntax" class="slide level2"> | ||
| 360 | <h1>YAML Syntax</h1> | ||
| 361 | <pre><code> | ||
| 362 | - hosts: '*.slave.openstack.org' | ||
| 363 | tasks: | ||
| 364 | - shell: 'rm -rf ~jenkins/workspace/*{{ project }}*' | ||
| 365 | </code></pre> | ||
| 366 | <p>That's executed:</p> | ||
| 367 | <pre> | ||
| 368 | ansible-playbook -f 10 /etc/ansible/clean_workspaces.yaml --extra-vars "project=$PROJECTNAME" | ||
| 369 | </pre> | ||
| 370 | </section> | ||
| 371 | |||
| 372 | <section id="ansible-organization" class="slide level2"> | ||
| 373 | <h1>Ansible Organization</h1> | ||
| 374 | <ul> | ||
| 375 | <li>modules</li> | ||
| 376 | <li>plays</li> | ||
| 377 | <li>playbooks</li> | ||
| 378 | <li>roles</li> | ||
| 379 | </ul> | ||
| 380 | </section> | ||
| 381 | |||
| 382 | <section id="use-ansible-to-run-puppet" class="slide level2"> | ||
| 383 | <h1>Use Ansible to Run Puppet!</h1> | ||
| 384 | </section> | ||
| 385 | |||
| 386 | <section id="puppet-module" class="slide level2"> | ||
| 387 | <h1>puppet module</h1> | ||
| 388 | <pre><code>def main(): | ||
| 389 | module = AnsibleModule(argument_spec=dict( | ||
| 390 | timeout=dict(default="30m"), | ||
| 391 | puppetmaster=dict(required=True), | ||
| 392 | show_diff=dict(default=False, aliases=['show-diff'], type='bool'), | ||
| 393 | )) | ||
| 394 | p = module.params | ||
| 395 | |||
| 396 | puppet_cmd = module.get_bin_path("puppet", False) | ||
| 397 | if not puppet_cmd: | ||
| 398 | module.fail_json(msg="Could not find puppet. Please ensure it is installed.") | ||
| 399 | </code></pre> | ||
| 400 | </section> | ||
| 401 | |||
| 402 | <section id="puppet-module-2" class="slide level2"> | ||
| 403 | <h1>puppet module (cont)</h1> | ||
| 404 | <pre><code class="python"> | ||
| 405 | cmd = ("timeout -s 9 %(timeout)s %(puppet_cmd)s agent --onetime" | ||
| 406 | " --server %(puppetmaster)s" | ||
| 407 | " --ignorecache --no-daemonize --no-usecacheonfailure --no-splay" | ||
| 408 | " --detailed-exitcodes --verbose") % dict( | ||
| 409 | timeout=pipes.quote(p['timeout']), puppet_cmd=PUPPET_CMD, | ||
| 410 | puppetmaster=pipes.quote(p['puppetmaster'])) | ||
| 411 | if p['show_diff']: | ||
| 412 | cmd += " --show-diff" | ||
| 413 | rc, stdout, stderr = module.run_command(cmd) | ||
| 414 | </code></pre> | ||
| 415 | </section> | ||
| 416 | |||
| 417 | <section id="puppet-module-3" class="slide level2"> | ||
| 418 | Please. Everyone. Marvel at the following logic | ||
| 419 | <pre><code> | ||
| 420 | if rc == 0: # success | ||
| 421 | module.exit_json(rc=rc, changed=False, stdout=stdout) | ||
| 422 | elif rc == 1: | ||
| 423 | # rc==1 could be because it's disabled OR there was a compilation failure | ||
| 424 | disabled = "administratively disabled" in stdout | ||
| 425 | if disabled: | ||
| 426 | msg = "puppet is disabled" | ||
| 427 | else: | ||
| 428 | msg = "puppet compilation failed" | ||
| 429 | module.fail_json(rc=rc, disabled=disabled, msg=msg, stdout=stdout, stderr=stderr) | ||
| 430 | elif rc == 2: # success with changes | ||
| 431 | module.exit_json(changed=True) | ||
| 432 | elif rc == 124: # timeout | ||
| 433 | module.exit_json(rc=rc, msg="%s timed out" % cmd, stdout=stdout, stderr=stderr) | ||
| 434 | else: # failure | ||
| 435 | module.fail_json(rc=rc, msg="%s failed" % (cmd), stdout=stdout, stderr=stderr) | ||
| 436 | </code></pre> | ||
| 437 | </section> | ||
| 438 | |||
| 439 | <section id="puppet-play" class="slide level2"> | ||
| 440 | <h1>puppet play</h1> | ||
| 441 | <pre><code> | ||
| 442 | - name: run puppet | ||
| 443 | puppet: | ||
| 444 | puppetmaster: "{{puppetmaster}}" | ||
| 445 | </code></pre> | ||
| 446 | </section> | ||
| 447 | |||
| 448 | <section id="puppet-role" class="slide level2"> | ||
| 449 | <h1>puppet role</h1> | ||
| 450 | <p>roles/remote_puppet/tasks/main.yml</p> | ||
| 451 | </section> | ||
| 452 | |||
| 453 | <section id="remote-puppet-playbook" class="slide level2"> | ||
| 454 | <h1>remote puppet playbook</h1> | ||
| 455 | <pre><code> | ||
| 456 | - hosts: git0* | ||
| 457 | gather_facts: false | ||
| 458 | max_fail_percentage: 1 | ||
| 459 | roles: | ||
| 460 | - { role: remote_puppet, puppetmaster: puppetmaster.openstack.org } | ||
| 461 | - hosts: review.openstack.org | ||
| 462 | gather_facts: false | ||
| 463 | roles: | ||
| 464 | - { role: remote_puppet, puppetmaster: puppetmaster.openstack.org } | ||
| 465 | - hosts: "!review.openstack.org:!git0*:!afs*" | ||
| 466 | gather_facts: false | ||
| 467 | roles: | ||
| 468 | - { role: remote_puppet, puppetmaster: puppetmaster.openstack.org } | ||
| 469 | </pre></code> | ||
| 470 | </section> | ||
| 471 | |||
| 472 | <section id="ansible-inventory" class="slide level2"> | ||
| 473 | <h1>ansible inventory</h1> | ||
| 474 | <ul> | ||
| 475 | <li>List of servers to operate on</li> | ||
| 476 | <li>Optionally variables associated with each server</li> | ||
| 477 | <li>Optional groups of servers</li> | ||
| 478 | <li>Simple file in /etc/ansible/hosts</li> | ||
| 479 | <li>Dynamic executable that returns JSON</li> | ||
| 480 | </ul> | ||
| 481 | </section> | ||
| 482 | |||
| 483 | <section id="ansible-inventory-from-file" class="slide level2"> | ||
| 484 | <h1>Simple inventory</h1> | ||
| 485 | <pre> | ||
| 486 | review.openstack.org | ||
| 487 | git01.openstack.org | ||
| 488 | git02.openstack.org | ||
| 489 | pypi.dfw.openstack.org | ||
| 490 | pypi.iad.openstack.org | ||
| 491 | |||
| 492 | [pypi] | ||
| 493 | pypi.dfw.openstack.org | ||
| 494 | pypi.iad.openstack.org | ||
| 495 | |||
| 496 | [git] | ||
| 497 | git01.openstack.org | ||
| 498 | git02.openstack.org | ||
| 499 | </pre> | ||
| 500 | </section> | ||
| 501 | |||
| 502 | <section id="ansible-inventory-from-puppet" class="slide level2"> | ||
| 503 | <h1>ansible inventory from puppet certs</h1> | ||
| 504 | <pre><code> | ||
| 505 | import json | ||
| 506 | import subprocess | ||
| 507 | |||
| 508 | output = [ | ||
| 509 | x.split()[1][1:-1] for x in subprocess.check_output( | ||
| 510 | ["puppet","cert","list","-a"]).split('\n') | ||
| 511 | if x.startswith('+') | ||
| 512 | ] | ||
| 513 | |||
| 514 | data = { | ||
| 515 | '_meta': {'hostvars': dict()}, | ||
| 516 | 'ungrouped': output, | ||
| 517 | } | ||
| 518 | print json.dumps(data, sort_keys=True, indent=2) | ||
| 519 | </code></pre> | ||
| 520 | </section> | ||
| 521 | |||
| 522 | <section> | ||
| 523 | <h1>Ansible for Cloud Management</h1> | ||
| 524 | </section> | ||
| 525 | |||
| 526 | <section> | ||
| 527 | <h1>ansible and OpenStack</h1> | ||
| 528 | <ul> | ||
| 529 | <li>Ansible modules are just python</li> | ||
| 530 | <li>playbooks are lists of steps to take</li> | ||
| 531 | <li>Have plays/roles that provision servers</li> | ||
| 532 | <li>Infrastructure as code - for real!</li> | ||
| 533 | </ul> | ||
| 534 | </section> | ||
| 535 | |||
| 536 | <section> | ||
| 537 | <h1>Consider this data</h1> | ||
| 538 | <pre><code> | ||
| 539 | pypi: | ||
| 540 | image_name: Ubuntu 12.04.4 | ||
| 541 | flavor_ram: 2048 | ||
| 542 | provision_group: ubuntu_hosts | ||
| 543 | volumes: | ||
| 544 | - size: 200 | ||
| 545 | mount: /srv | ||
| 546 | hosts: | ||
| 547 | pypi.dfw: | ||
| 548 | region: DFW | ||
| 549 | pypi.iad: | ||
| 550 | region: IAD | ||
| 551 | pypi.ord: | ||
| 552 | region: ORD | ||
| 553 | pypi.region-b.geo-1: | ||
| 554 | cloud: hp | ||
| 555 | </code></pre> | ||
| 556 | </section> | ||
| 557 | |||
| 558 | <section> | ||
| 559 | <h1>Steps to launch a node</h1> | ||
| 560 | <ol> | ||
| 561 | <li>Create a compute instance</li> | ||
| 562 | <li>Wait for instance to exist</li> | ||
| 563 | <li>Create a floating IP</li> | ||
| 564 | <li>Attach floating IP to instance</li> | ||
| 565 | <li>Create one or more volumes</li> | ||
| 566 | <li>Attach volumes to instance</li> | ||
| 567 | <li>Wait for SSH to work</li> | ||
| 568 | <li>On host, format each volume</li> | ||
| 569 | <li>On host, mount each volume</li> | ||
| 570 | <li>On host, install config management software</li> | ||
| 571 | <li>On host, run config management software</li> | ||
| 572 | </ol> | ||
| 573 | </section> | ||
| 574 | |||
| 575 | <section> | ||
| 576 | <h1>Launch a node</h1> | ||
| 577 | <pre><code> | ||
| 578 | --- | ||
| 579 | - name: Launch Node | ||
| 580 | os_compute: | ||
| 581 | cloud: "{{ cloud }}" | ||
| 582 | region_name: "{{ region_name }}" | ||
| 583 | name: "{{ name }}" | ||
| 584 | image_name: "{{ image_name }}" | ||
| 585 | flavor_ram: "{{ flavor_ram }}" | ||
| 586 | flavor_include: "{{ flavor_include }}" | ||
| 587 | meta: | ||
| 588 | group: "{{ group }}" | ||
| 589 | key_name: "{{ launch_keypair }}" | ||
| 590 | register: node | ||
| 591 | - name: Create volumes | ||
| 592 | os_volume: | ||
| 593 | cloud: "{{ cloud }}" | ||
| 594 | size: "{{ item.size }}" | ||
| 595 | display_name: "{{ item.display_name }}" | ||
| 596 | with_items: volumes | ||
| 597 | - name: Attach volumes | ||
| 598 | os_compute_volume: | ||
| 599 | cloud: "{{ cloud }}" | ||
| 600 | server_id: "{{ node.id }}" | ||
| 601 | volume_name: "{{ item.display_name }}" | ||
| 602 | with_items: volumes | ||
| 603 | register: attached_volumes | ||
| 604 | - debug: var=attached_volumes | ||
| 605 | - name: Re-request server to get up to date metadata after the volume loop | ||
| 606 | os_compute_facts: | ||
| 607 | cloud: "{{ cloud }}" | ||
| 608 | name: "{{ name }}" | ||
| 609 | when: attached_volumes.changed | ||
| 610 | - name: Wait for SSH to work | ||
| 611 | wait_for: host={{ node.openstack.interface_ip }} port=22 | ||
| 612 | when: node.changed == True | ||
| 613 | - name: Add SSH host key to known hosts | ||
| 614 | shell: ssh-keyscan "{{ node.openstack.interface_ip|quote }}" >> ~/.ssh/known_hosts | ||
| 615 | when: node.changed == True | ||
| 616 | - name: Add all instance public IPs to host group | ||
| 617 | add_host: | ||
| 618 | name: "{{ node.openstack.interface_ip }}" | ||
| 619 | groups: "{{ provision_group }}" | ||
| 620 | openstack: "{{ node.openstack }}" | ||
| 621 | when: attached_volumes|length == 0 | ||
| 622 | - name: Add all instance public IPs to host and volumes group | ||
| 623 | add_host: | ||
| 624 | name: "{{ node.openstack.interface_ip }}" | ||
| 625 | groups: "{{ provision_group }},hasvolumes" | ||
| 626 | openstack: "{{ node.openstack }}" | ||
| 627 | when: attached_volumes|length != 0 | ||
| 628 | </code></pre> | ||
| 629 | </section> | ||
| 630 | |||
| 631 | <section> | ||
| 632 | <h1> Cloud based inventory </h1> | ||
| 633 | <ul> | ||
| 634 | <li> Just ask the cloud for the inventory </li> | ||
| 635 | <li> All of the meta-data the cloud knows is available </li> | ||
| 636 | </ul> | ||
| 637 | </section> | ||
| 638 | |||
| 639 | <section> | ||
| 640 | <pre><code> | ||
| 641 | "pypi.dfw.openstack.org": { | ||
| 642 | "ansible_ssh_host": "23.253.237.8", | ||
| 643 | "openstack": { | ||
| 644 | "HUMAN_ID": true, | ||
| 645 | "NAME_ATTR": "name", | ||
| 646 | "OS-DCF:diskConfig": "MANUAL", | ||
| 647 | "OS-EXT-STS:power_state": 1, | ||
| 648 | "OS-EXT-STS:task_state": null, | ||
| 649 | "OS-EXT-STS:vm_state": "active", | ||
| 650 | "accessIPv4": "23.253.237.8", | ||
| 651 | "accessIPv6": "2001:4800:7817:104:d256:7a33:5187:7e1b", | ||
| 652 | "addresses": { | ||
| 653 | "private": [ | ||
| 654 | { | ||
| 655 | "addr": "10.208.195.50", | ||
| 656 | "version": 4 | ||
| 657 | } | ||
| 658 | ], | ||
| 659 | "public": [ | ||
| 660 | { | ||
| 661 | "addr": "23.253.237.8", | ||
| 662 | "version": 4 | ||
| 663 | }, | ||
| 664 | { | ||
| 665 | "addr": "2001:4800:7817:104:d256:7a33:5187:7e1b", | ||
| 666 | "version": 6 | ||
| 667 | } | ||
| 668 | ] | ||
| 669 | }, | ||
| 670 | "cloud": "rax", | ||
| 671 | "config_drive": "", | ||
| 672 | "created": "2014-09-05T15:32:14Z", | ||
| 673 | "flavor": { | ||
| 674 | "id": "performance1-4", | ||
| 675 | "links": [ | ||
| 676 | { | ||
| 677 | "href": "https://dfw.servers.api.rackspacecloud.com/610275/flavors/performance1-4", | ||
| 678 | "rel": "bookmark" | ||
| 679 | } | ||
| 680 | ], | ||
| 681 | "name": "4 GB Performance" | ||
| 682 | }, | ||
| 683 | "hostId": "adb603d4566efe0392756c76dab38ffcba22099368837c7973321e77", | ||
| 684 | "human_id": "pypidfwopenstackorg", | ||
| 685 | "id": "de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 686 | "image": { | ||
| 687 | "id": "928e709d-35f0-47eb-b296-d18e1b0a76b7", | ||
| 688 | "links": [ | ||
| 689 | { | ||
| 690 | "href": "https://dfw.servers.api.rackspacecloud.com/610275/images/928e709d-35f0-47eb-b296-d18e1b0a76b7", | ||
| 691 | "rel": "bookmark" | ||
| 692 | } | ||
| 693 | ] | ||
| 694 | }, | ||
| 695 | "interface_ip": "23.253.237.8", | ||
| 696 | "key_name": "launch-node-root", | ||
| 697 | "links": [ | ||
| 698 | { | ||
| 699 | "href": "https://dfw.servers.api.rackspacecloud.com/v2/610275/servers/de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 700 | "rel": "self" | ||
| 701 | }, | ||
| 702 | { | ||
| 703 | "href": "https://dfw.servers.api.rackspacecloud.com/610275/servers/de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 704 | "rel": "bookmark" | ||
| 705 | } | ||
| 706 | ], | ||
| 707 | "metadata": {}, | ||
| 708 | "name": "pypi.dfw.openstack.org", | ||
| 709 | "networks": { | ||
| 710 | "private": [ | ||
| 711 | "10.208.195.50" | ||
| 712 | ], | ||
| 713 | "public": [ | ||
| 714 | "23.253.237.8", | ||
| 715 | "2001:4800:7817:104:d256:7a33:5187:7e1b" | ||
| 716 | ] | ||
| 717 | }, | ||
| 718 | "progress": 100, | ||
| 719 | "region": "DFW", | ||
| 720 | "status": "ACTIVE", | ||
| 721 | "tenant_id": "610275", | ||
| 722 | "updated": "2014-09-05T15:32:49Z", | ||
| 723 | "user_id": "156284", | ||
| 724 | "volumes": [ | ||
| 725 | { | ||
| 726 | "HUMAN_ID": false, | ||
| 727 | "NAME_ATTR": "name", | ||
| 728 | "attachments": [ | ||
| 729 | { | ||
| 730 | "device": "/dev/xvdb", | ||
| 731 | "host_name": null, | ||
| 732 | "id": "c6f5229c-1cc0-47c4-aab7-60db1f6cf8e8", | ||
| 733 | "server_id": "de672205-9245-46b6-b3df-489ccf9e0c17", | ||
| 734 | "volume_id": "c6f5229c-1cc0-47c4-aab7-60db1f6cf8e8" | ||
| 735 | } | ||
| 736 | ], | ||
| 737 | "availability_zone": "nova", | ||
| 738 | "bootable": "false", | ||
| 739 | "created_at": "2014-09-05T14:37:42.000000", | ||
| 740 | "device": "/dev/xvdb", | ||
| 741 | "display_description": null, | ||
| 742 | "display_name": "pypi.dfw.openstack.org/main01", | ||
| 743 | "human_id": null, | ||
| 744 | "id": "c6f5229c-1cc0-47c4-aab7-60db1f6cf8e8", | ||
| 745 | "metadata": { | ||
| 746 | "readonly": "False", | ||
| 747 | "storage-node": "1845027a-5e07-47a1-9572-3eea4716f726" | ||
| 748 | }, | ||
| 749 | "os-vol-tenant-attr:tenant_id": "610275", | ||
| 750 | "size": 200, | ||
| 751 | "snapshot_id": null, | ||
| 752 | "source_volid": null, | ||
| 753 | "status": "in-use", | ||
| 754 | "volume_type": "SATA" | ||
| 755 | } | ||
| 756 | ] | ||
| 757 | } | ||
| 758 | }, | ||
| 759 | </code></pre> | ||
| 760 | </section> | ||
| 761 | |||
| 762 | |||
| 763 | <section> | ||
| 764 | <h1> Thank you! </h1> | ||
| 765 | <h3>Monty Taylor</h3> | ||
| 766 | <h3> twitter: @e_monty </h3> | ||
| 767 | <h4> <a href='http://inaugust.com/talks/tripleo-ansible.html'>http://inaugust.com/talks/tripleo-ansible.html</a> </h4> | ||
| 768 | </section> | ||
| 769 | |||
| 770 | </div> | ||
| 771 | |||
| 772 | <div class="footer"> | ||
| 773 | <a rel="license" href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 774 | <img alt="Creative Commons License" | ||
| 775 | style="border-width:0" | ||
| 776 | src="https://i.creativecommons.org/l/by/4.0/88x31.png" /> | ||
| 777 | </a><br /> | ||
| 778 | Licensed under a | ||
| 779 | <a rel="license" | ||
| 780 | href="http://creativecommons.org/licenses/by/4.0/"> | ||
| 781 | Creative Commons Attribution 4.0 International License | ||
| 782 | </a>. | ||
| 783 | <br /> | ||
| 784 | Source code available at <a href='http://git.inaugust.com/cgit/inaugust.com'>http://git.inaugust.com/cgit/inaugust.com</a> | ||
| 785 | </div> | ||
| 786 | </div> | ||
| 787 | |||
| 788 | <script src="/lib/js/head.min.js"></script> | ||
| 789 | <script src="/js/reveal.js"></script> | ||
| 790 | |||
| 791 | <script src="/js/this.js"></script> | ||
| 792 | |||
| 793 | </body> | ||
| 794 | </html> | ||
