From 8ab3b94833e341dc700ab1e7f62dc514848181f3 Mon Sep 17 00:00:00 2001 From: Monty Taylor <mordred@inaugust.com> Date: Tue, 9 May 2017 08:11:51 -0400 Subject: Add shade: Everything you ever wanted as a cloud user --- src/talks/everything-you-ever-wanted.hbs | 1335 ++++++++++++++++++++++++++++++ 1 file changed, 1335 insertions(+) create mode 100644 src/talks/everything-you-ever-wanted.hbs diff --git a/src/talks/everything-you-ever-wanted.hbs b/src/talks/everything-you-ever-wanted.hbs new file mode 100644 index 0000000..a5c79fc --- /dev/null +++ b/src/talks/everything-you-ever-wanted.hbs @@ -0,0 +1,1335 @@ +<!doctype html> +<html lang="en"> + + <head> + <meta charset="utf-8"> + + <title>shade: Everything You Ever Wanted As a Cloud User</title> + </head> + <body> + + <section id="who-am-i-redhat" class="slide level2"> + <h1>Who am I?</h1> + <img style="float:right; margin:24pt" src="/images/Logo_RH_CMYK_Default.jpg" /> + <p> Office of Technology </p> + <p> Zuul </p> + <p> Ansible </p> + </section> + + <section id="who-am-i-openstack" class="slide level2"> + <h1>Who am I?</h1> + <img style="float:right; margin-right:24pt; width:300px; height: auto" src="/images/openstack-cloud-software-vertical-large.png" /> + <p>Technical Committee</p> + <p>Developer Infrastructure Core Team</p> + </section> + + <section id="what-are-we-going-to-talk-about" class="slide level2"> + <h1>What are we going to talk about?</h1> + <ul> + <li>What</li> + <li>Why</li> + <li>Configuration</li> + <li>Basics</li> + <li>Advanced things + <ul> + <li>Caching</li> + <li>Task Management</li> + </ul> + </li> + </ul> + </section> + + <section class="slide level2"> + <h1>shade: a Python library to wrap business logic around + OpenStack resources and operations</h1> + </section> + + <section class="slide level2"> + <h1>Design Principles</h1> + <ul> + <li class='fragment'>Expose a single API that works on all clouds</li> + <li class='fragment'>Hide all vendor or deployer differences</li> + <li class='fragment'>Support multi-cloud (<em>write once, run anywhere</em>)</li> + <li class='fragment'>Simple to use (<em>sane defaults</em>)</li> + <li class='fragment'>No plugins *</li> + <li class='fragment'>Efficient at scale</li> + <li class='fragment'>API always backwards compatible</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Current Status</h1> + <h3>https://git.openstack.org/cgit/openstack-infra/shade</h3> + <h3>https://pypi.python.org/pypi/shade</h3> + <ul> + <li class='fragment'>Used in Ansible</li> + <li class='fragment'>Used in Infra Nodepool</li> + <li class='fragment'>Official OpenStack Project</li> + <li class='fragment'>Backend conversion to pure REST almost done</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Why?</h1> + </section> + + <section class="slide level2"> + <blockquote> + Brand experts insist that success comes from promoting your unique + attributes, but in practice differentiation is less profitable than + consolidation. + </blockquote> + </section> + + <section class="slide level2" data-transition='zoom'> + <h1>OpenStack Leaks Abstractions</h1> + </section> + + <section class="slide level2" data-transition='zoom'> + <h1>OpenStack Breaks APIs</h1> + </section> + + <section class="slide level2" data-transition='zoom'> + <h1>Basic concepts are needlessly complex</h1> + </section> + + <section class="slide level2" data-transition='zoom'> + <h1>Client libraries are really for server-server communication</h1> + </section> + + <section class="slide level2"> + <h1>Infra solved these problems</h1> + <p>Infra runs across 14 clouds at massive scale (20k servers per day)</p> + <p>Why not share what we've learned with other people?</p> + </section> + + <section class="slide level2"> + <h1>simplicity</h1> + <p>This is what using a cloud should look like</p> + <pre><code> +cloud = openstack_cloud('vexxhost') +image = cloud.create_image( + 'image-name', filename='image-filename.qcow2', wait=True) +flavor = cloud.get_flavor_by_ram(512) +cloud.create_server( + 'my-server', image=image, flavor=flavor, wait=True, auto_ip=True) + </code></pre> + </section> + + <section class="slide level2" data-transition='zoom'> + <h1>existence of shade is a bug</h1> + </section> + + <section class="slide level2" data-transition='zoom'> + <h1>existence of shade is a <em>feature</em></h1> + </section> + + <section class="slide level2"> + <h1>Using shade</h1> + <h2>Step One: Configuration</h2> + </section> + + <section class="slide level2"> + <h1>os-client-config</h1> + <h3>http://git.openstack.org/cgit/openstack/os-client-config</h3> + <ul> + <li>A library to handle config information for openstack clients</li> + <li>Tracks differences in vendors that can't be discovered</li> + <li>Also used for ansible, python-openstackclient and openstacksdk</li> + <li>Reads clouds.yaml config file, environment vars and argparse</li> + </ul> + </section> + + <section class="slide level2"> + <h1>clouds.yaml - simple</h1> + <pre><code> +clouds: + my-citycloud: + profile: citycloud + auth: + username: mordred + password: XXXXXXXXXXXXX + project_id: 65222a4d09ea4c68934fa1028c77f394 + user_domain_id: d0919bd5e8d74e49adf0e145807ffc38 + project_domain_id: d0919bd5e8d74e49adf0e145807ffc38 + </code></pre> + </section> + + <section class="slide level2"> + <h1>clouds.yaml - mildly complex</h1> + <pre><code> + my-vexxhost: + volume_api_version: 3 + profile: vexxhost + auth: + user_domain_id: default + project_domain_id: default + project_name: d8af8a8f-a573-48e6-898a-af333b970a2d + username: 0b8c435b-cc4d-4e05-8a47-a2ada0539af1 + password: XXXXXXXXXXXXX + regions: + - ca-ymq-1 + image_endpoint_override: https://image-ca-ymq-1.vexxhost.net/v2 + </code></pre> + </section> + + <section class="slide level2"> + <h1>clouds.yaml - complex</h1> + <pre><code> + my-internap: + auth: + auth_url: https://identity.api.cloud.iweb.com + username: api-55f9a00fb2619 + project_name: inap-17037 + password: XXXXXXXXXXXXX + identity_api_version: 3 + floating_ip_source: None + regions: + - name: ams01 + values: + networks: + - name: inap-17037-WAN1654 + routes_externally: true + default_interface: true + - name: inap-17037-LAN3631 + routes_externally: false + </code></pre> + </section> + + <section class="slide level2"> + <h1>Environment Variables</h1> + <ul> + <li>os-client-config also processes OS_ environment variables</li> + <li>If present, they go into a cloud named "envvars"</li> + </ul> + </section> + + <section class="slide level2"> + <h1>auth_type</h1> + <p>keystone has pluggable authentication</p> + + <ul> + <li>auth dict contents vary based on plugin</li> + <li>Defaults to 'password' which autodetects based on parameters</li> + <li>Other options: admin_token, v3oidcpassword, v3oidcauthcode</li> + <li>This is where I said "*" on "no plugins" earlier</li> + </ul> + </section> + + <section class="slide level2"> + <h1>auth_type - not part of auth_dict</h1> + <pre><code> + my-vexxhost: + profile: vexxhost + auth_type: v3password + auth: + user_domain_id: default + project_domain_id: default + project_name: d8af8a8f-a573-48e6-898a-af333b970a2d + username: 0b8c435b-cc4d-4e05-8a47-a2ada0539af1 + password: XXXXXXXXXXXXX + regions: + - ca-ymq-1 + </code></pre> + </section> + + <section class="slide level2"> + <h1>python-openstackclient</h1> + <pre> +openstack --os-cloud=my-vexxhost servers list + </pre> + Or + <pre> +export OS_CLOUD=my-vexxhost +openstack servers list + </pre> + </section> + + <section class="slide level2"> + <h1>shade inventory</h1> + <ul> + <li>Code behind ansible OpenStack dynamic inventory plugin</li> + <li>All resources in all of your clouds</li> + </ul> + <pre> +mordred@camelot:~$ shade-inventory --list --yaml + </pre> + <pre><code> +- accessIPv4: 199.204.45.50 + accessIPv6: 2604:e100:1:0:f816:3eff:feb5:ce98 + addresses: + public: + - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:b5:ce:98 + OS-EXT-IPS:type: fixed + addr: 2604:e100:1:0:f816:3eff:feb5:ce98 + version: 6 + - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:b5:ce:98 + OS-EXT-IPS:type: fixed + addr: 199.204.45.50 + version: 4 + adminPass: null + created: '2017-04-02T17:04:49Z' + disk_config: AUTO + flavor: + id: bbcb7eb5-5c8d-498f-9d7e-307c575d3566 + name: v1-standard-1 + has_config_drive: false + host_id: 5b303fdae29a6b2c39c90a2a1c6dc1191297840d8c395e09c4e319b7 + id: 5b3cde7d-ad17-4b28-9c08-9f4627abc659 + image: + id: dd33a4c8-5a86-40ec-9fe0-8578d6a6dc10 + name: Ubuntu 16.04.1 LTS [2017-03-03] + interface_ip: 199.204.45.50 + key_name: mordred + launched_at: '2017-04-02T17:05:02.000000' + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: ca-ymq-2 + metadata: {} + name: irc-bouncer + networks: + public: + - 2604:e100:1:0:f816:3eff:feb5:ce98 + - 199.204.45.50 + power_state: 1 + private_v4: '' + progress: 0 + properties: {} + public_v4: 199.204.45.50 + public_v6: 2604:e100:1:0:f816:3eff:feb5:ce98 + security_groups: + - description: default + id: 22f8104a-2359-4091-aa61-9a4c48d6df3f + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: null + name: default + properties: {} + security_group_rules: + - direction: ingress + ethertype: IPv4 + id: 2172daaa-e402-4cf0-b353-1a7cb7dae184 + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: null + port_range_max: null + port_range_min: null + properties: + group: {} + protocol: null + remote_group_id: null + remote_ip_prefix: 0.0.0.0/0 + security_group_id: 22f8104a-2359-4091-aa61-9a4c48d6df3f + - direction: ingress + ethertype: IPv4 + id: 47fb6b4c-2f9b-4279-a77c-a645f7fc964f + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: null + port_range_max: null + port_range_min: null + properties: + group: + name: default + tenant_id: db92b20496ae4fbda850a689ea9d563f + protocol: null + remote_group_id: null + remote_ip_prefix: null + security_group_id: 22f8104a-2359-4091-aa61-9a4c48d6df3f + - direction: ingress + ethertype: IPv4 + id: 8287f9be-614b-4145-9f00-62fa9c856362 + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: null + port_range_max: null + port_range_min: null + properties: + group: + name: default + tenant_id: db92b20496ae4fbda850a689ea9d563f + protocol: null + remote_group_id: null + remote_ip_prefix: null + security_group_id: 22f8104a-2359-4091-aa61-9a4c48d6df3f + - direction: ingress + ethertype: IPv4 + id: fdd91863-c19b-4b4b-a1bc-822cb8b0adf7 + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: null + port_range_max: null + port_range_min: null + properties: + group: {} + protocol: null + remote_group_id: null + remote_ip_prefix: 0.0.0.0/0 + security_group_id: 22f8104a-2359-4091-aa61-9a4c48d6df3f + status: ACTIVE + task_state: null + terminated_at: null + updated: '2017-04-02T17:05:02Z' + user_id: e9b21dc437d149858faee0898fb08e92 + vm_state: active + volumes: + - attachments: + - attached_at: '2017-04-11T19:11:09.000000' + attachment_id: 60f81620-4714-438b-b481-cd8fd7a4dd3e + device: /dev/vdc + host_name: null + id: 7aa81299-bb1c-449b-8a73-b55a9bdc5fca + server_id: 5b3cde7d-ad17-4b28-9c08-9f4627abc659 + volume_id: 7aa81299-bb1c-449b-8a73-b55a9bdc5fca + can_multiattach: false + consistencygroup_id: null + created_at: '2017-04-11T19:10:20.000000' + description: '' + device: /dev/vdc + host: null + id: 7aa81299-bb1c-449b-8a73-b55a9bdc5fca + is_bootable: false + is_encrypted: false + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: ca-ymq-2 + metadata: + attached_mode: rw + readonly: 'False' + migration_status: null + name: foo + properties: + volume_image_metadata: {} + replication_driver: null + replication_extended_status: null + replication_status: disabled + size: 1 + snapshot_id: null + source_volume_id: null + status: in-use + updated_at: '2017-04-11T19:11:09.000000' + volume_type: null + - attachments: + - attached_at: '2017-04-02T17:06:27.000000' + attachment_id: 4f24b276-ea5c-4a79-824f-434183c3c518 + device: /dev/vdb + host_name: null + id: 9b6110df-cd97-4e37-9596-af371653a791 + server_id: 5b3cde7d-ad17-4b28-9c08-9f4627abc659 + volume_id: 9b6110df-cd97-4e37-9596-af371653a791 + can_multiattach: false + consistencygroup_id: null + created_at: '2017-04-02T16:34:41.000000' + description: '' + device: /dev/vdb + host: null + id: 9b6110df-cd97-4e37-9596-af371653a791 + is_bootable: true + is_encrypted: false + location: + cloud: my-vexxhost + project: + domain_id: default + domain_name: null + id: db92b20496ae4fbda850a689ea9d563f + name: d8af8a8f-a573-48e6-898a-af333b970a2d + region_name: ca-ymq-1 + zone: ca-ymq-2 + metadata: + attached_mode: rw + readonly: 'False' + migration_status: null + name: mordred-irc-save + properties: + volume_image_metadata: + base_image_ref: 69c99b45-cd53-49de-afdc-f24789eb8f83 + checksum: dddc88434b12b5b61b6c7a3a5a880c35 + clean_attempts: '7' + container_format: bare + disk_format: raw + hw_rng_model: virtio + image_id: 1659da1f-7650-4b91-b316-c3d69623e3fc + image_location: snapshot + image_name: mordred-irc-save + image_state: available + image_type: snapshot + instance_uuid: 811c5197-dba7-4d3a-a3f6-68ca5328b9a7 + libvirt_use_agent: 'True' + min_disk: '40' + min_ram: '0' + network_allocated: 'True' + os_type: linux + owner_id: db92b20496ae4fbda850a689ea9d563f + size: '42949672960' + user_id: e9b21dc437d149858faee0898fb08e92 + replication_driver: null + replication_extended_status: null + replication_status: disabled + size: 40 + snapshot_id: null + source_volume_id: null + status: in-use + updated_at: '2017-04-02T17:06:27.000000' + volume_type: null +- accessIPv4: 89.40.216.229 + accessIPv6: '' + addresses: + private: + - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:29:6d:20 + OS-EXT-IPS:type: fixed + addr: 10.4.0.22 + version: 4 + - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:29:6d:20 + OS-EXT-IPS:type: floating + addr: 89.40.216.229 + version: 4 + adminPass: null + created: '2017-05-08T18:48:34Z' + disk_config: MANUAL + flavor: + id: 10973fe6-81ba-480c-bc36-bec93ee03d5c + name: 4C-4GB-100GB + has_config_drive: false + host_id: 8ed21447e12616c0c24f6e246f0ec423d6be1532716869a3617fbe80 + id: a72fd3ea-9357-482d-b33a-fe61d7c37bfc + image: + id: 95e4c449-8abf-486e-97d9-dc3f82417d2d + name: Ubuntu 16.04 Xenial Xerus + interface_ip: 89.40.216.229 + key_name: null + launched_at: '2017-05-08T18:48:39.000000' + location: + cloud: my-citycloud + project: + domain_id: d0919bd5e8d74e49adf0e145807ffc38 + domain_name: null + id: 65222a4d09ea4c68934fa1028c77f394 + name: null + region_name: Buf1 + zone: nova + metadata: {} + name: my-server + networks: + private: + - 10.4.0.22 + - 89.40.216.229 + power_state: 1 + private_v4: 10.4.0.22 + progress: 0 + properties: {} + public_v4: 89.40.216.229 + public_v6: '' + security_groups: + - description: Default security group + id: 9fb5ba44-5c46-4357-8e60-8b55526cab54 + location: + cloud: my-citycloud + project: + domain_id: d0919bd5e8d74e49adf0e145807ffc38 + domain_name: null + id: 65222a4d09ea4c68934fa1028c77f394 + name: null + region_name: Buf1 + zone: null + name: default + properties: {} + security_group_rules: + - direction: ingress + ethertype: IPv4 + id: 67cbd716-ace1-4822-93e3-1a14a2763efa + location: + cloud: my-citycloud + project: + domain_id: d0919bd5e8d74e49adf0e145807ffc38 + domain_name: null + id: 65222a4d09ea4c68934fa1028c77f394 + name: null + region_name: Buf1 + zone: null + port_range_max: null + port_range_min: null + properties: + group: + name: default + tenant_id: 65222a4d09ea4c68934fa1028c77f394 + protocol: null + remote_group_id: null + remote_ip_prefix: null + security_group_id: 9fb5ba44-5c46-4357-8e60-8b55526cab54 + - direction: ingress + ethertype: IPv4 + id: f5e80c4c-fe7a-4bfc-a8c0-0105b2c272d5 + location: + cloud: my-citycloud + project: + domain_id: d0919bd5e8d74e49adf0e145807ffc38 + domain_name: null + id: 65222a4d09ea4c68934fa1028c77f394 + name: null + region_name: Buf1 + zone: null + port_range_max: null + port_range_min: null + properties: + group: + name: default + tenant_id: 65222a4d09ea4c68934fa1028c77f394 + protocol: null + remote_group_id: null + remote_ip_prefix: null + security_group_id: 9fb5ba44-5c46-4357-8e60-8b55526cab54 + status: ACTIVE + task_state: null + terminated_at: null + updated: '2017-05-08T18:48:39Z' + user_id: c17534835f8f42bf98fc367e0bf35e09 + vm_state: active + volumes: [] +- accessIPv4: 173.231.180.251 + accessIPv6: '' + addresses: + inap-17037-WAN1654: + - OS-EXT-IPS-MAC:mac_addr: fa:16:3e:6b:97:9c + OS-EXT-IPS:type: fixed + addr: 173.231.180.251 + version: 4 + adminPass: null + created: '2017-05-08T18:48:56Z' + disk_config: null + flavor: + id: A1.4 + name: A1.4 + has_config_drive: true + host_id: 6d0c22f3176c3bcbb78a418a93809ba6bfa16267a4c38bd614a892d9 + id: ba864796-b56a-4154-b220-e1b96daf39c0 + image: + id: a6fefc9b-c208-4cd0-acc3-2ab09a38054b + name: Ubuntu 16.04 LTS (Xenial Xerus) + interface_ip: 173.231.180.251 + key_name: null + launched_at: null + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + metadata: {} + name: my-server + networks: + inap-17037-WAN1654: + - 173.231.180.251 + power_state: 1 + private_v4: '' + progress: 0 + properties: {} + public_v4: 173.231.180.251 + public_v6: '' + security_groups: + - description: default + id: 69871901-68c6-486b-9a1b-6a9adb88ea23 + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + name: default + properties: {} + security_group_rules: + - direction: ingress + ethertype: IPv4 + id: 24b2ce3a-bf29-44fb-849f-f958e3efb736 + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + port_range_max: 65535 + port_range_min: 1 + properties: + group: {} + protocol: tcp + remote_group_id: null + remote_ip_prefix: 0.0.0.0/0 + security_group_id: 69871901-68c6-486b-9a1b-6a9adb88ea23 + - direction: ingress + ethertype: IPv4 + id: 374aa520-c101-4de3-be88-f0025396bdb7 + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + port_range_max: null + port_range_min: null + properties: + group: + name: default + tenant_id: 760e6c137b3840d78472d313dfa3df45 + protocol: null + remote_group_id: null + remote_ip_prefix: null + security_group_id: 69871901-68c6-486b-9a1b-6a9adb88ea23 + - direction: ingress + ethertype: IPv4 + id: 376b67c2-284b-4a81-a989-e7b8f6d44a07 + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + port_range_max: 65535 + port_range_min: 1 + properties: + group: {} + protocol: udp + remote_group_id: null + remote_ip_prefix: 0.0.0.0/0 + security_group_id: 69871901-68c6-486b-9a1b-6a9adb88ea23 + - direction: ingress + ethertype: IPv4 + id: 481f9402-753d-4286-8079-be45a414023c + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + port_range_max: null + port_range_min: null + properties: + group: {} + protocol: icmp + remote_group_id: null + remote_ip_prefix: 0.0.0.0/0 + security_group_id: 69871901-68c6-486b-9a1b-6a9adb88ea23 + - direction: ingress + ethertype: IPv4 + id: 832aaf3b-b0fb-4103-b2fa-def955077667 + location: + cloud: my-internap + project: + domain_id: null + domain_name: null + id: 760e6c137b3840d78472d313dfa3df45 + name: inap-17037 + region_name: ams01 + zone: null + port_range_max: null + port_range_min: null + properties: + group: + name: default + tenant_id: 760e6c137b3840d78472d313dfa3df45 + protocol: null + remote_group_id: null + remote_ip_prefix: null + security_group_id: 69871901-68c6-486b-9a1b-6a9adb88ea23 + status: ACTIVE + task_state: null + terminated_at: null + updated: '2017-05-08T18:49:05Z' + user_id: c18bd37c27d14b1ba4c424e6d42f2ccd + vm_state: active + volumes: [] + </code></pre> + </section> + + <section class="slide level2"> + <h1>Cloud-Region</h1> + <p>Each OpenStackCloud object represents one region of one cloud</p> + </section> + + <section class="slide level2"> + <h1>Simplest Cloud Construction</h1> + <ul> + <li>Works if you only have one cloud</li> + </ul> + + <pre><code> +import shade +cloud = shade.openstack_cloud() + </code></pre> + </section> + + <section class="slide level2"> + <h1>Simplest Cloud Construction - More than one</h1> + <pre><code> +clients: + default: + cloud: vexxhost + </code></pre> + + <pre><code> +import shade +cloud = shade.openstack_cloud() + </code></pre> + </section> + + <section class="slide level2"> + <h1>Simple Cloud Construction</h1> + <pre><code> +import shade +cloud = shade.openstack_cloud(cloud='ustack', region_name='bj1') + </code></pre> + </section> + + <section class="slide level2"> + <h1>Complex Cloud Construction</h1> + <pre><code> +import os_client_config +import shade + +config = os_client_config.OpenStackConfig() +cloud_config = config.get_one_cloud( + cloud='ustack', region_name='bj1', + argparse=my_args, **other_arguments) +cloud = shade.OpenStackCloud(cloud_config=cloud_config) + </code></pre> + </section> + + <section class="slide level2"> + <h1>logging</h1> + <ul> + <li>Python logging</li> + <li>shade.simple_logging() helper function</li> + <li>Turns off annoying warnings</li> + <li>debug=True - turn on debug logging and request_id logging</li> + <li>http_debug=True - debug=True + http request tracing</li> + </ul> + </section> + + <section class="slide level2"> + <h1>A note on Exceptions</h1> + <ul> + <li>All (remaining) python*client exceptions are hidden and re-raised</li> + <li>Hiding exceptions is evil and I'm a bad person</li> + <li>Exceptions are part of the interface</li> + <li>Can't hide vendor choice if shade user catches underlying exception</li> + <li>See also: Restification later</li> + </section> + + <section class="slide level2"> + <h1>Exceptions</h1> + <ul> + <li>All Exceptions are subclasses of OpenStackCloudException</li> + <li>Direct REST calls throw OpenStackCloudHTTPError</li> + <li>OpenStackCloudHTTPError subclasses OpenStackCloudException + and requests.exceptions.HTTPError</li> + <li>OpenStackCloudURINotFound for 404</li> + <li>OpenStackCloudBadRequest for 400</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Putting it all together</h1> + <pre><code> +import shade + +shade.simple_logging(debug=True) +cloud = shade.openstack_cloud(cloud='vexxhost') +image = cloud.create_image( + 'ubuntu-trusty', filename='ubuntu-trusty.qcow2', wait=True) +flavor = cloud.get_flavor_by_ram(512) +cloud.create_server( + 'my-server', image=image['id'], flavor=flavor['id'], + wait=True, auto_ip=True) + </code></pre> + </section> + + <section class="slide level2"> + <h1>Problem: Image API version</h1> + <ul> + <li class='fragment'> + v1 PUT: Catalyst, Datacentred, Internap + </li> + <li class='fragment'> + v2 PUT: Auro, City Cloud, Dreamhost, Elastx, Enter Cloud Suite, OVH, RunAbove, Vexxhost, Ultimum, UnitedStack + </li> + <li class='fragment'> + v2 Tasks: Switch Engines, Rackspace + </li> + </ul> + </section> + + <section class="slide level2"> + <h1>Image Upload Code</h1> + <pre><code> +cloud = shade.openstack_cloud(cloud='vexxhost', region_name='ca-ymq-1') +cloud.create_image('my-image', filename='my-image.qcow2', wait=True) + </code></pre> + </section> + + <section class="slide level2"> + <h1>Image Upload PUT v2 Log</h1> + <pre><code> +Calculating hashes for my-image.qcow2 +Image file my-image.vhd md5:b28fa44077b94ba635842a73eecc1d8c sha256:9a3da71e37ef1579657c4a77ff2caf936dd9be8361d1a02595cc116f0ffd328d +Manager my-vexxhost:ca-ymq-1 running task image.GET.images +Manager my-vexxhost:ca-ymq-1 ran task image.GET.images in 1.38961982727s +GET call to image for https://image-ca-ymq-1.vexxhost.net/v2/images used request id req-2bbc8c10-6c4c-4aa3-8bb7-6ba425cb0ea7 +Manager my-vexxhost:ca-ymq-1 running task image.GET.images +Manager my-vexxhost:ca-ymq-1 ran task image.GET.images in 0.245140790939s +GET call to image for https://image-ca-ymq-1.vexxhost.net/v2/images?marker=5ebb74ba-c9c5-4006-8c77-ceebf9833428 used request id req-da204dc3-eb07-460d-b9f5-74b6ea5fe944 +Manager my-vexxhost:ca-ymq-1 running task image.POST.images +Manager my-vexxhost:ca-ymq-1 ran task image.POST.images in 0.114197969437s +POST call to image for https://image-ca-ymq-1.vexxhost.net/v2/images used request id req-02a5fd04-ca9a-4a71-bfe1-4ff80e49cdea returning object 28d3e4b2-4607-4778-84bd-e84ef217d21c +Manager my-vexxhost:ca-ymq-1 running task image.PUT.images.file +Manager my-vexxhost:ca-ymq-1 ran task image.PUT.images.file in 2.10132312775s +PUT call to image for https://image-ca-ymq-1.vexxhost.net/v2/images/28d3e4b2-4607-4778-84bd-e84ef217d21c/file used request id req-71e3ed5e-77dc-4062-9ba8-13d495713e13 + </code></pre> + </section> + + <section class="slide level2"> + <h1>Image Tasks</h1> + <pre><code> +cloud = shade.openstack_cloud(cloud='rax', region_name='DFW') +cloud.create_image('my-image', filename='my-image.<span style='color: red'>vhd</span>', wait=True) + </code></pre> + </section> + + + <section class="slide level2"> + <h1>Image Upload Tasks Log</h1> + <pre><code> +Calculating hashes for my-image.vhd +Image file my-image.vhd md5:b28fa44077b94ba635842a73eecc1d8c sha256:9a3da71e37ef1579657c4a77ff2caf936dd9be8361d1a02595cc116f0ffd328d +Manager rax:DFW running task image.GET.discovery +Manager rax:DFW ran task image.GET.discovery in 1.65571808815s +Glance version discovery failed, assuming endpoint in the catalog is already versioned. (404) Client Error: Not Found for https://dfw.images.api.rackspacecloud.com/v2/ +Manager rax:DFW running task image.GET.images +Manager rax:DFW ran task image.GET.images in 0.637840032578s +GET call to image for https://dfw.images.api.rackspacecloud.com/v2/images used request id req-b716bd86-121f-4cdc-b386-2f50546e8d88 +Manager rax:DFW running task image.GET.images +Manager rax:DFW ran task image.GET.images in 0.491132974625s +GET call to image for https://dfw.images.api.rackspacecloud.com/v2/v2/images?marker=e389ab51-1327-488a-867c-4d7c14e6c916 used request id req-90746cbf-610a-4c3d-aab8-f81e673b315a +Manager rax:DFW running task image.GET.images +Manager rax:DFW ran task image.GET.images in 0.371850967407s +GET call to image for https://dfw.images.api.rackspacecloud.com/v2/v2/images?marker=3eba4fbb-51da-4233-b699-8a4030561add used request id req-00937f32-f573-4ac3-86d5-0aeada7432d1 +Manager rax:DFW running task object-store.GET.info +Manager rax:DFW ran task object-store.GET.info in 0.335763931274s +Manager rax:DFW running task object-store.HEAD.images +Manager rax:DFW ran task object-store.HEAD.images in 0.393274784088s +Manager rax:DFW running task object-store.HEAD.images +Manager rax:DFW ran task object-store.HEAD.images in 3.68135595322s +swift stale check, no object: images/my-image +swift uploading my-image.vhd to images/my-image +Manager rax:DFW running task object-store.PUT.images +Manager rax:DFW ran task object-store.PUT.images in 3.57636904716s +Manager rax:DFW running task image.GET.images +Manager rax:DFW ran task image.GET.images in 0.616628885269s +GET call to image for https://dfw.images.api.rackspacecloud.com/v2/images used request id req-ee7091ae-88a2-4cdb-ac8e-35159ef4cda0 +Manager rax:DFW running task image.GET.images +Manager rax:DFW ran task image.GET.images in 0.527029037476s +GET call to image for https://dfw.images.api.rackspacecloud.com/v2/v2/images?marker=e389ab51-1327-488a-867c-4d7c14e6c916 used request id req-d2caefd2-d69f-46d2-86c1-a062f039ce74 +Manager rax:DFW running task image.GET.images +Manager rax:DFW ran task image.GET.images in 0.317733049393s +GET call to image for https://dfw.images.api.rackspacecloud.com/v2/v2/images?marker=3eba4fbb-51da-4233-b699-8a4030561add used request id req-56411197-65ce-4ac9-847d-0a7e7041766d +Manager rax:DFW running task image.POST.tasks +Manager rax:DFW ran task image.POST.tasks in 0.796432971954s +POST call to image for https://dfw.images.api.rackspacecloud.com/v2/tasks used request id req-edc7842f-5fca-4a08-96b3-55a67c840d16 returning object 45df1ffa-3837-4fbc-a8fb-f213f845b18c + </code></pre> + </section> + + <section class="slide level2"> + <h1>Problem: Version Discovery</h1> + <ul> + <li>OpenStack Services have version discovery documents</li> + <li>Deployers register versioned endpoints in catalog</li> + <li>Some services (cinder, mistral) have versioned service types (volumev2, workflowv2)</li> + <li>Users can't (sanely) do version discovery</li> + <li>shade fixes this for glance and cinder today</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Version Discovery Solutions</h1> + <ul> + <li>shade will fix for everything by end of the month</li> + <li>OpenStack Service Types Authority</li> + <li>API-WG specs on Full Service Type and Version Discovery Process</li> + <li>Work with other languages to implement API-WG specs</li> + <li>User requests:<ul> + <li>image version 2</li> + <li>workflow LATEST</li> + <li>compute</li> + <li>volume versions 2 or 3</li> + </ul></li> + <li>Later today we'll be discussing moving this forward (Room 102 at 4:40)</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Problem: Networking choices</h1> + <ul> + <li class='fragment'>Cloud has externally routable IP from neutron (OVH)</li> + <li class='fragment'>Cloud has externally routable IP neutron AND supports optional private tenant networks (vexxhost)</li> + <li class='fragment'>Cloud has private tenant network provided by neutron and requires floating IP (citycloud)</li> + <li class='fragment'>Cloud has private tenant network provided by nova-network and requires floating-ip for external routing (auro)</li> + <li class='fragment'>Cloud has externally routable IP from neutron but no neutron APIs (Rackspace)</li> + </ul> + </section> + + <section class="slide level2"> + <h1>The Floating IP Case</h1> + <pre><code> +import shade + +shade.simple_logging(debug=True) +cloud = shade.openstack_cloud(cloud='my-citycloud') +cloud.create_server( + 'my-server', + image='Ubuntu 16.04 Xenial Xerus', + flavor=dict(id='0dab10b5-42a2-438e-be7b-505741a7ffcc'), + wait=True, auto_ip=True) + </code></pre> + </section> + + <section class="slide level2"> + <h1>The Floating IP Case Log</h1> + <pre><code> +Manager my-citycloud:Buf1 running task network.GET.networks +Manager my-citycloud:Buf1 ran task network.GET.networks in 1.29657912254s +GET call to network for https://buf1.citycloud.com:9696/v2.0/networks.json used request id req-a65496e6-984f-4323-98d5-3adb339875c6 +Manager my-citycloud:Buf1 running task network.GET.subnets +Manager my-citycloud:Buf1 ran task network.GET.subnets in 0.713509082794s +GET call to network for https://buf1.citycloud.com:9696/v2.0/subnets.json used request id req-66ccbfce-ceb6-4e0a-a969-0eaf62605d92 +Manager my-citycloud:Buf1 running task image.GET.discovery +Manager my-citycloud:Buf1 ran task image.GET.discovery in 0.474808931351s +Manager my-citycloud:Buf1 running task image.GET.images +Manager my-citycloud:Buf1 ran task image.GET.images in 0.714179039001s +GET call to image for https://buf1.citycloud.com:9292/v2/images used request id req-9a7756c3-2ee4-416d-a6ad-804027330e5f +Manager my-citycloud:Buf1 running task ServerCreate +Manager my-citycloud:Buf1 ran task ServerCreate in 3.38534998894s +Manager my-citycloud:Buf1 running task ServerList +Manager my-citycloud:Buf1 ran task ServerList in 0.959898948669s +Waiting 2.0 seconds +Manager my-citycloud:Buf1 running task ServerList +Manager my-citycloud:Buf1 ran task ServerList in 1.21962594986s +Manager my-citycloud:Buf1 running task network.GET.ports +Manager my-citycloud:Buf1 ran task network.GET.ports in 0.287118911743s +GET call to network for https://buf1.citycloud.com:9696/v2.0/ports.json?device_id=1e2211f2-993f-4ba0-aa54-01743cc49181 used request id req-fd5acf95-21bd-40bb-ad08-e15ac7107874 +Manager my-citycloud:Buf1 running task network.GET.floatingips +Manager my-citycloud:Buf1 ran task network.GET.floatingips in 0.228282928467s +GET call to network for https://buf1.citycloud.com:9696/v2.0/floatingips.json?port_id=af554254-ac9f-4440-b93a-e4ec64783758 used request id req-ed52ca15-9924-488e-a3c0-3a805348eca3 +Manager my-citycloud:Buf1 running task network.GET.ports +Manager my-citycloud:Buf1 ran task network.GET.ports in 0.200226068497s +GET call to network for https://buf1.citycloud.com:9696/v2.0/ports.json?device_id=1e2211f2-993f-4ba0-aa54-01743cc49181 used request id req-f808b9e0-d136-4bd7-a51b-b08d3c6eddb3 +Manager my-citycloud:Buf1 running task network.GET.floatingips +Manager my-citycloud:Buf1 ran task network.GET.floatingips in 0.140635967255s +GET call to network for https://buf1.citycloud.com:9696/v2.0/floatingips.json used request id req-880919b7-4af3-42a3-8e07-4b80a6ae5bac +Manager my-citycloud:Buf1 running task network.GET.ports +Manager my-citycloud:Buf1 ran task network.GET.ports in 0.317744970322s +GET call to network for https://buf1.citycloud.com:9696/v2.0/ports.json?device_id=1e2211f2-993f-4ba0-aa54-01743cc49181 used request id req-af4bb447-e157-44c1-a727-2cecfb595c0e +Manager my-citycloud:Buf1 running task network.PUT.floatingips +Manager my-citycloud:Buf1 ran task network.PUT.floatingips in 0.425180912018s +PUT call to network for https://buf1.citycloud.com:9696/v2.0/floatingips/e69179dc-a904-4c9a-a4c9-891e2ecb984c.json used request id req-2241d8b6-bf50-41c1-9129-c8cfca2c73bc returning object e69179dc-a904-4c9a-a4c9-891e2ecb984c +Manager my-citycloud:Buf1 running task ServerList +Manager my-citycloud:Buf1 ran task ServerList in 1.19975018501s +Manager my-citycloud:Buf1 running task network.GET.ports +Manager my-citycloud:Buf1 ran task network.GET.ports in 0.184152126312s +GET call to network for https://buf1.citycloud.com:9696/v2.0/ports.json?device_id=1e2211f2-993f-4ba0-aa54-01743cc49181 used request id req-9f49b751-d741-45ff-a5fc-ca750826df63 +Manager my-citycloud:Buf1 running task network.GET.floatingips +Manager my-citycloud:Buf1 ran task network.GET.floatingips in 0.21661400795s +GET call to network for https://buf1.citycloud.com:9696/v2.0/floatingips.json?port_id=af554254-ac9f-4440-b93a-e4ec64783758 used request id req-81e231c6-a4d5-466a-bffd-0af6f4dc4be8 + </code></pre> + </section> + + <section class="slide level2"> + <h1>The Direct Attached IP Case</h1> + <pre><code> +import shade + +shade.simple_logging(debug=True) +cloud = shade.openstack_cloud(cloud='internap', region_name='ams01') +cloud.create_server( + 'my-server', + image='Ubuntu 16.04 LTS (Xenial Xerus)', + flavor=dict(id='A1.4'), + wait=True, auto_ip=True) + </code></pre> + </section> + + <section class="slide level2"> + <h1>Direct Attached IP Log</h1> + <pre><code> +Manager my-internap:ams01 running task network.GET.networks +Manager my-internap:ams01 ran task network.GET.networks in 0.783267021179s +GET call to network for https://network.api.ams01.cloud.iweb.com/v2.0/networks.json used request id req-807686be-238a-46e1-b0e5-5dfa31949316 +Manager my-internap:ams01 running task network.GET.subnets +Manager my-internap:ams01 ran task network.GET.subnets in 0.155231952667s +GET call to network for https://network.api.ams01.cloud.iweb.com/v2.0/subnets.json used request id req-9e71772d-70dc-4dc9-8da4-734639a10dd4 +Manager my-internap:ams01 running task image.GET.discovery +Manager my-internap:ams01 ran task image.GET.discovery in 0.589816093445s +Manager my-internap:ams01 running task image.GET.images +Manager my-internap:ams01 ran task image.GET.images in 0.568889856339s +GET call to image for https://image.api.ams01.cloud.iweb.com/v2/images used request id req-b0e25143-a581-4c6a-baab-2376626e529a +Manager my-internap:ams01 running task ServerCreate +Manager my-internap:ams01 ran task ServerCreate in 1.55480098724s +Manager my-internap:ams01 running task ServerList +Manager my-internap:ams01 ran task ServerList in 0.316590070724s +Waiting 2.0 seconds +Manager my-internap:ams01 running task ServerList +Manager my-internap:ams01 ran task ServerList in 0.317390918732s + </code></pre> + </section> + + <section class="slide level2"> + <h1>Problem: Finding an Image</h1> + <ul> + <li>Ubuntu 16.04.1 LTS [2017-03-03]</li> + <li>Ubuntu 16.04 Xenial Xerus</li> + <li>Ubuntu 16.04 LTS (Xenial Xerus)</li> + <li>Easy for a human - hard for a computer</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Solution: Sorry - no good solution yet</h1> + <ul> + <li>Always Upload your Own Images</li> + <li>Discussed in Glance Deployer Session Yesterday</li> + <li>Next step - collecting full set of User Stories</li> + <li>Hopefully we'll define common metadata</li> + <li>Hopefully we'll define "correct" image lifecycle</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Problem: Crazy Dependencies</h1> + <ul> + <li>shade started life depending on python-*client</li> + <li>Giant dependency chain</li> + <li>Users should ALWAYS use the latest shade</li> + <li>Conflicting depends makes this hard</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Solution: 'RESTification'</h1> + <ul> + <li>Have been hard at work moving to REST</li> + <li>Done with heat, magnum, swift, glance, trove</li> + <li>cinder and neutron done but unreleased</li> + <li>nova, ironic, designate left</li> + <li>Thanks Rosario and Slaweq!!!</li> + </ul> + <h2>BTW - The code is easier with REST</h2> + </section> + + <section class="slide level2"> + <h1>Advanced Things</h1> + <ul> + <li>shade should DTRT 95% of the time</li> + <li>Sometimes, you need to be extra advanced</li> + <li>We want to suppor that too</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Task Manager</h1> + <pre> +Manager zetta ran task ServerGet in 0.726951122284s + </pre> + <ul> + <li>Every API operation is a Task() that is run by a TaskManager()</li> + <li>Default shade TaskManager immediately executes the call</li> + <li>nodepool passes in a threaded TaskManager to manage API throttling</li> + <pre><code> +cloud = shade.OpenStackCloud( + cloud_config=cloud_config, + manager=MyTaskManager()) + </code></pre> + </ul> + </section> + + <section class="slide level2"> + <h1>list vs. get</h1> + <ul> + <li>get_server is a wrapper around list_servers</li> + <li>*sometimes* we can push filter conditions to the cloud</li> + <li>Actually done in SUPPORT of scaling</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Caching and Rate Limiting</h1> + <ul> + <li>API Operations can be expensive</li> + <li>Support grew from nodepool's needs</li> + <li>shade uses dogpile.cache - default to NullCache</li> + </li> + </section> + + <section class="slide level2"> + <h1>Cache Config</h1> + <h2>clouds.yaml</h2> + <pre><code> +cache: + class: dogpile.cache.dbm + expiration_time: 3600 + arguments: + filename: /home/mordred/.cache/openstack/shade.dbm + expiration: + server: 5 + </code></pre> + </section> + + <section class="slide level2"> + <h1>Caching Special Case</h1> + <ul> + <li>servers, ports and floating-ips have custome caching</li> + <li>Specifically done for nodepool</li> + <li>After RESTification, will migrate them to dogpile</li> + <li>Two levels - caching and rate-limiting</li> + </ul> + </section> + + <section class="slide level2"> + <h1>What if you're not a Python developer?</h1> + <p class='fragment'>Yes, we love you too</p> + </section> + + <section class="slide level2"> + <h1>ansible</h1> + <pre><code> +- os_image: + cloud: vexxhost + name: centos-7 + file: centos-7.qcow2 + wait: true +- os_server: + cloud: vexxhost + name: my-server + flavor_ram: 1024 + image: centos-7 + wait: true + </code></pre> + </section> + + <section class="slide level2"> + <h1>ansible</h1> + <h2>Add my keypair to 30 Regions in 13 clouds<h2> + <pre><code> +- os_keypair: + cloud: "{{ item.cloud }}" + region_name: "{{ item.region_name }}" + name: mordred + public_key_file: ~/.ssh/id_rsa.pub + with-items: + - {cloud: vexxhost, region_name: ca-ymq-1} + - {cloud: ovh, region_name: SBG1} + - {cloud: ovh, region_name: BHS1} + - {cloud: ovh, region_name: GRA1} + - {cloud: citycloud, region_name: Buf1} + - {cloud: citycloud, region_name: La1} + - {cloud: citycloud, region_name: Fra1} + - {cloud: citycloud, region_name: Lon1} + - {cloud: citycloud, region_name: Sto2} + - {cloud: citycloud, region_name: Kna1} + - {cloud: internap, region_name: ams01} + - {cloud: internap, region_name: da01} + - {cloud: internap, region_name: nyj01} + - {cloud: internap, region_name: sin01} + - {cloud: internap, region_name: sjc01} + - {cloud: infracloud, region_name: vanilla} + - {cloud: infracloud, region_name: chocolate} + - {cloud: fuga, region_name: cystack} + - {cloud: datacentred, region_name: sal01} + - {cloud: clouda, region_name: regionOne} + - {cloud: auro, region_name: van1} + - {cloud: ustack, region_name: bj1} + - {cloud: zetta, region_name: no-osl1} + - {cloud: kiss, region_name: region1} + - {cloud: rax, region_name: IAD} + - {cloud: rax, region_name: ORD} + - {cloud: rax, region_name: SYD} + - {cloud: rax, region_name: LON} + - {cloud: rax, region_name: DFW} + - {cloud: rax, region_name: HKG} + </code></pre> + </section> + + <section class="slide level2"> + <h1>The Future: Pushing Back Into OpenStack APIs</h1> + <ul> + <li>Discussions later today</li> + <li>Service Discovery</li> + <li>Version Discovery</li> + <li>Vendors Publishing Vendor Profile Documents</li> + <li>Capabilities Discovery</li> + <li>Connecting those to Interop Working Group</li> + </ul> + </section> + + <section class="slide level2"> + <h1>Thank You!</h1> + </section> + + </body> +</html> + -- cgit v1.2.3