Kaynağa Gözat

Add build_image playbook for OpenStack

This adds a playbook that will create a temporary VM, install the
dependencies, pull the container images and save as a new OpenStack
image.

This should speed up the deployments as well as scaling.
Tomas Sedovic 6 yıl önce
ebeveyn
işleme
fc114a2d52

+ 4 - 0
playbooks/openstack/README.md

@@ -174,6 +174,9 @@ any_errors_fatal = true
 This will abort the Ansible playbook execution as soon as any error is
 encountered.
 
+If you want, you can [Build the OpenShift node images at this
+point][build-images].
+
 Now, run the provision + install playbook. This will create OpenStack resources
 and deploy an OpenShift cluster on top of them:
 
@@ -240,3 +243,4 @@ $ ansible-playbook --user openshift \
 [access-the-ui]: ./post-install.md#access-the-ui
 [dynamic]: http://docs.ansible.com/ansible/latest/intro_dynamic_inventory.html
 [kuryr-sdn]: ./configuration.md#kuryr-networking-configuration
+[build-images]: ./configuration.md#building-node-images

+ 38 - 0
playbooks/openstack/configuration.md

@@ -15,6 +15,7 @@ Environment variables may also be used.
 * [Stack Name Configuration](#stack-name-configuration)
 * [DNS Configuration](#dns-configuration)
 * [All-in-one Deployment Configuration](#all-in-one-deployment-configuration)
+* [Building Node Images](#building-node-images)
 * [Kuryr Networking Configuration](#kuryr-networking-configuration)
 * [Provider Network Configuration](#provider-network-configuration)
 * [Multi-Master Configuration](#multi-master-configuration)
@@ -309,6 +310,43 @@ Note that the "all in one" node must be the "master". openshift-ansible
 expects at least one node in the `masters` Ansible group.
 
 
+## Building Node Images
+
+It is possible to build the OpenShift images in advance (instead of installing
+the dependencies during the deployment). This will reduce the disk and network
+throughput as well as speed up the installation.
+
+To do this, the inventory must already exist and be configured.
+
+Set the `openshift_openstack_default_image_name` value in
+`inventory/group_vars/all.yml` to a name you want this new image to be called
+(e.g. `origin-node`). This name must not exist in OpenStack yet.
+
+Next, set `openshift_openstack_build_base_image` to a name of an *existing*
+image that you want to use as a base. This should be the cloud image you would
+normally use for the deployment.
+
+And finally, run the `build_image.yml` playbook:
+
+    ansible-playbook -i inventory openshift-ansible/playbooks/openstack/openshift-cluster/build_image.yml
+
+This will create a temporary Neutron network, subnet and router, launch a
+server in that subnet, install all the packages and pull the necessary
+container images and upload an image with the name set in
+`openshift_openstack_default_image_name`.
+
+All the extra OpenStack resources (network, subnet, router) will then be
+deleted.
+
+Note that the subnet's CIDR will be `192.168.23.0/24`. If you need to use a
+different value, set `openshift_openstack_build_network_cidr` before running
+the `build_image` playbook.
+
+If you don't want to be setting the build variables in your inventory, you can
+pass them to ansible-playbook directly:
+
+    ansible-playbook -i inventory openshift-ansible/playbooks/openstack/openshift-cluster/build_image.yml -e openshift_openstack_build_base_image=CentOS-7-x86_64-GenericCloud-1805 -e openshift_openstack_build_network_cidr=192.168.42.0/24
+
 
 ## Kuryr Networking Configuration
 

+ 181 - 0
playbooks/openstack/openshift-cluster/build_image.yml

@@ -0,0 +1,181 @@
+---
+- name: Verify image build prerequisites
+  hosts: localhost
+  connection: local
+  gather_facts: no
+  tasks:
+  - name: Require openshift_openstack_build_base_image
+    fail:
+      msg: "A base image name is required for image building. Please ensure `openshift_openstack_build_base_image` is defined."
+    when: openshift_openstack_build_base_image is undefined
+
+  - name: Require openshift_openstack_default_image_name
+    fail:
+      msg: >
+        You must specify the name the new image will be saved to Glance as.
+        Please make sure `openshift_openstack_default_image_name` is defined.
+    when:
+    - openshift_openstack_default_image_name is not defined
+
+  - name: Get the target image facts
+    os_image_facts:
+      image: "{{ openshift_openstack_default_image_name }}"
+    register: image_check
+
+  - name: Verify that the resulting image does not exist
+    fail:
+      msg: >
+        The image "{{ openshift_openstack_default_image_name }}" specified in
+        `openshift_openstack_default_image_name` exists already.
+        Please choose a different name or delete it.
+    when: image_check.ansible_facts.openstack_image
+
+  - set_fact:
+      build_prefix: "{{ openshift_openstack_clusterid|default('build') }}.{{ openshift_openstack_public_dns_domain|default('example.com') }}-build"
+
+
+- name: Launch image build instance
+  hosts: localhost
+  connection: local
+  gather_facts: yes
+  tasks:
+  # NOTE: we create a temporary network, subnet, router and security groups
+  # to have a known environment to launch the image build VM in.
+  # They get deleted once we save the image.
+  # TODO(shadower): allow specifying an existing subnet etc. instead.
+  - name: Create a network
+    os_network:
+      name: "{{ build_prefix }}-network"
+    register: network
+
+  - name: Create a subnet
+    os_subnet:
+      name: "{{ build_prefix }}-subnet"
+      network_name: "{{ network.network.name }}"
+      cidr: "{{ openshift_openstack_build_network_cidr | default('192.168.23.0/24') }}"
+    register: subnet
+
+  - name: Create the router
+    os_router:
+      name: "{{ build_prefix }}-router"
+      network: "{{ openshift_openstack_external_network_name }}"
+      interfaces:
+      - "{{ subnet.id }}"
+    register: router
+
+  - name: Create a security group
+    os_security_group:
+      name: "{{ build_prefix }}-security-group"
+      description: Security group for the image build server
+    register: security_group
+
+  - name: Allow pinging the server
+    os_security_group_rule:
+      security_group: "{{ security_group.id }}"
+      protocol: icmp
+      port_range_min: -1
+      port_range_max: -1
+
+  - name: Allow SSH access
+    os_security_group_rule:
+      security_group: "{{ security_group.id }}"
+      protocol: tcp
+      port_range_min: 22
+      port_range_max: 22
+
+  - name: Launch the image build instance
+    os_server:
+      name: "{{ build_prefix }}-image-server"
+      network: "{{ network.id }}"
+      auto_ip: yes
+      flavor: "{{ openshift_openstack_default_flavor }}"
+      image: "{{ openshift_openstack_build_base_image }}"
+      key_name: "{{ openshift_openstack_keypair_name }}"
+      security_groups:
+      - "{{ security_group.id }}"
+      # Create a known SSH user so we can log in to the VM.
+      # TODO(shadower): should we create a temporary keypair & user here
+      # and delete it when done?
+      userdata: |
+        #cloud-config
+        system_info:
+          default_user:
+            name: {{ ansible_user|default('openshift') }}
+            sudo: ["ALL=(ALL) NOPASSWD: ALL"]
+        write_files:
+          - path: /etc/sudoers.d/00-openshift-no-requiretty
+            permissions: 440
+            content: |
+              Defaults:{{ ansible_user|default('openshift') }} !requiretty
+      state: present
+    register: image_vm
+
+  - name: Add host to nodes
+    add_host:
+      name: "{{ image_vm.openstack.accessIPv4 }}"
+      groups: nodes,OSEv3
+      ansible_become: true
+
+  - name: Wait for instance to respond to SSH
+    wait_for:
+      delay: 1
+      host: "{{ image_vm.openstack.accessIPv4 }}"
+      port: 22
+      state: started
+      timeout: 120
+
+
+- name: Wait for full SSH connection
+  hosts: nodes
+  gather_facts: no
+  tasks:
+  - wait_for_connection:
+  - setup:
+  - set_fact:
+      openshift_node_image_prep_packages:
+      - cloud-init
+      - cloud-utils-growpart
+
+
+# This is the part that installs all of the software and configs for the instance
+# to become a node.
+- import_playbook: ../../openshift-node/private/image_prep.yml
+
+
+- name: Commit image
+  hosts: localhost
+  connection: local
+  tasks:
+  - name: Stop the image VM
+    os_server_action:
+      server: "{{ image_vm.id }}"
+      action: stop
+
+  - name: Save the new image
+    command: openstack server image create --wait --name "{{ openshift_openstack_default_image_name }}" "{{ image_vm.id }}"
+
+  # Remove the temporary OpenStack resources
+  - name: Remove the imabe build instance
+    os_server:
+      name: "{{ image_vm.id }}"
+      state: absent
+
+  - name: Remove the security group
+    os_security_group:
+      name: "{{ security_group.id }}"
+      state: absent
+
+  - name: Remove the router
+    os_router:
+      name: "{{ router.id }}"
+      state: absent
+
+  - name: Remove the subnet
+    os_subnet:
+      name: "{{ subnet.id }}"
+      state: absent
+
+  - name: Remove the network
+    os_network:
+      name: "{{ network.id }}"
+      state: absent

+ 1 - 0
roles/openshift_openstack/defaults/main.yml

@@ -87,6 +87,7 @@ openshift_openstack_private_network: >-
   {%- else -%}
   {{ openshift_openstack_private_network_name | default ('openshift-ansible-' + openshift_openstack_stack_name + '-net') }}
   {%- endif -%}
+openshift_openstack_node_run_bootstrap_startup: true
 openshift_openstack_master_server_group_policies: []
 openshift_openstack_infra_server_group_policies: []
 openshift_openstack_docker_volume_size: 15

+ 4 - 1
roles/openshift_openstack/templates/heat_stack_server.yaml.j2

@@ -231,7 +231,10 @@ resources:
 {% endif %}
 {% endif %}
       user_data:
-        get_file: user-data
+        str_replace:
+          template: {get_file: user-data}
+          params:
+            "%OPENSHIFT_NODE_CONFIG_NAME%": { get_param: openshift_node_group_name }
       user_data_format: RAW
       user_data_update_policy: IGNORE
       metadata:

+ 11 - 0
roles/openshift_openstack/templates/user_data.j2

@@ -12,6 +12,12 @@ write_files:
     content: |
       Defaults:openshift !requiretty
 
+  - path: /root/openshift_bootstrap/openshift_settings.yaml
+    owner: 'root:root'
+    permissions: '0640'
+    content: |
+        openshift_node_config_name: %OPENSHIFT_NODE_CONFIG_NAME%
+
 {% if openshift_openstack_provision_user_commands %}
   - path: /root/ansible_install.sh
     permissions: '0544'
@@ -23,7 +29,12 @@ write_files:
       {{ cmd|join(' ') }}
 {% endif %}
 {% endfor %}
+{% endif %}
 
 runcmd:
+{% if openshift_openstack_provision_user_commands %}
   - /root/ansible_install.sh
 {% endif %}
+{% if openshift_openstack_node_run_bootstrap_startup %}
+  - [ ansible-playbook, /root/openshift_bootstrap/bootstrap.yml]
+{% endif %}