diff options
68 files changed, 2507 insertions, 516 deletions
| diff --git a/.tito/packages/openshift-ansible b/.tito/packages/openshift-ansible index fd9a1844f..d29838038 100644 --- a/.tito/packages/openshift-ansible +++ b/.tito/packages/openshift-ansible @@ -1 +1 @@ -3.5.1-1 ./ +3.5.2-1 ./ diff --git a/HOOKS.md b/HOOKS.md new file mode 100644 index 000000000..9c5f80054 --- /dev/null +++ b/HOOKS.md @@ -0,0 +1,70 @@ +# Hooks + +The ansible installer allows for operators to execute custom tasks during +specific operations through a system called hooks. Hooks allow operators to +provide files defining tasks to execute before and/or after specific areas +during installations and upgrades. This can be very helpful to validate +or modify custom infrastructure when installing/upgrading OpenShift. + +It is important to remember that when a hook fails the operation fails. This +means a good hook can run multiple times and provide the same results. A great +hook is idempotent. + +**Note**: There is currently not a standard interface for hooks. In the future +a standard interface will be defined and any hooks that existed previously will +need to be updated to meet the new standard. + +## Using Hooks + +Hooks are defined in the ``hosts`` inventory file under the ``OSEv3:vars`` +section. + +Each hook should point to a yaml file which defines Ansible tasks. This file +will be used as an include meaning that the file can not be a playbook but +a set of tasks. Best practice suggests using absolute paths to the hook file to avoid any ambiguity. + +### Example +```ini +[OSEv3:vars] +# <snip> +openshift_master_upgrade_pre_hook=/usr/share/custom/pre_master.yml +openshift_master_upgrade_hook=/usr/share/custom/master.yml +openshift_master_upgrade_post_hook=/usr/share/custom/post_master.yml +# <snip> +``` + +Hook files must be a yaml formatted file that defines a set of Ansible tasks. +The file may **not** be a playbook. + +### Example +```yaml +--- +# Trivial example forcing an operator to ack the start of an upgrade +# file=/usr/share/custom/pre_master.yml + +- name: note the start of a master upgrade +  debug: +      msg: "Master upgrade of {{ inventory_hostname }} is about to start" + +- name: require an operator agree to start an upgrade +  pause: +      prompt: "Hit enter to start the master upgrade" +``` + +## Upgrade Hooks + +### openshift_master_upgrade_pre_hook +- Runs **before** each master is upgraded. +- This hook runs against **each master** in serial. +- If a task needs to run against a different host, said task will need to use [``delegate_to`` or ``local_action``](http://docs.ansible.com/ansible/playbooks_delegation.html#delegation). + +### openshift_master_upgrade_hook +- Runs **after** each master is upgraded but **before** it's service/system restart. +- This hook runs against **each master** in serial. +- If a task needs to run against a different host, said task will need to use [``delegate_to`` or ``local_action``](http://docs.ansible.com/ansible/playbooks_delegation.html#delegation). + + +### openshift_master_upgrade_post_hook +- Runs **after** each master is upgraded and has had it's service/system restart. +- This hook runs against **each master** in serial. +- If a task needs to run against a different host, said task will need to use [``delegate_to`` or ``local_action``](http://docs.ansible.com/ansible/playbooks_delegation.html#delegation). @@ -74,6 +74,12 @@ you are not running a stable release.        - [OpenShift Enterprise](https://docs.openshift.com/enterprise/latest/install_config/install/advanced_install.html)        - [OpenShift Origin](https://docs.openshift.org/latest/install_config/install/advanced_install.html) + +## Installer Hooks + +See the [hooks documentation](HOOKS.md). + +  ## Contributing  See the [contribution guide](CONTRIBUTING.md). diff --git a/inventory/byo/hosts.origin.example b/inventory/byo/hosts.origin.example index dde172c4a..0a1b8c5c4 100644 --- a/inventory/byo/hosts.origin.example +++ b/inventory/byo/hosts.origin.example @@ -89,6 +89,25 @@ openshift_release=v1.4  # Skip upgrading Docker during an OpenShift upgrade, leaves the current Docker version alone.  # docker_upgrade=False + +# Upgrade Hooks +# +# Hooks are available to run custom tasks at various points during a cluster +# upgrade. Each hook should point to a file with Ansible tasks defined. Suggest using +# absolute paths, if not the path will be treated as relative to the file where the +# hook is actually used. +# +# Tasks to run before each master is upgraded. +# openshift_master_upgrade_pre_hook=/usr/share/custom/pre_master.yml +# +# Tasks to run to upgrade the master. These tasks run after the main openshift-ansible +# upgrade steps, but before we restart system/services. +# openshift_master_upgrade_hook=/usr/share/custom/master.yml +# +# Tasks to run after each master is upgraded and system/services have been restarted. +# openshift_master_upgrade_post_hook=/usr/share/custom/post_master.yml + +  # Alternate image format string, useful if you've got your own registry mirror  #oreg_url=example.com/openshift3/ose-${component}:${version}  # If oreg_url points to a registry other than registry.access.redhat.com we can diff --git a/inventory/byo/hosts.ose.example b/inventory/byo/hosts.ose.example index c0dd8a1e8..89b9d7e48 100644 --- a/inventory/byo/hosts.ose.example +++ b/inventory/byo/hosts.ose.example @@ -89,6 +89,25 @@ openshift_release=v3.4  # Skip upgrading Docker during an OpenShift upgrade, leaves the current Docker version alone.  # docker_upgrade=False + +# Upgrade Hooks +# +# Hooks are available to run custom tasks at various points during a cluster +# upgrade. Each hook should point to a file with Ansible tasks defined. Suggest using +# absolute paths, if not the path will be treated as relative to the file where the +# hook is actually used. +# +# Tasks to run before each master is upgraded. +# openshift_master_upgrade_pre_hook=/usr/share/custom/pre_master.yml +# +# Tasks to run to upgrade the master. These tasks run after the main openshift-ansible +# upgrade steps, but before we restart system/services. +# openshift_master_upgrade_hook=/usr/share/custom/master.yml +# +# Tasks to run after each master is upgraded and system/services have been restarted. +# openshift_master_upgrade_post_hook=/usr/share/custom/post_master.yml + +  # Alternate image format string, useful if you've got your own registry mirror  #oreg_url=example.com/openshift3/ose-${component}:${version}  # If oreg_url points to a registry other than registry.access.redhat.com we can diff --git a/openshift-ansible.spec b/openshift-ansible.spec index a2940e001..85675f5f9 100644 --- a/openshift-ansible.spec +++ b/openshift-ansible.spec @@ -5,7 +5,7 @@  }  Name:           openshift-ansible -Version:        3.5.1 +Version:        3.5.2  Release:        1%{?dist}  Summary:        Openshift and Atomic Enterprise Ansible  License:        ASL 2.0 @@ -18,6 +18,8 @@ Requires:      python2  Requires:      python-six  Requires:      tar  Requires:      openshift-ansible-docs = %{version}-%{release} +Requires:      java-1.8.0-openjdk-headless +Requires:      httpd-tools  %description  Openshift and Atomic Enterprise Ansible @@ -251,6 +253,67 @@ Atomic OpenShift Utilities includes  %changelog +* Wed Jan 25 2017 Scott Dodson <sdodson@redhat.com> 3.5.2-1 +- Sync latest image streams (sdodson@redhat.com) +- Fix containerized haproxy config (andrew@andrewklau.com) +- Allow RHEL subscription for OSE 3.4 (lhuard@amadeus.com) +- fixes BZ-1415447. Error when stopping heapster.  Modify to be conditional +  include (jcantril@redhat.com) +- override nodename for gce with cloudprovider (jdetiber@redhat.com) +- fixes jks generation, node labeling, and rerunning for oauth secrets +  (ewolinet@redhat.com) +- allow openshift_logging role to specify nodeSelectors (jcantril@redhat.com) +- Remove is_containerized check for firewalld installs (rteague@redhat.com) +- Clean up pylint for delete_empty_keys. (abutcher@redhat.com) +- [os_firewall] Fix default iptables args. (abutcher@redhat.com) +- Add new option 'openshift_docker_selinux_enabled' (rteague@redhat.com) +- Temporary work-around for flake8 vs maccabe version conflict +  (tbielawa@redhat.com) +- do not set empty proxy env variable defaults (bparees@redhat.com) +- fix BZ1414477. Use keytool on control node and require java +  (jcantril@redhat.com) +- Remove unused temporary directory in master config playbook. +  (abutcher@redhat.com) +- Added link to HOOKS in README (smilner@redhat.com) +- HOOKS.md added documenting new hooks (smilner@redhat.com) +- [os_firewall] Add -w flag to wait for iptables xtables lock. +  (abutcher@redhat.com) +- fixes BZ-1414625. Check for httpd-tools and java before install +  (jcantril@redhat.com) +- Add a mid upgrade hook, re-prefix variables. (dgoodwin@redhat.com) +- treat force_pull as a bool (bparees@redhat.com) +- Adding to ansible spec and changing logging jks generation to be a +  local_action (ewolinet@redhat.com) +- Add containzerized haproxy option (andrew@andrewklau.com) +- Reorder node dnsmasq dependency s.t. networkmanager is restarted after +  firewall changes have been applied. (abutcher@redhat.com) +- Removing docker run strategy and make java a requirement for control host +  (ewolinet@redhat.com) +- Adding version to lib_openshift (kwoodson@redhat.com) +- Updating to use docker run instead of scheduling jks gen pod +  (ewolinet@redhat.com) +- jenkins v1.3 templates should not enable oauth (gmontero@redhat.com) +- fix oc_apply to allow running on any control node (jcantril@redhat.com) +- g_master_mktemp in openshift-master conflicts with +  openshift_master_certificates (rmeggins@redhat.com) +- fixes #3127. Get files for oc_apply from remote host (jcantril@redhat.com) +- Debug message before running hooks. (dgoodwin@redhat.com) +- Cleaning repo cache earlier (rteague@redhat.com) +- Added tar as a requirement per BZ1388445 (smilner@redhat.com) +- fixes BZ141619.  Corrects the variable in the README (jcantril@redhat.com) +- Run user provided hooks prior to system/service restarts. +  (dgoodwin@redhat.com) +- Implement pre/post master upgrade hooks. (dgoodwin@redhat.com) +- Adding oc_obj to the lib_openshift library (kwoodson@redhat.com) +- Addressing found issues with logging role (ewolinet@redhat.com) +- Updated the generate.py scripts for tox and virtualenv. (kwoodson@redhat.com) +- Adding tox tests for generated code. (kwoodson@redhat.com) +- Perform master upgrades in a single play serially. (dgoodwin@redhat.com) +- Validate system restart policy during pre-upgrade. (dgoodwin@redhat.com) +- Correct consistency between upgrade playbooks (rteague@redhat.com) +- Wait for nodes to be ready before proceeding with upgrade. +  (dgoodwin@redhat.com) +  * Wed Jan 18 2017 Scott Dodson <sdodson@redhat.com> 3.5.1-1  - More reliable wait for master after full host reboot. (dgoodwin@redhat.com)  - kubelet must have rw to cgroups for pod/qos cgroups to function diff --git a/playbooks/common/openshift-cluster/config.yml b/playbooks/common/openshift-cluster/config.yml index 0f226f5f9..a95cb68b7 100644 --- a/playbooks/common/openshift-cluster/config.yml +++ b/playbooks/common/openshift-cluster/config.yml @@ -38,6 +38,9 @@    - set_fact:        openshift_docker_log_options: "{{ lookup('oo_option', 'docker_log_options') }}"      when: openshift_docker_log_options is not defined +  - set_fact: +      openshift_docker_selinux_enabled: "{{ lookup('oo_option', 'docker_selinux_enabled') }}" +    when: openshift_docker_selinux_enabled is not defined  - include: ../openshift-etcd/config.yml    tags: diff --git a/playbooks/common/openshift-cluster/upgrades/upgrade_control_plane.yml b/playbooks/common/openshift-cluster/upgrades/upgrade_control_plane.yml index 7f738ea0f..77b37cdc2 100644 --- a/playbooks/common/openshift-cluster/upgrades/upgrade_control_plane.yml +++ b/playbooks/common/openshift-cluster/upgrades/upgrade_control_plane.yml @@ -51,6 +51,8 @@    roles:    - openshift_master_facts +# The main master upgrade play. Should handle all changes to the system in one pass, with +# support for optional hooks to be defined.  - name: Upgrade master    hosts: oo_masters_to_config    vars: @@ -62,6 +64,14 @@    roles:    - openshift_facts    post_tasks: + +  # Run the pre-upgrade hook if defined: +  - debug: msg="Running master pre-upgrade hook {{ openshift_master_upgrade_pre_hook }}" +    when: openshift_master_upgrade_pre_hook is defined + +  - include: "{{ openshift_master_upgrade_pre_hook }}" +    when: openshift_master_upgrade_pre_hook is defined +    - include: rpm_upgrade.yml component=master      when: not openshift.common.is_containerized | bool @@ -102,12 +112,26 @@        state: link      when: ca_crt_stat.stat.isreg and not ca_bundle_stat.stat.exists +  # Run the upgrade hook prior to restarting services/system if defined: +  - debug: msg="Running master upgrade hook {{ openshift_master_upgrade_hook }}" +    when: openshift_master_upgrade_hook is defined + +  - include: "{{ openshift_master_upgrade_hook }}" +    when: openshift_master_upgrade_hook is defined +    - include: ../../openshift-master/restart_hosts.yml      when: openshift.common.rolling_restart_mode == 'system'    - include: ../../openshift-master/restart_services.yml      when: openshift.common.rolling_restart_mode == 'services' +  # Run the post-upgrade hook if defined: +  - debug: msg="Running master post-upgrade hook {{ openshift_master_upgrade_post_hook }}" +    when: openshift_master_upgrade_post_hook is defined + +  - include: "{{ openshift_master_upgrade_post_hook }}" +    when: openshift_master_upgrade_post_hook is defined +    - set_fact:        master_update_complete: True diff --git a/playbooks/common/openshift-cluster/upgrades/upgrade_nodes.yml b/playbooks/common/openshift-cluster/upgrades/upgrade_nodes.yml index 86b344d7a..2bb460815 100644 --- a/playbooks/common/openshift-cluster/upgrades/upgrade_nodes.yml +++ b/playbooks/common/openshift-cluster/upgrades/upgrade_nodes.yml @@ -87,6 +87,19 @@    - name: Restart rpm node service      service: name="{{ openshift.common.service_type }}-node" state=restarted      when: inventory_hostname in groups.oo_nodes_to_upgrade and not openshift.common.is_containerized | bool + +  - name: Wait for node to be ready +    command: > +      {{ hostvars[groups.oo_first_master.0].openshift.common.client_binary }} get node {{ openshift.common.hostname | lower }} --no-headers +    register: node_output +    delegate_to: "{{ groups.oo_first_master.0 }}" +    when: inventory_hostname in groups.oo_nodes_to_upgrade +    until: "{{ node_output.stdout.split()[1].startswith('Ready')}}" +    # Give the node two minutes to come back online. Note that we pre-pull images now +    # so containerized services should restart quickly as well. +    retries: 24 +    delay: 5 +    - name: Set node schedulability      command: >        {{ hostvars[groups.oo_first_master.0].openshift.common.client_binary }} adm manage-node {{ openshift.node.nodename | lower }} --schedulable=true diff --git a/playbooks/common/openshift-master/config.yml b/playbooks/common/openshift-master/config.yml index de36fd263..7a334e771 100644 --- a/playbooks/common/openshift-master/config.yml +++ b/playbooks/common/openshift-master/config.yml @@ -75,17 +75,6 @@          ha: "{{ openshift_master_ha | default(groups.oo_masters | length > 1) }}"          master_count: "{{ openshift_master_count | default(groups.oo_masters | length) }}" -- name: Create temp directory for syncing certs -  hosts: localhost -  connection: local -  become: no -  gather_facts: no -  tasks: -  - name: Create local temp directory for syncing certs -    local_action: command mktemp -d /tmp/openshift-ansible-XXXXXXX -    register: g_master_mktemp -    changed_when: False -  - name: Determine if session secrets must be generated    hosts: oo_first_master    roles: @@ -117,7 +106,6 @@    hosts: oo_masters_to_config    any_errors_fatal: true    vars: -    sync_tmpdir: "{{ hostvars.localhost.g_master_mktemp.stdout }}"      openshift_master_ha: "{{ openshift.master.ha }}"      openshift_master_count: "{{ openshift.master.master_count }}"      openshift_master_session_auth_secrets: "{{ hostvars[groups.oo_first_master.0].openshift.master.session_auth_secrets }}" @@ -144,12 +132,3 @@    - name: Create group for deployment type      group_by: key=oo_masters_deployment_type_{{ openshift.common.deployment_type }}      changed_when: False - -- name: Delete temporary directory on localhost -  hosts: localhost -  connection: local -  become: no -  gather_facts: no -  tasks: -  - file: name={{ g_master_mktemp.stdout }} state=absent -    changed_when: False diff --git a/roles/docker/tasks/main.yml b/roles/docker/tasks/main.yml index a8935370a..66c9cfa0f 100644 --- a/roles/docker/tasks/main.yml +++ b/roles/docker/tasks/main.yml @@ -96,7 +96,7 @@      dest: /etc/sysconfig/docker      regexp: '^OPTIONS=.*$'      line: "OPTIONS='\ -      {% if ansible_selinux and ansible_selinux.status == '''enabled''' %} --selinux-enabled{% endif %}\ +      {% if ansible_selinux.status | default(None) == '''enabled''' and docker_selinux_enabled | default(true) %} --selinux-enabled {% endif %}\        {% if docker_log_driver is defined  %} --log-driver {{ docker_log_driver }}{% endif %}\        {% if docker_log_options is defined %} {{ docker_log_options |  oo_split() | oo_prepend_strings_in_list('--log-opt ') | join(' ')}}{% endif %}\        {% if docker_options is defined %} {{ docker_options }}{% endif %}\ diff --git a/roles/lib_openshift/library/oc_edit.py b/roles/lib_openshift/library/oc_edit.py index 99a335dc4..1a361ae20 100644 --- a/roles/lib_openshift/library/oc_edit.py +++ b/roles/lib_openshift/library/oc_edit.py @@ -765,7 +765,7 @@ class OpenShiftCLI(object):      def _replace(self, fname, force=False):          '''replace the current object with oc replace''' -        cmd = ['-n', self.namespace, 'replace', '-f', fname] +        cmd = ['replace', '-f', fname]          if force:              cmd.append('--force')          return self.openshift_cmd(cmd) @@ -782,11 +782,11 @@ class OpenShiftCLI(object):      def _create(self, fname):          '''call oc create on a filename''' -        return self.openshift_cmd(['create', '-f', fname, '-n', self.namespace]) +        return self.openshift_cmd(['create', '-f', fname])      def _delete(self, resource, rname, selector=None):          '''call oc delete on a resource''' -        cmd = ['delete', resource, rname, '-n', self.namespace] +        cmd = ['delete', resource, rname]          if selector:              cmd.append('--selector=%s' % selector) @@ -800,8 +800,7 @@ class OpenShiftCLI(object):             params: the parameters for the template             template_data: the incoming template's data; instead of a file          ''' - -        cmd = ['process', '-n', self.namespace] +        cmd = ['process']          if template_data:              cmd.extend(['-f', '-'])          else: @@ -822,17 +821,13 @@ class OpenShiftCLI(object):          atexit.register(Utils.cleanup, [fname]) -        return self.openshift_cmd(['-n', self.namespace, 'create', '-f', fname]) +        return self.openshift_cmd(['create', '-f', fname])      def _get(self, resource, rname=None, selector=None):          '''return a resource by name '''          cmd = ['get', resource]          if selector:              cmd.append('--selector=%s' % selector) -        if self.all_namespaces: -            cmd.extend(['--all-namespaces']) -        elif self.namespace: -            cmd.extend(['-n', self.namespace])          cmd.extend(['-o', 'json']) @@ -906,6 +901,10 @@ class OpenShiftCLI(object):          return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') +    def _version(self): +        ''' return the openshift version''' +        return self.openshift_cmd(['version'], output=True, output_type='raw') +      def _import_image(self, url=None, name=None, tag=None):          ''' perform image import '''          cmd = ['import-image'] @@ -924,7 +923,7 @@ class OpenShiftCLI(object):          cmd.append('--confirm')          return self.openshift_cmd(cmd) -    # pylint: disable=too-many-arguments +    # pylint: disable=too-many-arguments,too-many-branches      def openshift_cmd(self, cmd, oadm=False, output=False, output_type='json', input_data=None):          '''Base command for oc '''          cmds = [] @@ -933,6 +932,11 @@ class OpenShiftCLI(object):          else:              cmds = ['/usr/bin/oc'] +        if self.all_namespaces: +            cmds.extend(['--all-namespaces']) +        elif self.namespace: +            cmds.extend(['-n', self.namespace]) +          cmds.extend(cmd)          rval = {} @@ -1058,6 +1062,56 @@ class Utils(object):          return contents +    @staticmethod +    def filter_versions(stdout): +        ''' filter the oc version output ''' + +        version_dict = {} +        version_search = ['oc', 'openshift', 'kubernetes'] + +        for line in stdout.strip().split('\n'): +            for term in version_search: +                if not line: +                    continue +                if line.startswith(term): +                    version_dict[term] = line.split()[-1] + +        # horrible hack to get openshift version in Openshift 3.2 +        #  By default "oc version in 3.2 does not return an "openshift" version +        if "openshift" not in version_dict: +            version_dict["openshift"] = version_dict["oc"] + +        return version_dict + +    @staticmethod +    def add_custom_versions(versions): +        ''' create custom versions strings ''' + +        versions_dict = {} + +        for tech, version in versions.items(): +            # clean up "-" from version +            if "-" in version: +                version = version.split("-")[0] + +            if version.startswith('v'): +                versions_dict[tech + '_numeric'] = version[1:].split('+')[0] +                # "v3.3.0.33" is what we have, we want "3.3" +                versions_dict[tech + '_short'] = version[1:4] + +        return versions_dict + +    @staticmethod +    def openshift_installed(): +        ''' check if openshift is installed ''' +        import yum + +        yum_base = yum.YumBase() +        if yum_base.rpmdb.searchNevra(name='atomic-openshift'): +            return True + +        return False +      # Disabling too-many-branches.  This is a yaml dictionary comparison function      # pylint: disable=too-many-branches,too-many-return-statements,too-many-statements      @staticmethod diff --git a/roles/lib_openshift/library/oc_obj.py b/roles/lib_openshift/library/oc_obj.py index e4552c1f0..5b501484b 100644 --- a/roles/lib_openshift/library/oc_obj.py +++ b/roles/lib_openshift/library/oc_obj.py @@ -744,7 +744,7 @@ class OpenShiftCLI(object):      def _replace(self, fname, force=False):          '''replace the current object with oc replace''' -        cmd = ['-n', self.namespace, 'replace', '-f', fname] +        cmd = ['replace', '-f', fname]          if force:              cmd.append('--force')          return self.openshift_cmd(cmd) @@ -761,11 +761,11 @@ class OpenShiftCLI(object):      def _create(self, fname):          '''call oc create on a filename''' -        return self.openshift_cmd(['create', '-f', fname, '-n', self.namespace]) +        return self.openshift_cmd(['create', '-f', fname])      def _delete(self, resource, rname, selector=None):          '''call oc delete on a resource''' -        cmd = ['delete', resource, rname, '-n', self.namespace] +        cmd = ['delete', resource, rname]          if selector:              cmd.append('--selector=%s' % selector) @@ -779,8 +779,7 @@ class OpenShiftCLI(object):             params: the parameters for the template             template_data: the incoming template's data; instead of a file          ''' - -        cmd = ['process', '-n', self.namespace] +        cmd = ['process']          if template_data:              cmd.extend(['-f', '-'])          else: @@ -801,17 +800,13 @@ class OpenShiftCLI(object):          atexit.register(Utils.cleanup, [fname]) -        return self.openshift_cmd(['-n', self.namespace, 'create', '-f', fname]) +        return self.openshift_cmd(['create', '-f', fname])      def _get(self, resource, rname=None, selector=None):          '''return a resource by name '''          cmd = ['get', resource]          if selector:              cmd.append('--selector=%s' % selector) -        if self.all_namespaces: -            cmd.extend(['--all-namespaces']) -        elif self.namespace: -            cmd.extend(['-n', self.namespace])          cmd.extend(['-o', 'json']) @@ -885,6 +880,10 @@ class OpenShiftCLI(object):          return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') +    def _version(self): +        ''' return the openshift version''' +        return self.openshift_cmd(['version'], output=True, output_type='raw') +      def _import_image(self, url=None, name=None, tag=None):          ''' perform image import '''          cmd = ['import-image'] @@ -903,7 +902,7 @@ class OpenShiftCLI(object):          cmd.append('--confirm')          return self.openshift_cmd(cmd) -    # pylint: disable=too-many-arguments +    # pylint: disable=too-many-arguments,too-many-branches      def openshift_cmd(self, cmd, oadm=False, output=False, output_type='json', input_data=None):          '''Base command for oc '''          cmds = [] @@ -912,6 +911,11 @@ class OpenShiftCLI(object):          else:              cmds = ['/usr/bin/oc'] +        if self.all_namespaces: +            cmds.extend(['--all-namespaces']) +        elif self.namespace: +            cmds.extend(['-n', self.namespace]) +          cmds.extend(cmd)          rval = {} @@ -1037,6 +1041,56 @@ class Utils(object):          return contents +    @staticmethod +    def filter_versions(stdout): +        ''' filter the oc version output ''' + +        version_dict = {} +        version_search = ['oc', 'openshift', 'kubernetes'] + +        for line in stdout.strip().split('\n'): +            for term in version_search: +                if not line: +                    continue +                if line.startswith(term): +                    version_dict[term] = line.split()[-1] + +        # horrible hack to get openshift version in Openshift 3.2 +        #  By default "oc version in 3.2 does not return an "openshift" version +        if "openshift" not in version_dict: +            version_dict["openshift"] = version_dict["oc"] + +        return version_dict + +    @staticmethod +    def add_custom_versions(versions): +        ''' create custom versions strings ''' + +        versions_dict = {} + +        for tech, version in versions.items(): +            # clean up "-" from version +            if "-" in version: +                version = version.split("-")[0] + +            if version.startswith('v'): +                versions_dict[tech + '_numeric'] = version[1:].split('+')[0] +                # "v3.3.0.33" is what we have, we want "3.3" +                versions_dict[tech + '_short'] = version[1:4] + +        return versions_dict + +    @staticmethod +    def openshift_installed(): +        ''' check if openshift is installed ''' +        import yum + +        yum_base = yum.YumBase() +        if yum_base.rpmdb.searchNevra(name='atomic-openshift'): +            return True + +        return False +      # Disabling too-many-branches.  This is a yaml dictionary comparison function      # pylint: disable=too-many-branches,too-many-return-statements,too-many-statements      @staticmethod diff --git a/roles/lib_openshift/library/oc_route.py b/roles/lib_openshift/library/oc_route.py index b08858567..19c7462ea 100644 --- a/roles/lib_openshift/library/oc_route.py +++ b/roles/lib_openshift/library/oc_route.py @@ -769,7 +769,7 @@ class OpenShiftCLI(object):      def _replace(self, fname, force=False):          '''replace the current object with oc replace''' -        cmd = ['-n', self.namespace, 'replace', '-f', fname] +        cmd = ['replace', '-f', fname]          if force:              cmd.append('--force')          return self.openshift_cmd(cmd) @@ -786,11 +786,11 @@ class OpenShiftCLI(object):      def _create(self, fname):          '''call oc create on a filename''' -        return self.openshift_cmd(['create', '-f', fname, '-n', self.namespace]) +        return self.openshift_cmd(['create', '-f', fname])      def _delete(self, resource, rname, selector=None):          '''call oc delete on a resource''' -        cmd = ['delete', resource, rname, '-n', self.namespace] +        cmd = ['delete', resource, rname]          if selector:              cmd.append('--selector=%s' % selector) @@ -804,8 +804,7 @@ class OpenShiftCLI(object):             params: the parameters for the template             template_data: the incoming template's data; instead of a file          ''' - -        cmd = ['process', '-n', self.namespace] +        cmd = ['process']          if template_data:              cmd.extend(['-f', '-'])          else: @@ -826,17 +825,13 @@ class OpenShiftCLI(object):          atexit.register(Utils.cleanup, [fname]) -        return self.openshift_cmd(['-n', self.namespace, 'create', '-f', fname]) +        return self.openshift_cmd(['create', '-f', fname])      def _get(self, resource, rname=None, selector=None):          '''return a resource by name '''          cmd = ['get', resource]          if selector:              cmd.append('--selector=%s' % selector) -        if self.all_namespaces: -            cmd.extend(['--all-namespaces']) -        elif self.namespace: -            cmd.extend(['-n', self.namespace])          cmd.extend(['-o', 'json']) @@ -910,6 +905,10 @@ class OpenShiftCLI(object):          return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') +    def _version(self): +        ''' return the openshift version''' +        return self.openshift_cmd(['version'], output=True, output_type='raw') +      def _import_image(self, url=None, name=None, tag=None):          ''' perform image import '''          cmd = ['import-image'] @@ -928,7 +927,7 @@ class OpenShiftCLI(object):          cmd.append('--confirm')          return self.openshift_cmd(cmd) -    # pylint: disable=too-many-arguments +    # pylint: disable=too-many-arguments,too-many-branches      def openshift_cmd(self, cmd, oadm=False, output=False, output_type='json', input_data=None):          '''Base command for oc '''          cmds = [] @@ -937,6 +936,11 @@ class OpenShiftCLI(object):          else:              cmds = ['/usr/bin/oc'] +        if self.all_namespaces: +            cmds.extend(['--all-namespaces']) +        elif self.namespace: +            cmds.extend(['-n', self.namespace]) +          cmds.extend(cmd)          rval = {} @@ -1062,6 +1066,56 @@ class Utils(object):          return contents +    @staticmethod +    def filter_versions(stdout): +        ''' filter the oc version output ''' + +        version_dict = {} +        version_search = ['oc', 'openshift', 'kubernetes'] + +        for line in stdout.strip().split('\n'): +            for term in version_search: +                if not line: +                    continue +                if line.startswith(term): +                    version_dict[term] = line.split()[-1] + +        # horrible hack to get openshift version in Openshift 3.2 +        #  By default "oc version in 3.2 does not return an "openshift" version +        if "openshift" not in version_dict: +            version_dict["openshift"] = version_dict["oc"] + +        return version_dict + +    @staticmethod +    def add_custom_versions(versions): +        ''' create custom versions strings ''' + +        versions_dict = {} + +        for tech, version in versions.items(): +            # clean up "-" from version +            if "-" in version: +                version = version.split("-")[0] + +            if version.startswith('v'): +                versions_dict[tech + '_numeric'] = version[1:].split('+')[0] +                # "v3.3.0.33" is what we have, we want "3.3" +                versions_dict[tech + '_short'] = version[1:4] + +        return versions_dict + +    @staticmethod +    def openshift_installed(): +        ''' check if openshift is installed ''' +        import yum + +        yum_base = yum.YumBase() +        if yum_base.rpmdb.searchNevra(name='atomic-openshift'): +            return True + +        return False +      # Disabling too-many-branches.  This is a yaml dictionary comparison function      # pylint: disable=too-many-branches,too-many-return-statements,too-many-statements      @staticmethod @@ -1204,7 +1258,9 @@ class RouteConfig(object):                   key=None,                   host=None,                   tls_termination=None, -                 service_name=None): +                 service_name=None, +                 wildcard_policy=None, +                 weight=None):          ''' constructor for handling route options '''          self.kubeconfig = kubeconfig          self.name = sname @@ -1217,6 +1273,12 @@ class RouteConfig(object):          self.key = key          self.service_name = service_name          self.data = {} +        self.wildcard_policy = wildcard_policy +        if wildcard_policy is None: +            self.wildcard_policy = 'None' +        self.weight = weight +        if weight is None: +            self.weight = 100          self.create_dict() @@ -1241,14 +1303,19 @@ class RouteConfig(object):              self.data['spec']['tls']['certificate'] = self.cert              self.data['spec']['tls']['termination'] = self.tls_termination -        self.data['spec']['to'] = {'kind': 'Service', 'name': self.service_name} +        self.data['spec']['to'] = {'kind': 'Service', +                                   'name': self.service_name, +                                   'weight': self.weight} +        self.data['spec']['wildcardPolicy'] = self.wildcard_policy  # pylint: disable=too-many-instance-attributes,too-many-public-methods  class Route(Yedit):      ''' Class to wrap the oc command line tools ''' +    wildcard_policy = "spec.wildcardPolicy"      host_path = "spec.host"      service_path = "spec.to.name" +    weight_path = "spec.to.weight"      cert_path = "spec.tls.certificate"      cacert_path = "spec.tls.caCertificate"      destcacert_path = "spec.tls.destinationCACertificate" @@ -1280,6 +1347,10 @@ class Route(Yedit):          ''' return service name '''          return self.get(Route.service_path) +    def get_weight(self): +        ''' return service weight ''' +        return self.get(Route.weight_path) +      def get_termination(self):          ''' return tls termination'''          return self.get(Route.termination_path) @@ -1288,6 +1359,10 @@ class Route(Yedit):          ''' return host '''          return self.get(Route.host_path) +    def get_wildcard_policy(self): +        ''' return wildcardPolicy ''' +        return self.get(Route.wildcard_policy) +  # pylint: disable=too-many-instance-attributes  class OCRoute(OpenShiftCLI): @@ -1375,7 +1450,9 @@ class OCRoute(OpenShiftCLI):                                files['key']['value'],                                params['host'],                                params['tls_termination'], -                              params['service_name']) +                              params['service_name'], +                              params['wildcard_policy'], +                              params['weight'])          oc_route = OCRoute(rconfig, verbose=params['debug']) @@ -1418,13 +1495,13 @@ class OCRoute(OpenShiftCLI):                  api_rval = oc_route.create()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  # return the created object                  api_rval = oc_route.get()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  return {'changed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 @@ -1439,13 +1516,13 @@ class OCRoute(OpenShiftCLI):                  api_rval = oc_route.update()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  # return the created object                  api_rval = oc_route.get()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  return {'changed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 @@ -1493,6 +1570,8 @@ def main():              key_content=dict(default=None, type='str'),              service_name=dict(default=None, type='str'),              host=dict(default=None, type='str'), +            wildcard_policy=dict(default=None, type='str'), +            weight=dict(default=None, type='int'),          ),          mutually_exclusive=[('dest_cacert_path', 'dest_cacert_content'),                              ('cacert_path', 'cacert_content'), diff --git a/roles/lib_openshift/library/oc_version.py b/roles/lib_openshift/library/oc_version.py new file mode 100644 index 000000000..197a0a947 --- /dev/null +++ b/roles/lib_openshift/library/oc_version.py @@ -0,0 +1,1232 @@ +#!/usr/bin/env python +# pylint: disable=missing-docstring +# flake8: noqa: T001 +#     ___ ___ _  _ ___ ___    _ _____ ___ ___ +#    / __| __| \| | __| _ \  /_\_   _| __|   \ +#   | (_ | _|| .` | _||   / / _ \| | | _|| |) | +#    \___|___|_|\_|___|_|_\/_/_\_\_|_|___|___/_ _____ +#   |   \ / _ \  | \| |/ _ \_   _| | __|   \_ _|_   _| +#   | |) | (_) | | .` | (_) || |   | _|| |) | |  | | +#   |___/ \___/  |_|\_|\___/ |_|   |___|___/___| |_| +# +# Copyright 2016 Red Hat, Inc. and/or its affiliates +# and other contributors as indicated by the @author tags. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +#    http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +''' +   OpenShiftCLI class that wraps the oc commands in a subprocess +''' +# pylint: disable=too-many-lines + +from __future__ import print_function +import atexit +import json +import os +import re +import shutil +import subprocess +# pylint: disable=import-error +import ruamel.yaml as yaml +from ansible.module_utils.basic import AnsibleModule + +DOCUMENTATION = ''' +--- +module: oc_version +short_description: Return the current openshift version +description: +  - Return the openshift installed version.  `oc version` +options: +  state: +    description: +    - Currently list is only supported state. +    required: true +    default: list +    choices: ["list"] +    aliases: [] +  kubeconfig: +    description: +    - The path for the kubeconfig file to use for authentication +    required: false +    default: /etc/origin/master/admin.kubeconfig +    aliases: [] +  debug: +    description: +    - Turn on debug output. +    required: false +    default: False +    aliases: [] +author: +- "Kenny Woodson <kwoodson@redhat.com>" +extends_documentation_fragment: [] +''' + +EXAMPLES = ''' +oc_version: +- name: get oc version +  oc_version: +  register: oc_version +''' +# noqa: E301,E302 + + +class YeditException(Exception): +    ''' Exception class for Yedit ''' +    pass + + +# pylint: disable=too-many-public-methods +class Yedit(object): +    ''' Class to modify yaml files ''' +    re_valid_key = r"(((\[-?\d+\])|([0-9a-zA-Z%s/_-]+)).?)+$" +    re_key = r"(?:\[(-?\d+)\])|([0-9a-zA-Z%s/_-]+)" +    com_sep = set(['.', '#', '|', ':']) + +    # pylint: disable=too-many-arguments +    def __init__(self, +                 filename=None, +                 content=None, +                 content_type='yaml', +                 separator='.', +                 backup=False): +        self.content = content +        self._separator = separator +        self.filename = filename +        self.__yaml_dict = content +        self.content_type = content_type +        self.backup = backup +        self.load(content_type=self.content_type) +        if self.__yaml_dict is None: +            self.__yaml_dict = {} + +    @property +    def separator(self): +        ''' getter method for yaml_dict ''' +        return self._separator + +    @separator.setter +    def separator(self): +        ''' getter method for yaml_dict ''' +        return self._separator + +    @property +    def yaml_dict(self): +        ''' getter method for yaml_dict ''' +        return self.__yaml_dict + +    @yaml_dict.setter +    def yaml_dict(self, value): +        ''' setter method for yaml_dict ''' +        self.__yaml_dict = value + +    @staticmethod +    def parse_key(key, sep='.'): +        '''parse the key allowing the appropriate separator''' +        common_separators = list(Yedit.com_sep - set([sep])) +        return re.findall(Yedit.re_key % ''.join(common_separators), key) + +    @staticmethod +    def valid_key(key, sep='.'): +        '''validate the incoming key''' +        common_separators = list(Yedit.com_sep - set([sep])) +        if not re.match(Yedit.re_valid_key % ''.join(common_separators), key): +            return False + +        return True + +    @staticmethod +    def remove_entry(data, key, sep='.'): +        ''' remove data at location key ''' +        if key == '' and isinstance(data, dict): +            data.clear() +            return True +        elif key == '' and isinstance(data, list): +            del data[:] +            return True + +        if not (key and Yedit.valid_key(key, sep)) and \ +           isinstance(data, (list, dict)): +            return None + +        key_indexes = Yedit.parse_key(key, sep) +        for arr_ind, dict_key in key_indexes[:-1]: +            if dict_key and isinstance(data, dict): +                data = data.get(dict_key, None) +            elif (arr_ind and isinstance(data, list) and +                  int(arr_ind) <= len(data) - 1): +                data = data[int(arr_ind)] +            else: +                return None + +        # process last index for remove +        # expected list entry +        if key_indexes[-1][0]: +            if isinstance(data, list) and int(key_indexes[-1][0]) <= len(data) - 1:  # noqa: E501 +                del data[int(key_indexes[-1][0])] +                return True + +        # expected dict entry +        elif key_indexes[-1][1]: +            if isinstance(data, dict): +                del data[key_indexes[-1][1]] +                return True + +    @staticmethod +    def add_entry(data, key, item=None, sep='.'): +        ''' Get an item from a dictionary with key notation a.b.c +            d = {'a': {'b': 'c'}}} +            key = a#b +            return c +        ''' +        if key == '': +            pass +        elif (not (key and Yedit.valid_key(key, sep)) and +              isinstance(data, (list, dict))): +            return None + +        key_indexes = Yedit.parse_key(key, sep) +        for arr_ind, dict_key in key_indexes[:-1]: +            if dict_key: +                if isinstance(data, dict) and dict_key in data and data[dict_key]:  # noqa: E501 +                    data = data[dict_key] +                    continue + +                elif data and not isinstance(data, dict): +                    return None + +                data[dict_key] = {} +                data = data[dict_key] + +            elif (arr_ind and isinstance(data, list) and +                  int(arr_ind) <= len(data) - 1): +                data = data[int(arr_ind)] +            else: +                return None + +        if key == '': +            data = item + +        # process last index for add +        # expected list entry +        elif key_indexes[-1][0] and isinstance(data, list) and int(key_indexes[-1][0]) <= len(data) - 1:  # noqa: E501 +            data[int(key_indexes[-1][0])] = item + +        # expected dict entry +        elif key_indexes[-1][1] and isinstance(data, dict): +            data[key_indexes[-1][1]] = item + +        return data + +    @staticmethod +    def get_entry(data, key, sep='.'): +        ''' Get an item from a dictionary with key notation a.b.c +            d = {'a': {'b': 'c'}}} +            key = a.b +            return c +        ''' +        if key == '': +            pass +        elif (not (key and Yedit.valid_key(key, sep)) and +              isinstance(data, (list, dict))): +            return None + +        key_indexes = Yedit.parse_key(key, sep) +        for arr_ind, dict_key in key_indexes: +            if dict_key and isinstance(data, dict): +                data = data.get(dict_key, None) +            elif (arr_ind and isinstance(data, list) and +                  int(arr_ind) <= len(data) - 1): +                data = data[int(arr_ind)] +            else: +                return None + +        return data + +    def write(self): +        ''' write to file ''' +        if not self.filename: +            raise YeditException('Please specify a filename.') + +        if self.backup and self.file_exists(): +            shutil.copy(self.filename, self.filename + '.orig') + +        tmp_filename = self.filename + '.yedit' +        with open(tmp_filename, 'w') as yfd: +            # pylint: disable=no-member +            if hasattr(self.yaml_dict, 'fa'): +                self.yaml_dict.fa.set_block_style() + +            yfd.write(yaml.dump(self.yaml_dict, Dumper=yaml.RoundTripDumper)) + +        os.rename(tmp_filename, self.filename) + +        return (True, self.yaml_dict) + +    def read(self): +        ''' read from file ''' +        # check if it exists +        if self.filename is None or not self.file_exists(): +            return None + +        contents = None +        with open(self.filename) as yfd: +            contents = yfd.read() + +        return contents + +    def file_exists(self): +        ''' return whether file exists ''' +        if os.path.exists(self.filename): +            return True + +        return False + +    def load(self, content_type='yaml'): +        ''' return yaml file ''' +        contents = self.read() + +        if not contents and not self.content: +            return None + +        if self.content: +            if isinstance(self.content, dict): +                self.yaml_dict = self.content +                return self.yaml_dict +            elif isinstance(self.content, str): +                contents = self.content + +        # check if it is yaml +        try: +            if content_type == 'yaml' and contents: +                self.yaml_dict = yaml.load(contents, yaml.RoundTripLoader) +                # pylint: disable=no-member +                if hasattr(self.yaml_dict, 'fa'): +                    self.yaml_dict.fa.set_block_style() +            elif content_type == 'json' and contents: +                self.yaml_dict = json.loads(contents) +        except yaml.YAMLError as err: +            # Error loading yaml or json +            raise YeditException('Problem with loading yaml file. %s' % err) + +        return self.yaml_dict + +    def get(self, key): +        ''' get a specified key''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, key, self.separator) +        except KeyError: +            entry = None + +        return entry + +    def pop(self, path, key_or_item): +        ''' remove a key, value pair from a dict or an item for a list''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        except KeyError: +            entry = None + +        if entry is None: +            return (False, self.yaml_dict) + +        if isinstance(entry, dict): +            # pylint: disable=no-member,maybe-no-member +            if key_or_item in entry: +                entry.pop(key_or_item) +                return (True, self.yaml_dict) +            return (False, self.yaml_dict) + +        elif isinstance(entry, list): +            # pylint: disable=no-member,maybe-no-member +            ind = None +            try: +                ind = entry.index(key_or_item) +            except ValueError: +                return (False, self.yaml_dict) + +            entry.pop(ind) +            return (True, self.yaml_dict) + +        return (False, self.yaml_dict) + +    def delete(self, path): +        ''' remove path from a dict''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        except KeyError: +            entry = None + +        if entry is None: +            return (False, self.yaml_dict) + +        result = Yedit.remove_entry(self.yaml_dict, path, self.separator) +        if not result: +            return (False, self.yaml_dict) + +        return (True, self.yaml_dict) + +    def exists(self, path, value): +        ''' check if value exists at path''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        except KeyError: +            entry = None + +        if isinstance(entry, list): +            if value in entry: +                return True +            return False + +        elif isinstance(entry, dict): +            if isinstance(value, dict): +                rval = False +                for key, val in value.items(): +                    if entry[key] != val: +                        rval = False +                        break +                else: +                    rval = True +                return rval + +            return value in entry + +        return entry == value + +    def append(self, path, value): +        '''append value to a list''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        except KeyError: +            entry = None + +        if entry is None: +            self.put(path, []) +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        if not isinstance(entry, list): +            return (False, self.yaml_dict) + +        # pylint: disable=no-member,maybe-no-member +        entry.append(value) +        return (True, self.yaml_dict) + +    # pylint: disable=too-many-arguments +    def update(self, path, value, index=None, curr_value=None): +        ''' put path, value into a dict ''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        except KeyError: +            entry = None + +        if isinstance(entry, dict): +            # pylint: disable=no-member,maybe-no-member +            if not isinstance(value, dict): +                raise YeditException('Cannot replace key, value entry in ' + +                                     'dict with non-dict type. value=[%s] [%s]' % (value, type(value)))  # noqa: E501 + +            entry.update(value) +            return (True, self.yaml_dict) + +        elif isinstance(entry, list): +            # pylint: disable=no-member,maybe-no-member +            ind = None +            if curr_value: +                try: +                    ind = entry.index(curr_value) +                except ValueError: +                    return (False, self.yaml_dict) + +            elif index is not None: +                ind = index + +            if ind is not None and entry[ind] != value: +                entry[ind] = value +                return (True, self.yaml_dict) + +            # see if it exists in the list +            try: +                ind = entry.index(value) +            except ValueError: +                # doesn't exist, append it +                entry.append(value) +                return (True, self.yaml_dict) + +            # already exists, return +            if ind is not None: +                return (False, self.yaml_dict) +        return (False, self.yaml_dict) + +    def put(self, path, value): +        ''' put path, value into a dict ''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, path, self.separator) +        except KeyError: +            entry = None + +        if entry == value: +            return (False, self.yaml_dict) + +        # deepcopy didn't work +        tmp_copy = yaml.load(yaml.round_trip_dump(self.yaml_dict, +                                                  default_flow_style=False), +                             yaml.RoundTripLoader) +        # pylint: disable=no-member +        if hasattr(self.yaml_dict, 'fa'): +            tmp_copy.fa.set_block_style() +        result = Yedit.add_entry(tmp_copy, path, value, self.separator) +        if not result: +            return (False, self.yaml_dict) + +        self.yaml_dict = tmp_copy + +        return (True, self.yaml_dict) + +    def create(self, path, value): +        ''' create a yaml file ''' +        if not self.file_exists(): +            # deepcopy didn't work +            tmp_copy = yaml.load(yaml.round_trip_dump(self.yaml_dict, default_flow_style=False),  # noqa: E501 +                                 yaml.RoundTripLoader) +            # pylint: disable=no-member +            if hasattr(self.yaml_dict, 'fa'): +                tmp_copy.fa.set_block_style() +            result = Yedit.add_entry(tmp_copy, path, value, self.separator) +            if result: +                self.yaml_dict = tmp_copy +                return (True, self.yaml_dict) + +        return (False, self.yaml_dict) + +    @staticmethod +    def get_curr_value(invalue, val_type): +        '''return the current value''' +        if invalue is None: +            return None + +        curr_value = invalue +        if val_type == 'yaml': +            curr_value = yaml.load(invalue) +        elif val_type == 'json': +            curr_value = json.loads(invalue) + +        return curr_value + +    @staticmethod +    def parse_value(inc_value, vtype=''): +        '''determine value type passed''' +        true_bools = ['y', 'Y', 'yes', 'Yes', 'YES', 'true', 'True', 'TRUE', +                      'on', 'On', 'ON', ] +        false_bools = ['n', 'N', 'no', 'No', 'NO', 'false', 'False', 'FALSE', +                       'off', 'Off', 'OFF'] + +        # It came in as a string but you didn't specify value_type as string +        # we will convert to bool if it matches any of the above cases +        if isinstance(inc_value, str) and 'bool' in vtype: +            if inc_value not in true_bools and inc_value not in false_bools: +                raise YeditException('Not a boolean type. str=[%s] vtype=[%s]' +                                     % (inc_value, vtype)) +        elif isinstance(inc_value, bool) and 'str' in vtype: +            inc_value = str(inc_value) + +        # If vtype is not str then go ahead and attempt to yaml load it. +        if isinstance(inc_value, str) and 'str' not in vtype: +            try: +                inc_value = yaml.load(inc_value) +            except Exception: +                raise YeditException('Could not determine type of incoming ' + +                                     'value. value=[%s] vtype=[%s]' +                                     % (type(inc_value), vtype)) + +        return inc_value + +    # pylint: disable=too-many-return-statements,too-many-branches +    @staticmethod +    def run_ansible(module): +        '''perform the idempotent crud operations''' +        yamlfile = Yedit(filename=module.params['src'], +                         backup=module.params['backup'], +                         separator=module.params['separator']) + +        if module.params['src']: +            rval = yamlfile.load() + +            if yamlfile.yaml_dict is None and \ +               module.params['state'] != 'present': +                return {'failed': True, +                        'msg': 'Error opening file [%s].  Verify that the ' + +                               'file exists, that it is has correct' + +                               ' permissions, and is valid yaml.'} + +        if module.params['state'] == 'list': +            if module.params['content']: +                content = Yedit.parse_value(module.params['content'], +                                            module.params['content_type']) +                yamlfile.yaml_dict = content + +            if module.params['key']: +                rval = yamlfile.get(module.params['key']) or {} + +            return {'changed': False, 'result': rval, 'state': "list"} + +        elif module.params['state'] == 'absent': +            if module.params['content']: +                content = Yedit.parse_value(module.params['content'], +                                            module.params['content_type']) +                yamlfile.yaml_dict = content + +            if module.params['update']: +                rval = yamlfile.pop(module.params['key'], +                                    module.params['value']) +            else: +                rval = yamlfile.delete(module.params['key']) + +            if rval[0] and module.params['src']: +                yamlfile.write() + +            return {'changed': rval[0], 'result': rval[1], 'state': "absent"} + +        elif module.params['state'] == 'present': +            # check if content is different than what is in the file +            if module.params['content']: +                content = Yedit.parse_value(module.params['content'], +                                            module.params['content_type']) + +                # We had no edits to make and the contents are the same +                if yamlfile.yaml_dict == content and \ +                   module.params['value'] is None: +                    return {'changed': False, +                            'result': yamlfile.yaml_dict, +                            'state': "present"} + +                yamlfile.yaml_dict = content + +            # we were passed a value; parse it +            if module.params['value']: +                value = Yedit.parse_value(module.params['value'], +                                          module.params['value_type']) +                key = module.params['key'] +                if module.params['update']: +                    # pylint: disable=line-too-long +                    curr_value = Yedit.get_curr_value(Yedit.parse_value(module.params['curr_value']),  # noqa: E501 +                                                      module.params['curr_value_format'])  # noqa: E501 + +                    rval = yamlfile.update(key, value, module.params['index'], curr_value)  # noqa: E501 + +                elif module.params['append']: +                    rval = yamlfile.append(key, value) +                else: +                    rval = yamlfile.put(key, value) + +                if rval[0] and module.params['src']: +                    yamlfile.write() + +                return {'changed': rval[0], +                        'result': rval[1], 'state': "present"} + +            # no edits to make +            if module.params['src']: +                # pylint: disable=redefined-variable-type +                rval = yamlfile.write() +                return {'changed': rval[0], +                        'result': rval[1], +                        'state': "present"} + +        return {'failed': True, 'msg': 'Unkown state passed'} +# pylint: disable=too-many-lines +# noqa: E301,E302,E303,T001 + + +class OpenShiftCLIError(Exception): +    '''Exception class for openshiftcli''' +    pass + + +# pylint: disable=too-few-public-methods +class OpenShiftCLI(object): +    ''' Class to wrap the command line tools ''' +    def __init__(self, +                 namespace, +                 kubeconfig='/etc/origin/master/admin.kubeconfig', +                 verbose=False, +                 all_namespaces=False): +        ''' Constructor for OpenshiftCLI ''' +        self.namespace = namespace +        self.verbose = verbose +        self.kubeconfig = kubeconfig +        self.all_namespaces = all_namespaces + +    # Pylint allows only 5 arguments to be passed. +    # pylint: disable=too-many-arguments +    def _replace_content(self, resource, rname, content, force=False, sep='.'): +        ''' replace the current object with the content ''' +        res = self._get(resource, rname) +        if not res['results']: +            return res + +        fname = '/tmp/%s' % rname +        yed = Yedit(fname, res['results'][0], separator=sep) +        changes = [] +        for key, value in content.items(): +            changes.append(yed.put(key, value)) + +        if any([change[0] for change in changes]): +            yed.write() + +            atexit.register(Utils.cleanup, [fname]) + +            return self._replace(fname, force) + +        return {'returncode': 0, 'updated': False} + +    def _replace(self, fname, force=False): +        '''replace the current object with oc replace''' +        cmd = ['replace', '-f', fname] +        if force: +            cmd.append('--force') +        return self.openshift_cmd(cmd) + +    def _create_from_content(self, rname, content): +        '''create a temporary file and then call oc create on it''' +        fname = '/tmp/%s' % rname +        yed = Yedit(fname, content=content) +        yed.write() + +        atexit.register(Utils.cleanup, [fname]) + +        return self._create(fname) + +    def _create(self, fname): +        '''call oc create on a filename''' +        return self.openshift_cmd(['create', '-f', fname]) + +    def _delete(self, resource, rname, selector=None): +        '''call oc delete on a resource''' +        cmd = ['delete', resource, rname] +        if selector: +            cmd.append('--selector=%s' % selector) + +        return self.openshift_cmd(cmd) + +    def _process(self, template_name, create=False, params=None, template_data=None):  # noqa: E501 +        '''process a template + +           template_name: the name of the template to process +           create: whether to send to oc create after processing +           params: the parameters for the template +           template_data: the incoming template's data; instead of a file +        ''' +        cmd = ['process'] +        if template_data: +            cmd.extend(['-f', '-']) +        else: +            cmd.append(template_name) +        if params: +            param_str = ["%s=%s" % (key, value) for key, value in params.items()] +            cmd.append('-v') +            cmd.extend(param_str) + +        results = self.openshift_cmd(cmd, output=True, input_data=template_data) + +        if results['returncode'] != 0 or not create: +            return results + +        fname = '/tmp/%s' % template_name +        yed = Yedit(fname, results['results']) +        yed.write() + +        atexit.register(Utils.cleanup, [fname]) + +        return self.openshift_cmd(['create', '-f', fname]) + +    def _get(self, resource, rname=None, selector=None): +        '''return a resource by name ''' +        cmd = ['get', resource] +        if selector: +            cmd.append('--selector=%s' % selector) + +        cmd.extend(['-o', 'json']) + +        if rname: +            cmd.append(rname) + +        rval = self.openshift_cmd(cmd, output=True) + +        # Ensure results are retuned in an array +        if 'items' in rval: +            rval['results'] = rval['items'] +        elif not isinstance(rval['results'], list): +            rval['results'] = [rval['results']] + +        return rval + +    def _schedulable(self, node=None, selector=None, schedulable=True): +        ''' perform oadm manage-node scheduable ''' +        cmd = ['manage-node'] +        if node: +            cmd.extend(node) +        else: +            cmd.append('--selector=%s' % selector) + +        cmd.append('--schedulable=%s' % schedulable) + +        return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw')  # noqa: E501 + +    def _list_pods(self, node=None, selector=None, pod_selector=None): +        ''' perform oadm list pods + +            node: the node in which to list pods +            selector: the label selector filter if provided +            pod_selector: the pod selector filter if provided +        ''' +        cmd = ['manage-node'] +        if node: +            cmd.extend(node) +        else: +            cmd.append('--selector=%s' % selector) + +        if pod_selector: +            cmd.append('--pod-selector=%s' % pod_selector) + +        cmd.extend(['--list-pods', '-o', 'json']) + +        return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') + +    # pylint: disable=too-many-arguments +    def _evacuate(self, node=None, selector=None, pod_selector=None, dry_run=False, grace_period=None, force=False): +        ''' perform oadm manage-node evacuate ''' +        cmd = ['manage-node'] +        if node: +            cmd.extend(node) +        else: +            cmd.append('--selector=%s' % selector) + +        if dry_run: +            cmd.append('--dry-run') + +        if pod_selector: +            cmd.append('--pod-selector=%s' % pod_selector) + +        if grace_period: +            cmd.append('--grace-period=%s' % int(grace_period)) + +        if force: +            cmd.append('--force') + +        cmd.append('--evacuate') + +        return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') + +    def _version(self): +        ''' return the openshift version''' +        return self.openshift_cmd(['version'], output=True, output_type='raw') + +    def _import_image(self, url=None, name=None, tag=None): +        ''' perform image import ''' +        cmd = ['import-image'] + +        image = '{0}'.format(name) +        if tag: +            image += ':{0}'.format(tag) + +        cmd.append(image) + +        if url: +            cmd.append('--from={0}/{1}'.format(url, image)) + +        cmd.append('-n{0}'.format(self.namespace)) + +        cmd.append('--confirm') +        return self.openshift_cmd(cmd) + +    # pylint: disable=too-many-arguments,too-many-branches +    def openshift_cmd(self, cmd, oadm=False, output=False, output_type='json', input_data=None): +        '''Base command for oc ''' +        cmds = [] +        if oadm: +            cmds = ['/usr/bin/oadm'] +        else: +            cmds = ['/usr/bin/oc'] + +        if self.all_namespaces: +            cmds.extend(['--all-namespaces']) +        elif self.namespace: +            cmds.extend(['-n', self.namespace]) + +        cmds.extend(cmd) + +        rval = {} +        results = '' +        err = None + +        if self.verbose: +            print(' '.join(cmds)) + +        proc = subprocess.Popen(cmds, +                                stdin=subprocess.PIPE, +                                stdout=subprocess.PIPE, +                                stderr=subprocess.PIPE, +                                env={'KUBECONFIG': self.kubeconfig}) + +        stdout, stderr = proc.communicate(input_data) +        rval = {"returncode": proc.returncode, +                "results": results, +                "cmd": ' '.join(cmds)} + +        if proc.returncode == 0: +            if output: +                if output_type == 'json': +                    try: +                        rval['results'] = json.loads(stdout) +                    except ValueError as err: +                        if "No JSON object could be decoded" in err.args: +                            err = err.args +                elif output_type == 'raw': +                    rval['results'] = stdout + +            if self.verbose: +                print("STDOUT: {0}".format(stdout)) +                print("STDERR: {0}".format(stderr)) + +            if err: +                rval.update({"err": err, +                             "stderr": stderr, +                             "stdout": stdout, +                             "cmd": cmds}) + +        else: +            rval.update({"stderr": stderr, +                         "stdout": stdout, +                         "results": {}}) + +        return rval + + +class Utils(object): +    ''' utilities for openshiftcli modules ''' +    @staticmethod +    def create_file(rname, data, ftype='yaml'): +        ''' create a file in tmp with name and contents''' +        path = os.path.join('/tmp', rname) +        with open(path, 'w') as fds: +            if ftype == 'yaml': +                fds.write(yaml.dump(data, Dumper=yaml.RoundTripDumper)) + +            elif ftype == 'json': +                fds.write(json.dumps(data)) +            else: +                fds.write(data) + +        # Register cleanup when module is done +        atexit.register(Utils.cleanup, [path]) +        return path + +    @staticmethod +    def create_files_from_contents(content, content_type=None): +        '''Turn an array of dict: filename, content into a files array''' +        if not isinstance(content, list): +            content = [content] +        files = [] +        for item in content: +            path = Utils.create_file(item['path'], item['data'], ftype=content_type) +            files.append({'name': os.path.basename(path), 'path': path}) +        return files + +    @staticmethod +    def cleanup(files): +        '''Clean up on exit ''' +        for sfile in files: +            if os.path.exists(sfile): +                if os.path.isdir(sfile): +                    shutil.rmtree(sfile) +                elif os.path.isfile(sfile): +                    os.remove(sfile) + +    @staticmethod +    def exists(results, _name): +        ''' Check to see if the results include the name ''' +        if not results: +            return False + +        if Utils.find_result(results, _name): +            return True + +        return False + +    @staticmethod +    def find_result(results, _name): +        ''' Find the specified result by name''' +        rval = None +        for result in results: +            if 'metadata' in result and result['metadata']['name'] == _name: +                rval = result +                break + +        return rval + +    @staticmethod +    def get_resource_file(sfile, sfile_type='yaml'): +        ''' return the service file ''' +        contents = None +        with open(sfile) as sfd: +            contents = sfd.read() + +        if sfile_type == 'yaml': +            contents = yaml.load(contents, yaml.RoundTripLoader) +        elif sfile_type == 'json': +            contents = json.loads(contents) + +        return contents + +    @staticmethod +    def filter_versions(stdout): +        ''' filter the oc version output ''' + +        version_dict = {} +        version_search = ['oc', 'openshift', 'kubernetes'] + +        for line in stdout.strip().split('\n'): +            for term in version_search: +                if not line: +                    continue +                if line.startswith(term): +                    version_dict[term] = line.split()[-1] + +        # horrible hack to get openshift version in Openshift 3.2 +        #  By default "oc version in 3.2 does not return an "openshift" version +        if "openshift" not in version_dict: +            version_dict["openshift"] = version_dict["oc"] + +        return version_dict + +    @staticmethod +    def add_custom_versions(versions): +        ''' create custom versions strings ''' + +        versions_dict = {} + +        for tech, version in versions.items(): +            # clean up "-" from version +            if "-" in version: +                version = version.split("-")[0] + +            if version.startswith('v'): +                versions_dict[tech + '_numeric'] = version[1:].split('+')[0] +                # "v3.3.0.33" is what we have, we want "3.3" +                versions_dict[tech + '_short'] = version[1:4] + +        return versions_dict + +    @staticmethod +    def openshift_installed(): +        ''' check if openshift is installed ''' +        import yum + +        yum_base = yum.YumBase() +        if yum_base.rpmdb.searchNevra(name='atomic-openshift'): +            return True + +        return False + +    # Disabling too-many-branches.  This is a yaml dictionary comparison function +    # pylint: disable=too-many-branches,too-many-return-statements,too-many-statements +    @staticmethod +    def check_def_equal(user_def, result_def, skip_keys=None, debug=False): +        ''' Given a user defined definition, compare it with the results given back by our query.  ''' + +        # Currently these values are autogenerated and we do not need to check them +        skip = ['metadata', 'status'] +        if skip_keys: +            skip.extend(skip_keys) + +        for key, value in result_def.items(): +            if key in skip: +                continue + +            # Both are lists +            if isinstance(value, list): +                if key not in user_def: +                    if debug: +                        print('User data does not have key [%s]' % key) +                        print('User data: %s' % user_def) +                    return False + +                if not isinstance(user_def[key], list): +                    if debug: +                        print('user_def[key] is not a list key=[%s] user_def[key]=%s' % (key, user_def[key])) +                    return False + +                if len(user_def[key]) != len(value): +                    if debug: +                        print("List lengths are not equal.") +                        print("key=[%s]: user_def[%s] != value[%s]" % (key, len(user_def[key]), len(value))) +                        print("user_def: %s" % user_def[key]) +                        print("value: %s" % value) +                    return False + +                for values in zip(user_def[key], value): +                    if isinstance(values[0], dict) and isinstance(values[1], dict): +                        if debug: +                            print('sending list - list') +                            print(type(values[0])) +                            print(type(values[1])) +                        result = Utils.check_def_equal(values[0], values[1], skip_keys=skip_keys, debug=debug) +                        if not result: +                            print('list compare returned false') +                            return False + +                    elif value != user_def[key]: +                        if debug: +                            print('value should be identical') +                            print(value) +                            print(user_def[key]) +                        return False + +            # recurse on a dictionary +            elif isinstance(value, dict): +                if key not in user_def: +                    if debug: +                        print("user_def does not have key [%s]" % key) +                    return False +                if not isinstance(user_def[key], dict): +                    if debug: +                        print("dict returned false: not instance of dict") +                    return False + +                # before passing ensure keys match +                api_values = set(value.keys()) - set(skip) +                user_values = set(user_def[key].keys()) - set(skip) +                if api_values != user_values: +                    if debug: +                        print("keys are not equal in dict") +                        print(api_values) +                        print(user_values) +                    return False + +                result = Utils.check_def_equal(user_def[key], value, skip_keys=skip_keys, debug=debug) +                if not result: +                    if debug: +                        print("dict returned false") +                        print(result) +                    return False + +            # Verify each key, value pair is the same +            else: +                if key not in user_def or value != user_def[key]: +                    if debug: +                        print("value not equal; user_def does not have key") +                        print(key) +                        print(value) +                        if key in user_def: +                            print(user_def[key]) +                    return False + +        if debug: +            print('returning true') +        return True + + +class OpenShiftCLIConfig(object): +    '''Generic Config''' +    def __init__(self, rname, namespace, kubeconfig, options): +        self.kubeconfig = kubeconfig +        self.name = rname +        self.namespace = namespace +        self._options = options + +    @property +    def config_options(self): +        ''' return config options ''' +        return self._options + +    def to_option_list(self): +        '''return all options as a string''' +        return self.stringify() + +    def stringify(self): +        ''' return the options hash as cli params in a string ''' +        rval = [] +        for key, data in self.config_options.items(): +            if data['include'] \ +               and (data['value'] or isinstance(data['value'], int)): +                rval.append('--%s=%s' % (key.replace('_', '-'), data['value'])) + +        return rval + + + +# pylint: disable=too-many-instance-attributes +class OCVersion(OpenShiftCLI): +    ''' Class to wrap the oc command line tools ''' +    # pylint allows 5 +    # pylint: disable=too-many-arguments +    def __init__(self, +                 config, +                 debug): +        ''' Constructor for OCVersion ''' +        super(OCVersion, self).__init__(None, config) +        self.debug = debug + +    def get(self): +        '''get and return version information ''' + +        results = {} + +        version_results = self._version() + +        if version_results['returncode'] == 0: +            filtered_vers = Utils.filter_versions(version_results['results']) +            custom_vers = Utils.add_custom_versions(filtered_vers) + +            results['returncode'] = version_results['returncode'] +            results.update(filtered_vers) +            results.update(custom_vers) + +            return results + +        raise OpenShiftCLIError('Problem detecting openshift version.') + +    @staticmethod +    def run_ansible(params): +        '''run the idempotent ansible code''' +        oc_version = OCVersion(params['kubeconfig'], params['debug']) + +        if params['state'] == 'list': + +            #pylint: disable=protected-access +            result = oc_version.get() +            return {'state': params['state'], +                    'results': result, +                    'changed': False} + +def main(): +    ''' ansible oc module for version ''' + +    module = AnsibleModule( +        argument_spec=dict( +            kubeconfig=dict(default='/etc/origin/master/admin.kubeconfig', type='str'), +            state=dict(default='list', type='str', +                       choices=['list']), +            debug=dict(default=False, type='bool'), +        ), +        supports_check_mode=True, +    ) + +    rval = OCVersion.run_ansible(module.params) +    if 'failed' in rval: +        module.fail_json(**rval) + + +    module.exit_json(**rval) + + +if __name__ == '__main__': +    main() diff --git a/roles/lib_openshift/src/ansible/oc_route.py b/roles/lib_openshift/src/ansible/oc_route.py index 3dcae052c..c87e6738f 100644 --- a/roles/lib_openshift/src/ansible/oc_route.py +++ b/roles/lib_openshift/src/ansible/oc_route.py @@ -40,6 +40,8 @@ def main():              key_content=dict(default=None, type='str'),              service_name=dict(default=None, type='str'),              host=dict(default=None, type='str'), +            wildcard_policy=dict(default=None, type='str'), +            weight=dict(default=None, type='int'),          ),          mutually_exclusive=[('dest_cacert_path', 'dest_cacert_content'),                              ('cacert_path', 'cacert_content'), diff --git a/roles/lib_openshift/src/ansible/oc_version.py b/roles/lib_openshift/src/ansible/oc_version.py new file mode 100644 index 000000000..57ef849ca --- /dev/null +++ b/roles/lib_openshift/src/ansible/oc_version.py @@ -0,0 +1,26 @@ +# pylint: skip-file +# flake8: noqa + +def main(): +    ''' ansible oc module for version ''' + +    module = AnsibleModule( +        argument_spec=dict( +            kubeconfig=dict(default='/etc/origin/master/admin.kubeconfig', type='str'), +            state=dict(default='list', type='str', +                       choices=['list']), +            debug=dict(default=False, type='bool'), +        ), +        supports_check_mode=True, +    ) + +    rval = OCVersion.run_ansible(module.params) +    if 'failed' in rval: +        module.fail_json(**rval) + + +    module.exit_json(**rval) + + +if __name__ == '__main__': +    main() diff --git a/roles/lib_openshift/src/class/oc_route.py b/roles/lib_openshift/src/class/oc_route.py index 05b1be409..42af2c01c 100644 --- a/roles/lib_openshift/src/class/oc_route.py +++ b/roles/lib_openshift/src/class/oc_route.py @@ -88,7 +88,9 @@ class OCRoute(OpenShiftCLI):                                files['key']['value'],                                params['host'],                                params['tls_termination'], -                              params['service_name']) +                              params['service_name'], +                              params['wildcard_policy'], +                              params['weight'])          oc_route = OCRoute(rconfig, verbose=params['debug']) @@ -131,13 +133,13 @@ class OCRoute(OpenShiftCLI):                  api_rval = oc_route.create()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  # return the created object                  api_rval = oc_route.get()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  return {'changed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 @@ -152,13 +154,13 @@ class OCRoute(OpenShiftCLI):                  api_rval = oc_route.update()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  # return the created object                  api_rval = oc_route.get()                  if api_rval['returncode'] != 0: -                    return {'failed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 +                    return {'failed': True, 'msg': api_rval, 'state': "present"}  # noqa: E501                  return {'changed': True, 'results': api_rval, 'state': "present"}  # noqa: E501 diff --git a/roles/lib_openshift/src/class/oc_version.py b/roles/lib_openshift/src/class/oc_version.py new file mode 100644 index 000000000..7f8c721d8 --- /dev/null +++ b/roles/lib_openshift/src/class/oc_version.py @@ -0,0 +1,47 @@ +# flake8: noqa +# pylint: skip-file + + +# pylint: disable=too-many-instance-attributes +class OCVersion(OpenShiftCLI): +    ''' Class to wrap the oc command line tools ''' +    # pylint allows 5 +    # pylint: disable=too-many-arguments +    def __init__(self, +                 config, +                 debug): +        ''' Constructor for OCVersion ''' +        super(OCVersion, self).__init__(None, config) +        self.debug = debug + +    def get(self): +        '''get and return version information ''' + +        results = {} + +        version_results = self._version() + +        if version_results['returncode'] == 0: +            filtered_vers = Utils.filter_versions(version_results['results']) +            custom_vers = Utils.add_custom_versions(filtered_vers) + +            results['returncode'] = version_results['returncode'] +            results.update(filtered_vers) +            results.update(custom_vers) + +            return results + +        raise OpenShiftCLIError('Problem detecting openshift version.') + +    @staticmethod +    def run_ansible(params): +        '''run the idempotent ansible code''' +        oc_version = OCVersion(params['kubeconfig'], params['debug']) + +        if params['state'] == 'list': + +            #pylint: disable=protected-access +            result = oc_version.get() +            return {'state': params['state'], +                    'results': result, +                    'changed': False} diff --git a/roles/lib_openshift/src/doc/version b/roles/lib_openshift/src/doc/version new file mode 100644 index 000000000..c0fdd53e7 --- /dev/null +++ b/roles/lib_openshift/src/doc/version @@ -0,0 +1,40 @@ +# flake8: noqa +# pylint: skip-file + +DOCUMENTATION = ''' +--- +module: oc_version +short_description: Return the current openshift version +description: +  - Return the openshift installed version.  `oc version` +options: +  state: +    description: +    - Currently list is only supported state. +    required: true +    default: list +    choices: ["list"] +    aliases: [] +  kubeconfig: +    description: +    - The path for the kubeconfig file to use for authentication +    required: false +    default: /etc/origin/master/admin.kubeconfig +    aliases: [] +  debug: +    description: +    - Turn on debug output. +    required: false +    default: False +    aliases: [] +author: +- "Kenny Woodson <kwoodson@redhat.com>" +extends_documentation_fragment: [] +''' + +EXAMPLES = ''' +oc_version: +- name: get oc version +  oc_version: +  register: oc_version +''' diff --git a/roles/lib_openshift/src/lib/base.py b/roles/lib_openshift/src/lib/base.py index cf2f1b14d..db5f4e890 100644 --- a/roles/lib_openshift/src/lib/base.py +++ b/roles/lib_openshift/src/lib/base.py @@ -48,7 +48,7 @@ class OpenShiftCLI(object):      def _replace(self, fname, force=False):          '''replace the current object with oc replace''' -        cmd = ['-n', self.namespace, 'replace', '-f', fname] +        cmd = ['replace', '-f', fname]          if force:              cmd.append('--force')          return self.openshift_cmd(cmd) @@ -65,11 +65,11 @@ class OpenShiftCLI(object):      def _create(self, fname):          '''call oc create on a filename''' -        return self.openshift_cmd(['create', '-f', fname, '-n', self.namespace]) +        return self.openshift_cmd(['create', '-f', fname])      def _delete(self, resource, rname, selector=None):          '''call oc delete on a resource''' -        cmd = ['delete', resource, rname, '-n', self.namespace] +        cmd = ['delete', resource, rname]          if selector:              cmd.append('--selector=%s' % selector) @@ -83,8 +83,7 @@ class OpenShiftCLI(object):             params: the parameters for the template             template_data: the incoming template's data; instead of a file          ''' - -        cmd = ['process', '-n', self.namespace] +        cmd = ['process']          if template_data:              cmd.extend(['-f', '-'])          else: @@ -105,17 +104,13 @@ class OpenShiftCLI(object):          atexit.register(Utils.cleanup, [fname]) -        return self.openshift_cmd(['-n', self.namespace, 'create', '-f', fname]) +        return self.openshift_cmd(['create', '-f', fname])      def _get(self, resource, rname=None, selector=None):          '''return a resource by name '''          cmd = ['get', resource]          if selector:              cmd.append('--selector=%s' % selector) -        if self.all_namespaces: -            cmd.extend(['--all-namespaces']) -        elif self.namespace: -            cmd.extend(['-n', self.namespace])          cmd.extend(['-o', 'json']) @@ -189,6 +184,10 @@ class OpenShiftCLI(object):          return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') +    def _version(self): +        ''' return the openshift version''' +        return self.openshift_cmd(['version'], output=True, output_type='raw') +      def _import_image(self, url=None, name=None, tag=None):          ''' perform image import '''          cmd = ['import-image'] @@ -207,7 +206,7 @@ class OpenShiftCLI(object):          cmd.append('--confirm')          return self.openshift_cmd(cmd) -    # pylint: disable=too-many-arguments +    # pylint: disable=too-many-arguments,too-many-branches      def openshift_cmd(self, cmd, oadm=False, output=False, output_type='json', input_data=None):          '''Base command for oc '''          cmds = [] @@ -216,6 +215,11 @@ class OpenShiftCLI(object):          else:              cmds = ['/usr/bin/oc'] +        if self.all_namespaces: +            cmds.extend(['--all-namespaces']) +        elif self.namespace: +            cmds.extend(['-n', self.namespace]) +          cmds.extend(cmd)          rval = {} @@ -341,6 +345,56 @@ class Utils(object):          return contents +    @staticmethod +    def filter_versions(stdout): +        ''' filter the oc version output ''' + +        version_dict = {} +        version_search = ['oc', 'openshift', 'kubernetes'] + +        for line in stdout.strip().split('\n'): +            for term in version_search: +                if not line: +                    continue +                if line.startswith(term): +                    version_dict[term] = line.split()[-1] + +        # horrible hack to get openshift version in Openshift 3.2 +        #  By default "oc version in 3.2 does not return an "openshift" version +        if "openshift" not in version_dict: +            version_dict["openshift"] = version_dict["oc"] + +        return version_dict + +    @staticmethod +    def add_custom_versions(versions): +        ''' create custom versions strings ''' + +        versions_dict = {} + +        for tech, version in versions.items(): +            # clean up "-" from version +            if "-" in version: +                version = version.split("-")[0] + +            if version.startswith('v'): +                versions_dict[tech + '_numeric'] = version[1:].split('+')[0] +                # "v3.3.0.33" is what we have, we want "3.3" +                versions_dict[tech + '_short'] = version[1:4] + +        return versions_dict + +    @staticmethod +    def openshift_installed(): +        ''' check if openshift is installed ''' +        import yum + +        yum_base = yum.YumBase() +        if yum_base.rpmdb.searchNevra(name='atomic-openshift'): +            return True + +        return False +      # Disabling too-many-branches.  This is a yaml dictionary comparison function      # pylint: disable=too-many-branches,too-many-return-statements,too-many-statements      @staticmethod diff --git a/roles/lib_openshift/src/lib/route.py b/roles/lib_openshift/src/lib/route.py index df062b0dd..3130e7358 100644 --- a/roles/lib_openshift/src/lib/route.py +++ b/roles/lib_openshift/src/lib/route.py @@ -17,7 +17,9 @@ class RouteConfig(object):                   key=None,                   host=None,                   tls_termination=None, -                 service_name=None): +                 service_name=None, +                 wildcard_policy=None, +                 weight=None):          ''' constructor for handling route options '''          self.kubeconfig = kubeconfig          self.name = sname @@ -30,6 +32,12 @@ class RouteConfig(object):          self.key = key          self.service_name = service_name          self.data = {} +        self.wildcard_policy = wildcard_policy +        if wildcard_policy is None: +            self.wildcard_policy = 'None' +        self.weight = weight +        if weight is None: +            self.weight = 100          self.create_dict() @@ -54,14 +62,19 @@ class RouteConfig(object):              self.data['spec']['tls']['certificate'] = self.cert              self.data['spec']['tls']['termination'] = self.tls_termination -        self.data['spec']['to'] = {'kind': 'Service', 'name': self.service_name} +        self.data['spec']['to'] = {'kind': 'Service', +                                   'name': self.service_name, +                                   'weight': self.weight} +        self.data['spec']['wildcardPolicy'] = self.wildcard_policy  # pylint: disable=too-many-instance-attributes,too-many-public-methods  class Route(Yedit):      ''' Class to wrap the oc command line tools ''' +    wildcard_policy = "spec.wildcardPolicy"      host_path = "spec.host"      service_path = "spec.to.name" +    weight_path = "spec.to.weight"      cert_path = "spec.tls.certificate"      cacert_path = "spec.tls.caCertificate"      destcacert_path = "spec.tls.destinationCACertificate" @@ -93,6 +106,10 @@ class Route(Yedit):          ''' return service name '''          return self.get(Route.service_path) +    def get_weight(self): +        ''' return service weight ''' +        return self.get(Route.weight_path) +      def get_termination(self):          ''' return tls termination'''          return self.get(Route.termination_path) @@ -100,3 +117,7 @@ class Route(Yedit):      def get_host(self):          ''' return host '''          return self.get(Route.host_path) + +    def get_wildcard_policy(self): +        ''' return wildcardPolicy ''' +        return self.get(Route.wildcard_policy) diff --git a/roles/lib_openshift/src/sources.yml b/roles/lib_openshift/src/sources.yml index 50d70d1e5..f1fd558d3 100644 --- a/roles/lib_openshift/src/sources.yml +++ b/roles/lib_openshift/src/sources.yml @@ -27,3 +27,12 @@ oc_route.py:  - lib/route.py  - class/oc_route.py  - ansible/oc_route.py +oc_version.py: +- doc/generated +- doc/license +- lib/import.py +- doc/version +- ../../lib_utils/src/class/yedit.py +- lib/base.py +- class/oc_version.py +- ansible/oc_version.py diff --git a/roles/lib_openshift/src/test/integration/route.yml b/roles/lib_openshift/src/test/integration/oc_route.yml index 6a96b334f..620d5d5e7 100644..100755 --- a/roles/lib_openshift/src/test/integration/route.yml +++ b/roles/lib_openshift/src/test/integration/oc_route.yml @@ -1,5 +1,5 @@ -#!/usr/bin/ansible-playbook -# ./route.yml -M ../../../library -e "cli_master_test=$OPENSHIFT_MASTER +#!/usr/bin/ansible-playbook --module-path=../../../library/ +# ./oc_route.yml -M ../../../library -e "cli_master_test=$OPENSHIFT_MASTER  ---  - hosts: "{{ cli_master_test }}"    gather_facts: no @@ -8,15 +8,20 @@    - name: create route      oc_route:        name: test -      namespace: test +      namespace: default        tls_termination: edge        cert_content: testing cert        cacert_content: testing cacert +      key_content: key content        service_name: test        host: test.example      register: routeout    - debug: var=routeout +  - assert: +      that: "routeout.results.results[0]['metadata']['name'] == 'test'" +      msg: route create failed +    - name: get route      oc_route:        state: list @@ -25,6 +30,10 @@      register: routeout    - debug: var=routeout +  - assert: +      that: "routeout.results[0]['metadata']['name'] == 'test'" +      msg: get route failed +    - name: delete route      oc_route:        state: absent @@ -33,13 +42,18 @@      register: routeout    - debug: var=routeout +  - assert: +      that: "routeout.results.returncode == 0" +      msg: delete route failed +    - name: create route      oc_route:        name: test -      namespace: test +      namespace: default        tls_termination: edge        cert_content: testing cert        cacert_content: testing cacert +      key_content: testing key        service_name: test        host: test.example      register: routeout @@ -48,11 +62,16 @@    - name: create route noop      oc_route:        name: test -      namespace: test +      namespace: default        tls_termination: edge        cert_content: testing cert        cacert_content: testing cacert +      key_content: testing key        service_name: test        host: test.example      register: routeout    - debug: var=routeout + +  - assert: +      that: "routeout.changed == False" +      msg: Route create not idempotent diff --git a/roles/lib_openshift/src/test/integration/oc_version.yml b/roles/lib_openshift/src/test/integration/oc_version.yml new file mode 100755 index 000000000..52336d8da --- /dev/null +++ b/roles/lib_openshift/src/test/integration/oc_version.yml @@ -0,0 +1,17 @@ +#!/usr/bin/ansible-playbook --module-path=../../../library/ +# ./oc_version.yml -e "cli_master_test=$OPENSHIFT_MASTER +--- +- hosts: "{{ cli_master_test }}" +  gather_facts: no +  user: root +  tasks: +  - name: Get openshift version +    oc_version: +    register: versionout + +  - debug: var=versionout + +  - assert: +      that: +      - "'oc_numeric' in versionout.results.keys()" +      msg: "Did not find 'oc_numeric' in version results." diff --git a/roles/lib_openshift/src/test/unit/oc_version.py b/roles/lib_openshift/src/test/unit/oc_version.py new file mode 100755 index 000000000..8d9128187 --- /dev/null +++ b/roles/lib_openshift/src/test/unit/oc_version.py @@ -0,0 +1,70 @@ +#!/usr/bin/env python2 +''' + Unit tests for oc version +''' +# To run +# python -m unittest version +# +# . +# Ran 1 test in 0.597s +# +# OK + +import os +import sys +import unittest + +# Removing invalid variable names for tests so that I can +# keep them brief +# pylint: disable=invalid-name,no-name-in-module +# Disable import-error b/c our libraries aren't loaded in jenkins +# pylint: disable=import-error,wrong-import-position +# place class in our python path +module_path = os.path.join('/'.join(os.path.realpath(__file__).split('/')[:-4]), 'library')  # noqa: E501 +sys.path.insert(0, module_path) +from oc_version import OCVersion  # noqa: E402 + + +# pylint: disable=unused-argument +def oc_cmd_mock(cmd, oadm=False, output=False, output_type='json', input_data=None): +    '''mock command for openshift_cmd''' +    version = '''oc v3.4.0.39 +kubernetes v1.4.0+776c994 +features: Basic-Auth GSSAPI Kerberos SPNEGO + +Server https://internal.api.opstest.openshift.com +openshift v3.4.0.39 +kubernetes v1.4.0+776c994 +''' +    if 'version' in cmd: +        return {'stderr': None, +                'stdout': version, +                'returncode': 0, +                'results': version, +                'cmd': cmd} + + +class OCVersionTest(unittest.TestCase): +    ''' +     Test class for OCVersion +    ''' + +    def setUp(self): +        ''' setup method will create a file and set to known configuration ''' +        self.oc_ver = OCVersion(None, False) +        self.oc_ver.openshift_cmd = oc_cmd_mock + +    def test_get(self): +        ''' Testing a get ''' +        results = self.oc_ver.get() +        self.assertEqual(results['oc_short'], '3.4') +        self.assertEqual(results['oc_numeric'], '3.4.0.39') +        self.assertEqual(results['kubernetes_numeric'], '1.4.0') + +    def tearDown(self): +        '''TearDown method''' +        pass + + +if __name__ == "__main__": +    unittest.main() diff --git a/roles/openshift_builddefaults/vars/main.yml b/roles/openshift_builddefaults/vars/main.yml index c9ec3b82f..fe6069ea9 100644 --- a/roles/openshift_builddefaults/vars/main.yml +++ b/roles/openshift_builddefaults/vars/main.yml @@ -23,7 +23,6 @@ builddefaults_yaml:        imageLabels: "{{ openshift_builddefaults_image_labels | default(None) }}"        nodeSelector: "{{ openshift_builddefaults_nodeselectors | default(None) }}"        annotations: "{{ openshift_builddefaults_annotations | default(None) }}" -      #resources: "{{ openshift.builddefaults.resources | default(None) }}"        resources:          requests:            cpu: "{{ openshift_builddefaults_resources_requests_cpu | default(None) }}" diff --git a/roles/openshift_buildoverrides/tasks/main.yml b/roles/openshift_buildoverrides/tasks/main.yml index 82fce1c5b..87d0e6f21 100644 --- a/roles/openshift_buildoverrides/tasks/main.yml +++ b/roles/openshift_buildoverrides/tasks/main.yml @@ -1,13 +1,4 @@  --- -#- name: Set buildoverrides -#  openshift_facts: -#    role: buildoverrides -#    local_facts: -#      force_pull: "{{ openshift_buildoverrides_force_pull | default(None) }}" -#      image_labels: "{{ openshift_buildoverrides_image_labels | default(None) }}" -#      nodeselectors: "{{ openshift_buildoverrides_nodeselectors | default(None) }}" -#      annotations: "{{ openshift_buildoverrides_annotations | default(None) }}" -  - name: Set buildoverrides config structure    openshift_facts:      role: buildoverrides diff --git a/roles/openshift_buildoverrides/vars/main.yml b/roles/openshift_buildoverrides/vars/main.yml index f0f9c255b..cf49a6ebf 100644 --- a/roles/openshift_buildoverrides/vars/main.yml +++ b/roles/openshift_buildoverrides/vars/main.yml @@ -1,10 +1,11 @@  --- +force_pull: "{{ openshift_buildoverrides_force_pull | default('') }}"  buildoverrides_yaml:    BuildOverrides:      configuration:        apiVersion: v1        kind: BuildOverridesConfig -      forcePull: "{{ openshift_buildoverrides_force_pull | default('', true) }}" +      forcePull: "{{ '' if force_pull == '' else force_pull | bool }}"        imageLabels: "{{ openshift_buildoverrides_image_labels | default(None) }}"        nodeSelector: "{{ openshift_buildoverrides_nodeselectors | default(None) }}"        annotations: "{{ openshift_buildoverrides_annotations | default(None) }}" diff --git a/roles/openshift_docker_facts/tasks/main.yml b/roles/openshift_docker_facts/tasks/main.yml index 613c237a3..049ceffe0 100644 --- a/roles/openshift_docker_facts/tasks/main.yml +++ b/roles/openshift_docker_facts/tasks/main.yml @@ -9,6 +9,7 @@        additional_registries: "{{ openshift_docker_additional_registries | default(None) }}"        blocked_registries: "{{ openshift_docker_blocked_registries | default(None) }}"        insecure_registries: "{{ openshift_docker_insecure_registries | default(None) }}" +      selinux_enabled: "{{ openshift_docker_selinux_enabled | default(None) }}"        log_driver: "{{ openshift_docker_log_driver | default(None) }}"        log_options: "{{ openshift_docker_log_options | default(None) }}"        options: "{{ openshift_docker_options | default(None) }}" @@ -23,6 +24,7 @@                                     | default(omit) }}"      docker_insecure_registries: "{{ openshift.docker.insecure_registries                                      | default(omit) }}" +    docker_selinux_enabled: "{{ openshift.docker.selinux_enabled | default(omit) }}"      docker_log_driver: "{{ openshift.docker.log_driver | default(omit) }}"      docker_log_options: "{{ openshift.docker.log_options | default(omit) }}"      docker_push_dockerhub: "{{ openshift.docker.disable_push_dockerhub diff --git a/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-ephemeral-template.json b/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-ephemeral-template.json index 62ccc5b7f..ab1c85b7e 100644 --- a/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-ephemeral-template.json +++ b/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-ephemeral-template.json @@ -98,14 +98,6 @@                  },                  "env": [                    { -                    "name": "OPENSHIFT_ENABLE_OAUTH", -                    "value": "${ENABLE_OAUTH}" -                  }, -                  { -                    "name": "OPENSHIFT_ENABLE_REDIRECT_PROMPT", -                    "value": "true" -                  }, -                  {                      "name": "KUBERNETES_MASTER",                      "value": "https://kubernetes.default:443"                    }, @@ -245,12 +237,6 @@        "value": "jenkins-jnlp"      },      { -      "name": "ENABLE_OAUTH", -      "displayName": "Enable OAuth in Jenkins", -      "description": "Whether to enable OAuth OpenShift integration. If false, the static account 'admin' will be initialized with the password 'password'.", -      "value": "true" -    }, -    {        "name": "MEMORY_LIMIT",        "displayName": "Memory Limit",        "description": "Maximum amount of memory the container can use.", diff --git a/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-persistent-template.json b/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-persistent-template.json index 50c4ad566..87c439ad2 100644 --- a/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-persistent-template.json +++ b/roles/openshift_examples/files/examples/v1.3/quickstart-templates/jenkins-persistent-template.json @@ -115,14 +115,6 @@                  },                  "env": [                    { -                    "name": "OPENSHIFT_ENABLE_OAUTH", -                    "value": "${ENABLE_OAUTH}" -                  }, -                  { -                    "name": "OPENSHIFT_ENABLE_REDIRECT_PROMPT", -                    "value": "true" -                  }, -                  {                      "name": "KUBERNETES_MASTER",                      "value": "https://kubernetes.default:443"                    }, @@ -262,12 +254,6 @@        "value": "jenkins-jnlp"      },      { -      "name": "ENABLE_OAUTH", -      "displayName": "Enable OAuth in Jenkins", -      "description": "Whether to enable OAuth OpenShift integration. If false, the static account 'admin' will be initialized with the password 'password'.", -      "value": "true" -    }, -    {        "name": "MEMORY_LIMIT",        "displayName": "Memory Limit",        "description": "Maximum amount of memory the container can use.", diff --git a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/cakephp-mysql.json b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/cakephp-mysql.json index 9dbbf89d1..9732e59e1 100644 --- a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/cakephp-mysql.json +++ b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/cakephp-mysql.json @@ -22,8 +22,11 @@          "name": "${NAME}"        },        "stringData" : { -        "databaseUser" : "${DATABASE_USER}", -        "databasePassword" : "${DATABASE_PASSWORD}" +        "database-user" : "${DATABASE_USER}", +        "database-password" : "${DATABASE_PASSWORD}", +        "cakephp-secret-token" : "${CAKEPHP_SECRET_TOKEN}", +        "cakephp-security-salt" : "${CAKEPHP_SECURITY_SALT}", +        "cakephp-security-cipher-seed" : "${CAKEPHP_SECURITY_CIPHER_SEED}"        }      },      { @@ -97,12 +100,12 @@              "from": {                "kind": "ImageStreamTag",                "namespace": "${NAMESPACE}", -              "name": "php:5.6" +              "name": "php:7.0"              },              "env":  [                { -                  "name": "COMPOSER_MIRROR", -                  "value": "${COMPOSER_MIRROR}" +                "name": "COMPOSER_MIRROR", +                "value": "${COMPOSER_MIRROR}"                }              ]            } @@ -201,12 +204,12 @@                    }                  },                  "livenessProbe": { -                    "timeoutSeconds": 3, -                    "initialDelaySeconds": 30, -                    "httpGet": { -                        "path": "/", -                        "port": 8080 -                    } +                  "timeoutSeconds": 3, +                  "initialDelaySeconds": 30, +                  "httpGet": { +                    "path": "/", +                    "port": 8080 +                  }                  },                  "env": [                    { @@ -226,7 +229,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -235,21 +238,36 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    },                    {                      "name": "CAKEPHP_SECRET_TOKEN", -                    "value": "${CAKEPHP_SECRET_TOKEN}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "cakephp-secret-token" +                      } +                    }                    },                    {                      "name": "CAKEPHP_SECURITY_SALT", -                    "value": "${CAKEPHP_SECURITY_SALT}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "cakephp-security-salt" +                      } +                    }                    },                    {                      "name": "CAKEPHP_SECURITY_CIPHER_SEED", -                    "value": "${CAKEPHP_SECURITY_CIPHER_SEED}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "cakephp-security-cipher-seed" +                      } +                    }                    },                    {                      "name": "OPCACHE_REVALIDATE_FREQ", @@ -257,9 +275,9 @@                    }                  ],                  "resources": { -		    "limits": { -			"memory": "${MEMORY_LIMIT}" -		    } +                  "limits": { +                    "memory": "${MEMORY_LIMIT}" +                  }                  }                }              ] @@ -313,7 +331,7 @@                "from": {                  "kind": "ImageStreamTag",                  "namespace": "${NAMESPACE}", -                "name": "mysql:5.6" +                "name": "mysql:5.7"                }              }            }, @@ -362,40 +380,40 @@                    }                  },                  "livenessProbe": { -                    "timeoutSeconds": 1, -                    "initialDelaySeconds": 30, -                    "tcpSocket": { -                        "port": 3306 -                    } +                  "timeoutSeconds": 1, +                  "initialDelaySeconds": 30, +                  "tcpSocket": { +                    "port": 3306 +                  }                  },                  "env": [ -                    { -                      "name": "MYSQL_USER", -                      "valueFrom": { -                        "secretKeyRef" : { -                          "name" : "${NAME}", -                          "key" : "databaseUser" -                        } +                  { +                    "name": "MYSQL_USER", +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "database-user" +                      } +                    } +                  }, +                  { +                    "name": "MYSQL_PASSWORD", +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "database-password"                        } -                    }, -                    { -                      "name": "MYSQL_PASSWORD", -                      "valueFrom": { -                        "secretKeyRef" : { -                          "name" : "${NAME}", -                          "key" : "databasePassword" -                          } -                        } -                    }, -                    { -                        "name": "MYSQL_DATABASE", -                        "value": "${DATABASE_NAME}"                      } +                  }, +                  { +                    "name": "MYSQL_DATABASE", +                    "value": "${DATABASE_NAME}" +                  }                  ],                  "resources": { -                    "limits": { -                        "memory": "${MEMORY_MYSQL_LIMIT}" -                    } +                  "limits": { +                    "memory": "${MEMORY_MYSQL_LIMIT}" +                  }                  }                }              ] diff --git a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/dancer-mysql.json b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/dancer-mysql.json index dccb8bf7f..18100974b 100644 --- a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/dancer-mysql.json +++ b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/dancer-mysql.json @@ -22,8 +22,9 @@          "name": "${NAME}"        },        "stringData" : { -        "databaseUser" : "${DATABASE_USER}", -        "databasePassword" : "${DATABASE_PASSWORD}" +        "database-user" : "${DATABASE_USER}", +        "database-password" : "${DATABASE_PASSWORD}", +        "keybase" : "${SECRET_KEY_BASE}"        }      },      { @@ -97,7 +98,7 @@              "from": {                "kind": "ImageStreamTag",                "namespace": "${NAMESPACE}", -              "name": "perl:5.20" +              "name": "perl:5.24"              },              "env":  [                { @@ -207,7 +208,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -216,7 +217,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, @@ -226,7 +227,12 @@                    },                    {                      "name": "SECRET_KEY_BASE", -                    "value": "${SECRET_KEY_BASE}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "keybase" +                      } +                    }                    },                    {                      "name": "PERL_APACHE2_RELOAD", @@ -290,7 +296,7 @@                "from": {                  "kind": "ImageStreamTag",                  "namespace": "${NAMESPACE}", -                "name": "mysql:5.6" +                "name": "mysql:5.7"                }              }            }, @@ -351,7 +357,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -360,7 +366,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, diff --git a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/django-postgresql.json b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/django-postgresql.json index 59ff8a988..64b914e61 100644 --- a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/django-postgresql.json +++ b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/django-postgresql.json @@ -22,8 +22,9 @@          "name": "${NAME}"        },        "stringData" : { -        "databaseUser" : "${DATABASE_USER}", -        "databasePassword" : "${DATABASE_PASSWORD}" +        "database-user" : "${DATABASE_USER}", +        "database-password" : "${DATABASE_PASSWORD}", +        "django-secret-key" : "${DJANGO_SECRET_KEY}"        }      },      { @@ -218,7 +219,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -227,7 +228,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, @@ -237,7 +238,12 @@                    },                    {                      "name": "DJANGO_SECRET_KEY", -                    "value": "${DJANGO_SECRET_KEY}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "django-secret-key" +                      } +                    }                    }                  ],                  "resources": { @@ -338,7 +344,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -347,7 +353,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, diff --git a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/nodejs-mongodb.json b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/nodejs-mongodb.json index 91f9ec7b3..6a55f0251 100644 --- a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/nodejs-mongodb.json +++ b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/nodejs-mongodb.json @@ -22,9 +22,9 @@          "name": "${NAME}"        },        "stringData": { -        "databaseUser": "${DATABASE_USER}", -        "databasePassword": "${DATABASE_PASSWORD}", -        "databaseAdminPassword" : "${DATABASE_ADMIN_PASSWORD}" +        "database-user": "${DATABASE_USER}", +        "database-password": "${DATABASE_PASSWORD}", +        "database-admin-password" : "${DATABASE_ADMIN_PASSWORD}"        }      },      { @@ -201,7 +201,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -210,7 +210,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, @@ -223,7 +223,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseAdminPassword" +                        "key" : "database-admin-password"                        }                      }                    } @@ -336,7 +336,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -345,7 +345,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, @@ -358,7 +358,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseAdminPassword" +                        "key" : "database-admin-password"                        }                      }                    } diff --git a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/rails-postgresql.json b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/rails-postgresql.json index 6373562c4..043554c79 100644 --- a/roles/openshift_examples/files/examples/v1.5/quickstart-templates/rails-postgresql.json +++ b/roles/openshift_examples/files/examples/v1.5/quickstart-templates/rails-postgresql.json @@ -22,11 +22,11 @@          "name": "${NAME}"        },        "stringData" : { -        "databaseUser" : "${DATABASE_USER}", -        "databasePassword" : "${DATABASE_PASSWORD}", -        "applicationUser" : "${APPLICATION_USER}", -        "applicationPassword" : "${APPLICATION_PASSWORD}", -        "keyBase" : "${SECRET_KEY_BASE}" +        "database-user" : "${DATABASE_USER}", +        "database-password" : "${DATABASE_PASSWORD}", +        "application-user" : "${APPLICATION_USER}", +        "application-password" : "${APPLICATION_PASSWORD}", +        "keybase" : "${SECRET_KEY_BASE}"        }      },      { @@ -104,8 +104,8 @@              },              "env": [                { -                  "name": "RUBYGEM_MIRROR", -                  "value": "${RUBYGEM_MIRROR}" +                "name": "RUBYGEM_MIRROR", +                "value": "${RUBYGEM_MIRROR}"                }              ]            } @@ -148,7 +148,7 @@          "strategy": {            "type": "Recreate",            "recreateParams": { -          "pre": { +            "pre": {                "failurePolicy": "Abort",                "execNewPod": {                  "command": [ @@ -224,7 +224,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databaseUser" +                        "key" : "database-user"                        }                      }                    }, @@ -233,7 +233,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "databasePassword" +                        "key" : "database-password"                        }                      }                    }, @@ -246,7 +246,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "keyBase" +                        "key" : "keybase"                        }                      }                    }, @@ -267,7 +267,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "applicationUser" +                        "key" : "application-user"                        }                      }                    }, @@ -276,7 +276,7 @@                      "valueFrom": {                        "secretKeyRef" : {                          "name" : "${NAME}", -                        "key" : "applicationPassword" +                        "key" : "application-password"                        }                      }                    }, @@ -286,9 +286,9 @@                    }                  ],                  "resources": { -		      "limits": { -			  "memory": "${MEMORY_LIMIT}" -		      } +                  "limits": { +                    "memory": "${MEMORY_LIMIT}" +                  }                  }                }              ] @@ -400,11 +400,21 @@                  "env": [                    {                      "name": "POSTGRESQL_USER", -                    "value": "${DATABASE_USER}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "database-user" +                      } +                    }                    },                    {                      "name": "POSTGRESQL_PASSWORD", -                    "value": "${DATABASE_PASSWORD}" +                    "valueFrom": { +                      "secretKeyRef" : { +                        "name" : "${NAME}", +                        "key" : "database-password" +                      } +                    }                    },                    {                      "name": "POSTGRESQL_DATABASE", @@ -420,9 +430,9 @@                    }                  ],                  "resources": { -		      "limits": { -			  "memory": "${MEMORY_POSTGRESQL_LIMIT}" -		      } +                  "limits": { +                    "memory": "${MEMORY_POSTGRESQL_LIMIT}" +                  }                  }                }              ] diff --git a/roles/openshift_facts/library/openshift_facts.py b/roles/openshift_facts/library/openshift_facts.py index 10121f82a..f7506bd63 100755 --- a/roles/openshift_facts/library/openshift_facts.py +++ b/roles/openshift_facts/library/openshift_facts.py @@ -1032,6 +1032,8 @@ def set_nodename(facts):      if 'node' in facts and 'common' in facts:          if 'cloudprovider' in facts and facts['cloudprovider']['kind'] == 'openstack':              facts['node']['nodename'] = facts['provider']['metadata']['hostname'].replace('.novalocal', '') +        elif 'cloudprovider' in facts and facts['cloudprovider']['kind'] == 'gce': +            facts['node']['nodename'] = '.'.split(facts['provider']['metadata']['hostname'])[0]          else:              facts['node']['nodename'] = facts['common']['hostname'].lower()      return facts @@ -1690,9 +1692,38 @@ def set_builddefaults_facts(facts):              if 'admission_plugin_config' not in facts['master']:                  facts['master']['admission_plugin_config'] = dict()              facts['master']['admission_plugin_config'].update(builddefaults['config']) +            # if the user didn't actually provide proxy values, delete the proxy env variable defaults. +            delete_empty_keys(facts['master']['admission_plugin_config']['BuildDefaults']['configuration']['env']) +      return facts +def delete_empty_keys(keylist): +    """ Delete dictionary elements from keylist where "value" is empty. + +        Args: +          keylist(list): A list of builddefault configuration envs. + +        Returns: +          none + +        Example: +          keylist = [{'name': 'HTTP_PROXY', 'value': 'http://file.rdu.redhat.com:3128'}, +                     {'name': 'HTTPS_PROXY', 'value': 'http://file.rdu.redhat.com:3128'}, +                     {'name': 'NO_PROXY', 'value': ''}] + +          After calling delete_empty_keys the provided list is modified to become: + +                    [{'name': 'HTTP_PROXY', 'value': 'http://file.rdu.redhat.com:3128'}, +                     {'name': 'HTTPS_PROXY', 'value': 'http://file.rdu.redhat.com:3128'}] +    """ +    count = 0 +    for i in range(0, len(keylist)): +        if len(keylist[i - count]['value']) == 0: +            del keylist[i - count] +            count += 1 + +  def set_buildoverrides_facts(facts):      """ Set build overrides diff --git a/roles/openshift_loadbalancer/defaults/main.yml b/roles/openshift_loadbalancer/defaults/main.yml index d096019af..6190383b6 100644 --- a/roles/openshift_loadbalancer/defaults/main.yml +++ b/roles/openshift_loadbalancer/defaults/main.yml @@ -2,7 +2,7 @@  haproxy_frontends:  - name: main    binds: -  - "*:8443" +  - "*:{{ openshift_master_api_port | default(8443) }}"    default_backend: default  haproxy_backends: diff --git a/roles/openshift_loadbalancer/tasks/main.yml b/roles/openshift_loadbalancer/tasks/main.yml index 400f80715..68bb4ace8 100644 --- a/roles/openshift_loadbalancer/tasks/main.yml +++ b/roles/openshift_loadbalancer/tasks/main.yml @@ -1,14 +1,31 @@  --- -- fail: msg="Cannot use containerized=true for load balancer hosts." -  when: openshift.common.is_containerized | bool -  - name: Install haproxy    package: name=haproxy state=present +  when: not openshift.common.is_containerized | bool + +- name: Pull haproxy image +  command: > +    docker pull {{ openshift.common.router_image }}:{{ openshift_image_tag }} +  when: openshift.common.is_containerized | bool + +- name: Create config directory for haproxy +  file: +    path: /etc/haproxy +    state: directory +  when: openshift.common.is_containerized | bool + +- name: Create the systemd unit files +  template: +    src: "haproxy.docker.service.j2" +    dest: "/etc/systemd/system/haproxy.service" +  when: openshift.common.is_containerized | bool +  notify: restart haproxy  - name: Configure systemd service directory for haproxy    file:      path: /etc/systemd/system/haproxy.service.d      state: directory +  when: not openshift.common.is_containerized | bool  # Work around ini_file create option in 2.2 which defaults to no  - name: Create limits.conf file @@ -19,6 +36,7 @@      owner: root      group: root    changed_when: false +  when: not openshift.common.is_containerized | bool  - name: Configure the nofile limits for haproxy    ini_file: @@ -27,6 +45,7 @@      option: LimitNOFILE      value: "{{ openshift_loadbalancer_limit_nofile | default(100000) }}"    notify: restart haproxy +  when: not openshift.common.is_containerized | bool  - name: Configure haproxy    template: diff --git a/roles/openshift_loadbalancer/templates/haproxy.cfg.j2 b/roles/openshift_loadbalancer/templates/haproxy.cfg.j2 index 79e695001..24fd635ec 100644 --- a/roles/openshift_loadbalancer/templates/haproxy.cfg.j2 +++ b/roles/openshift_loadbalancer/templates/haproxy.cfg.j2 @@ -1,16 +1,20 @@  # Global settings  #---------------------------------------------------------------------  global +    maxconn     {{ openshift_loadbalancer_global_maxconn | default(20000) }} +    log         /dev/log local0 info +{% if openshift.common.is_containerized | bool %} +    stats socket /var/lib/haproxy/run/haproxy.sock mode 600 level admin +{% else %}      chroot      /var/lib/haproxy      pidfile     /var/run/haproxy.pid -    maxconn     {{ openshift_loadbalancer_global_maxconn | default(20000) }}      user        haproxy      group       haproxy      daemon -    log         /dev/log local0 info      # turn on stats unix socket      stats socket /var/lib/haproxy/stats +{% endif %}  #---------------------------------------------------------------------  # common defaults that all the 'listen' and 'backend' sections will diff --git a/roles/openshift_loadbalancer/templates/haproxy.docker.service.j2 b/roles/openshift_loadbalancer/templates/haproxy.docker.service.j2 new file mode 100644 index 000000000..5385df3b7 --- /dev/null +++ b/roles/openshift_loadbalancer/templates/haproxy.docker.service.j2 @@ -0,0 +1,17 @@ +[Unit] +After=docker.service +Requires=docker.service +PartOf=docker.service + +[Service] +ExecStartPre=-/usr/bin/docker rm -f openshift_loadbalancer +ExecStart=/usr/bin/docker run --rm --name openshift_loadbalancer -p {{ openshift_master_api_port | default(8443) }}:{{ openshift_master_api_port | default(8443) }} -v /etc/haproxy/haproxy.cfg:/etc/haproxy/haproxy.cfg:ro --entrypoint=haproxy {{ openshift.common.router_image }}:{{ openshift_image_tag }} -f /etc/haproxy/haproxy.cfg +ExecStartPost=/usr/bin/sleep 10 +ExecStop=/usr/bin/docker stop openshift_loadbalancer +LimitNOFILE={{ openshift_loadbalancer_limit_nofile | default(100000) }} +LimitCORE=infinity +Restart=always +RestartSec=5s + +[Install] +WantedBy=docker.service diff --git a/roles/openshift_logging/README.md b/roles/openshift_logging/README.md index 2cc2c48ee..856cfa2b9 100644 --- a/roles/openshift_logging/README.md +++ b/roles/openshift_logging/README.md @@ -6,6 +6,9 @@ This role is used for installing the Aggregated Logging stack. It should be run  a single host, it will create any missing certificates and API objects that the current  [logging deployer](https://github.com/openshift/origin-aggregated-logging/tree/master/deployer) does. +This role requires that the control host it is run on has Java installed as part of keystore +generation for Elasticsearch (it uses JKS) as well as openssl to sign certificates. +  As part of the installation, it is recommended that you add the Fluentd node selector label  to the list of persisted [node labels](https://docs.openshift.org/latest/install_config/install/advanced_install.html#configuring-node-host-labels). @@ -32,6 +35,7 @@ When both `openshift_logging_install_logging` and `openshift_logging_upgrade_log  - `openshift_logging_curator_log_level`: The log level for the Curator process. Defaults to 'ERROR'.  - `openshift_logging_curator_cpu_limit`: The amount of CPU to allocate to Curator. Default is '100m'.  - `openshift_logging_curator_memory_limit`: The amount of memory to allocate to Curator. Unset if not specified. +- `openshift_logging_curator_nodeselector`: A map of labels (e.g. {"node":"infra","region":"west"} to select the nodes where the curator pod will land.  - `openshift_logging_kibana_hostname`: The Kibana hostname. Defaults to 'kibana.example.com'.  - `openshift_logging_kibana_cpu_limit`: The amount of CPU to allocate to Kibana or unset if not specified. @@ -40,6 +44,7 @@ When both `openshift_logging_install_logging` and `openshift_logging_upgrade_log  - `openshift_logging_kibana_proxy_cpu_limit`: The amount of CPU to allocate to Kibana proxy or unset if not specified.  - `openshift_logging_kibana_proxy_memory_limit`: The amount of memory to allocate to Kibana proxy or unset if not specified.  - `openshift_logging_kibana_replica_count`: The number of replicas Kibana should be scaled up to. Defaults to 1. +- `openshift_logging_kibana_nodeselector`: A map of labels (e.g. {"node":"infra","region":"west"} to select the nodes where the pod will land.  - `openshift_logging_fluentd_nodeselector`: The node selector that the Fluentd daemonset uses to determine where to deploy to. Defaults to '"logging-infra-fluentd": "true"'.  - `openshift_logging_fluentd_cpu_limit`: The CPU limit for Fluentd pods. Defaults to '100m'. @@ -64,6 +69,7 @@ When both `openshift_logging_install_logging` and `openshift_logging_upgrade_log  - `openshift_logging_es_pvc_prefix`: The prefix for the generated PVCs. Defaults to 'logging-es'.  - `openshift_logging_es_recover_after_time`: The amount of time ES will wait before it tries to recover. Defaults to '5m'.  - `openshift_logging_es_storage_group`: The storage group used for ES. Defaults to '65534'. +- `openshift_logging_es_nodeselector`: A map of labels (e.g. {"node":"infra","region":"west"} to select the nodes where the pod will land.  When `openshift_logging_use_ops` is `True`, there are some additional vars. These work the  same as above for their non-ops counterparts, but apply to the OPS cluster instance: diff --git a/roles/openshift_logging/files/generate-jks.sh b/roles/openshift_logging/files/generate-jks.sh index 995ec0b98..9fe557f83 100644 --- a/roles/openshift_logging/files/generate-jks.sh +++ b/roles/openshift_logging/files/generate-jks.sh @@ -1,6 +1,10 @@  #! /bin/sh  set -ex +function usage() { +  echo Usage: `basename $0` cert_directory [logging_namespace] 1>&2 +} +  function generate_JKS_chain() {      dir=${SCRATCH_DIR:-_output}      ADD_OID=$1 @@ -147,8 +151,14 @@ function createTruststore() {      -noprompt -alias sig-ca  } -dir="$CERT_DIR" +if [ $# -lt 1 ]; then +  usage +  exit 1 +fi + +dir=$1  SCRATCH_DIR=$dir +PROJECT=${2:-logging}  if [[ ! -f $dir/system.admin.jks || -z "$(keytool -list -keystore $dir/system.admin.jks -storepass kspass | grep sig-ca)" ]]; then    generate_JKS_client_cert "system.admin" diff --git a/roles/openshift_logging/tasks/generate_certs.yaml b/roles/openshift_logging/tasks/generate_certs.yaml index e16071e46..740e490e1 100644 --- a/roles/openshift_logging/tasks/generate_certs.yaml +++ b/roles/openshift_logging/tasks/generate_certs.yaml @@ -85,133 +85,15 @@    loop_control:      loop_var: node_name -- name: Check for jks-generator service account -  command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get serviceaccount/jks-generator --no-headers -n {{openshift_logging_namespace}} -  register: serviceaccount_result -  ignore_errors: yes -  when: not ansible_check_mode -  changed_when: no - -- name: Create jks-generator service account -  command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig create serviceaccount jks-generator -n {{openshift_logging_namespace}} -  when: not ansible_check_mode and "not found" in serviceaccount_result.stderr - -- name: Check for hostmount-anyuid scc entry -  command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get scc hostmount-anyuid -o jsonpath='{.users}' -  register: scc_result -  when: not ansible_check_mode -  changed_when: no - -- name: Add to hostmount-anyuid scc -  command: > -    {{ openshift.common.admin_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig policy add-scc-to-user hostmount-anyuid -z jks-generator -n {{openshift_logging_namespace}} -  when: -    - not ansible_check_mode -    - scc_result.stdout.find("system:serviceaccount:{{openshift_logging_namespace}}:jks-generator") == -1 - -- name: Copy JKS generation script -  copy: -    src: generate-jks.sh -    dest: "{{generated_certs_dir}}/generate-jks.sh" -  check_mode: no +- name: Creating necessary JKS certs +  include: generate_jks.yaml -- name: Generate JKS pod template -  template: -    src: jks_pod.j2 -    dest: "{{mktemp.stdout}}/jks_pod.yaml" -  check_mode: no -  changed_when: no - -# check if pod generated files exist -- if they all do don't run the pod -- name: Checking for elasticsearch.jks -  stat: path="{{generated_certs_dir}}/elasticsearch.jks" -  register: elasticsearch_jks -  check_mode: no - -- name: Checking for logging-es.jks -  stat: path="{{generated_certs_dir}}/logging-es.jks" -  register: logging_es_jks -  check_mode: no - -- name: Checking for system.admin.jks -  stat: path="{{generated_certs_dir}}/system.admin.jks" -  register: system_admin_jks -  check_mode: no - -- name: Checking for truststore.jks -  stat: path="{{generated_certs_dir}}/truststore.jks" -  register: truststore_jks -  check_mode: no - -- name: create JKS generation pod -  command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig create -f {{mktemp.stdout}}/jks_pod.yaml -n {{openshift_logging_namespace}} -o name -  register: podoutput -  check_mode: no -  when: not elasticsearch_jks.stat.exists or not logging_es_jks.stat.exists or not system_admin_jks.stat.exists or not truststore_jks.stat.exists - -- command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get {{podoutput.stdout}} -o jsonpath='{.status.phase}' -n {{openshift_logging_namespace}} -  register: result -  until: result.stdout.find("Succeeded") != -1 -  retries: 5 -  delay: 10 -  changed_when: no -  when: not elasticsearch_jks.stat.exists or not logging_es_jks.stat.exists or not system_admin_jks.stat.exists or not truststore_jks.stat.exists - -# check for secret/logging-kibana-proxy -- command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get secret/logging-kibana-proxy -n {{openshift_logging_namespace}} -o jsonpath='{.data.oauth-secret}' -  register: kibana_secret_oauth_check -  ignore_errors: yes -  changed_when: no -  check_mode: no - -- command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get secret/logging-kibana-proxy -n {{openshift_logging_namespace}} -o jsonpath='{.data.session-secret}' -  register: kibana_secret_session_check -  ignore_errors: yes -  changed_when: no -  check_mode: no - -# check for oauthclient secret -- command: > -    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get oauthclient/kibana-proxy -n {{openshift_logging_namespace}} -o jsonpath='{.secret}' -  register: oauth_secret_check -  ignore_errors: yes -  changed_when: no -  check_mode: no - -# set or generate as needed +# TODO: make idempotent  - name: Generate proxy session    set_fact: session_secret={{'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'|random_word(200)}}    check_mode: no -  when: -    - kibana_secret_session_check.stdout is not defined or kibana_secret_session_check.stdout == '' - -- name: Generate proxy session -  set_fact: session_secret={{kibana_secret_session_check.stdout | b64decode }} -  check_mode: no -  when: -    - kibana_secret_session_check.stdout is defined -    - kibana_secret_session_check.stdout != '' +# TODO: make idempotent  - name: Generate oauth client secret    set_fact: oauth_secret={{'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'|random_word(64)}}    check_mode: no -  when: kibana_secret_oauth_check.stdout is not defined or kibana_secret_oauth_check.stdout == '' -    or oauth_secret_check.stdout is not defined or oauth_secret_check.stdout == '' -    or kibana_secret_oauth_check.stdout | b64decode != oauth_secret_check.stdout - -- name: Generate oauth client secret -  set_fact: oauth_secret={{kibana_secret_oauth_check.stdout | b64decode}} -  check_mode: no -  when: -    - kibana_secret_oauth_check is defined -    - kibana_secret_oauth_check.stdout != '' -    - oauth_secret_check.stdout is defined -    - oauth_secret_check.stdout != '' -    - kibana_secret_oauth_check.stdout | b64decode == oauth_secret_check.stdout diff --git a/roles/openshift_logging/tasks/generate_jks.yaml b/roles/openshift_logging/tasks/generate_jks.yaml new file mode 100644 index 000000000..c6e2ccbc0 --- /dev/null +++ b/roles/openshift_logging/tasks/generate_jks.yaml @@ -0,0 +1,98 @@ +--- +# check if pod generated files exist -- if they all do don't run the pod +- name: Checking for elasticsearch.jks +  stat: path="{{generated_certs_dir}}/elasticsearch.jks" +  register: elasticsearch_jks +  check_mode: no + +- name: Checking for logging-es.jks +  stat: path="{{generated_certs_dir}}/logging-es.jks" +  register: logging_es_jks +  check_mode: no + +- name: Checking for system.admin.jks +  stat: path="{{generated_certs_dir}}/system.admin.jks" +  register: system_admin_jks +  check_mode: no + +- name: Checking for truststore.jks +  stat: path="{{generated_certs_dir}}/truststore.jks" +  register: truststore_jks +  check_mode: no + +- name: Create temp directory for doing work in +  local_action: command mktemp -d /tmp/openshift-logging-ansible-XXXXXX +  register: local_tmp +  changed_when: False +  check_mode: no + +- name: Create placeholder for previously created JKS certs to prevent recreating... +  local_action: file path="{{local_tmp.stdout}}/elasticsearch.jks" state=touch mode="u=rw,g=r,o=r" +  when: elasticsearch_jks.stat.exists +  changed_when: False + +- name: Create placeholder for previously created JKS certs to prevent recreating... +  local_action: file path="{{local_tmp.stdout}}/logging-es.jks" state=touch mode="u=rw,g=r,o=r" +  when: logging_es_jks.stat.exists +  changed_when: False + +- name: Create placeholder for previously created JKS certs to prevent recreating... +  local_action: file path="{{local_tmp.stdout}}/system.admin.jks" state=touch mode="u=rw,g=r,o=r" +  when: system_admin_jks.stat.exists +  changed_when: False + +- name: Create placeholder for previously created JKS certs to prevent recreating... +  local_action: file path="{{local_tmp.stdout}}/truststore.jks" state=touch mode="u=rw,g=r,o=r" +  when: truststore_jks.stat.exists +  changed_when: False + +- name: pulling down signing items from host +  fetch: +    src: "{{generated_certs_dir}}/{{item}}" +    dest: "{{local_tmp.stdout}}/{{item}}" +    flat: yes +  with_items: +    - ca.crt +    - ca.key +    - ca.serial.txt +    - ca.crl.srl +    - ca.db +  when: not elasticsearch_jks.stat.exists or not logging_es_jks.stat.exists or not system_admin_jks.stat.exists or not truststore_jks.stat.exists + +- local_action: template src=signing.conf.j2 dest={{local_tmp.stdout}}/signing.conf +  vars: +    - top_dir: "{{local_tmp.stdout}}" +  when: not elasticsearch_jks.stat.exists or not logging_es_jks.stat.exists or not system_admin_jks.stat.exists or not truststore_jks.stat.exists + +- name: Run JKS generation script +  local_action: script generate-jks.sh {{local_tmp.stdout}} {{openshift_logging_namespace}} +  check_mode: no +  when: not elasticsearch_jks.stat.exists or not logging_es_jks.stat.exists or not system_admin_jks.stat.exists or not truststore_jks.stat.exists + +- name: Pushing locally generated JKS certs to remote host... +  copy: +    src: "{{local_tmp.stdout}}/elasticsearch.jks" +    dest: "{{generated_certs_dir}}/elasticsearch.jks" +  when: not elasticsearch_jks.stat.exists + +- name: Pushing locally generated JKS certs to remote host... +  copy: +    src: "{{local_tmp.stdout}}/logging-es.jks" +    dest: "{{generated_certs_dir}}/logging-es.jks" +  when: not logging_es_jks.stat.exists + +- name: Pushing locally generated JKS certs to remote host... +  copy: +    src: "{{local_tmp.stdout}}/system.admin.jks" +    dest: "{{generated_certs_dir}}/system.admin.jks" +  when: not system_admin_jks.stat.exists + +- name: Pushing locally generated JKS certs to remote host... +  copy: +    src: "{{local_tmp.stdout}}/truststore.jks" +    dest: "{{generated_certs_dir}}/truststore.jks" +  when: not truststore_jks.stat.exists + +- name: Cleaning up temp dir +  local_action: file path="{{local_tmp.stdout}}" state=absent +  changed_when: False diff --git a/roles/openshift_logging/tasks/install_curator.yaml b/roles/openshift_logging/tasks/install_curator.yaml index 8f2825552..fcfce4e1e 100644 --- a/roles/openshift_logging/tasks/install_curator.yaml +++ b/roles/openshift_logging/tasks/install_curator.yaml @@ -31,6 +31,7 @@      curator_cpu_limit: "{{openshift_logging_curator_cpu_limit }}"      curator_memory_limit: "{{openshift_logging_curator_memory_limit }}"      replicas: "{{curator_replica_count.stdout | default (0)}}" +    curator_node_selector: "{{openshift_logging_curator_nodeselector | default({}) }}"    check_mode: no    changed_when: no @@ -46,6 +47,7 @@      curator_cpu_limit: "{{openshift_logging_curator_ops_cpu_limit }}"      curator_memory_limit: "{{openshift_logging_curator_ops_memory_limit }}"      replicas: "{{curator_ops_replica_count.stdout | default (0)}}" +    curator_node_selector: "{{openshift_logging_curator_ops_nodeselector | default({}) }}"    when: openshift_logging_use_ops    check_mode: no    changed_when: no diff --git a/roles/openshift_logging/tasks/install_elasticsearch.yaml b/roles/openshift_logging/tasks/install_elasticsearch.yaml index fbba46a35..9b1c004f2 100644 --- a/roles/openshift_logging/tasks/install_elasticsearch.yaml +++ b/roles/openshift_logging/tasks/install_elasticsearch.yaml @@ -33,6 +33,7 @@      volume_names: "{{es_pvc_pool | default([])}}"      pvc_claim: "{{(volume_names | length > item.0) | ternary(volume_names[item.0], None)}}"      deploy_name: "{{item.1}}" +    es_node_selector: "{{openshift_logging_es_nodeselector | default({})}}"    with_indexed_items:      - "{{es_dc_pool | default([])}}"    check_mode: no @@ -98,6 +99,7 @@      es_recover_after_nodes: "{{es_ops_recover_after_nodes}}"      es_recover_expected_nodes: "{{es_ops_recover_expected_nodes}}"      openshift_logging_es_recover_after_time: "{{openshift_logging_es_ops_recover_after_time}}" +    es_node_selector: "{{openshift_logging_es_ops_nodeselector | default({})}}"    with_indexed_items:      - "{{es_dc_pool_ops | default([])}}"    when: diff --git a/roles/openshift_logging/tasks/install_kibana.yaml b/roles/openshift_logging/tasks/install_kibana.yaml index de4b018dd..f4df7de0c 100644 --- a/roles/openshift_logging/tasks/install_kibana.yaml +++ b/roles/openshift_logging/tasks/install_kibana.yaml @@ -35,6 +35,7 @@      kibana_proxy_cpu_limit: "{{openshift_logging_kibana_proxy_cpu_limit }}"      kibana_proxy_memory_limit: "{{openshift_logging_kibana_proxy_memory_limit }}"      replicas: "{{kibana_replica_count.stdout | default (0)}}" +    kibana_node_selector: "{{openshift_logging_kibana_nodeselector | default({}) }}"    check_mode: no    changed_when: no @@ -53,6 +54,7 @@      kibana_proxy_cpu_limit: "{{openshift_logging_kibana_ops_proxy_cpu_limit }}"      kibana_proxy_memory_limit: "{{openshift_logging_kibana_ops_proxy_memory_limit }}"      replicas: "{{kibana_ops_replica_count.stdout | default (0)}}" +    kibana_node_selector: "{{openshift_logging_kibana_ops_nodeselector | default({}) }}"    when: openshift_logging_use_ops    check_mode: no    changed_when: no diff --git a/roles/openshift_logging/tasks/install_logging.yaml b/roles/openshift_logging/tasks/install_logging.yaml index af03e9371..a9699adb8 100644 --- a/roles/openshift_logging/tasks/install_logging.yaml +++ b/roles/openshift_logging/tasks/install_logging.yaml @@ -23,23 +23,30 @@    loop_control:      loop_var: install_component +- find: paths={{ mktemp.stdout }}/templates patterns=*.yaml +  register: object_def_files +  changed_when: no + +- slurp: src={{item}} +  register: object_defs +  with_items: "{{object_def_files.files | map(attribute='path') | list | sort}}" +  changed_when: no +  - name: Create objects    include: oc_apply.yaml    vars:      - kubeconfig: "{{ mktemp.stdout }}/admin.kubeconfig"      - namespace: "{{ openshift_logging_namespace }}" -    - file_name: "{{ file }}" -    - file_content: "{{ lookup('file', file) | from_yaml }}" -  with_fileglob: -    - "{{ mktemp.stdout }}/templates/*.yaml" +    - file_name: "{{ file.source }}" +    - file_content: "{{ file.content | b64decode | from_yaml }}" +  with_items: "{{ object_defs.results }}"    loop_control:      loop_var: file    when: not ansible_check_mode  - name: Printing out objects to create -  debug: msg="{{lookup('file', file)|quote}}" -  with_fileglob: -    - "{{mktemp.stdout}}/templates/*.yaml" +  debug: msg={{file.content | b64decode }} +  with_items: "{{ object_defs.results }}"    loop_control:      loop_var: file    when: ansible_check_mode diff --git a/roles/openshift_logging/tasks/label_node.yaml b/roles/openshift_logging/tasks/label_node.yaml index aecb5d81b..bd5073381 100644 --- a/roles/openshift_logging/tasks/label_node.yaml +++ b/roles/openshift_logging/tasks/label_node.yaml @@ -1,11 +1,34 @@  ---  - command: >      {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get node {{host}} +    -o jsonpath='{.metadata.labels}' +  register: node_labels +  when: not ansible_check_mode +  changed_when: no + +- command: > +    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig label node {{host}} {{label}}={{value}} +  register: label_result +  failed_when: label_result.rc == 1 and 'exists' not in label_result.stderr +  when: +  - value is defined +  - node_labels.stdout is defined +  - label not in node_labels.stdout +  - unlabel is not defined or not unlabel +  - not ansible_check_mode + +- command: > +    {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig get node {{host}}      -o jsonpath='{.metadata.labels.{{ label }}}'    register: label_value -  failed_when: label_value.rc == 1 and 'exists' not in label_value.stderr -  when: not ansible_check_mode +  ignore_errors: yes    changed_when: no +  when: +  - value is defined +  - node_labels.stdout is defined +  - label in node_labels.stdout +  - unlabel is not defined or not unlabel +  - not ansible_check_mode  - command: >      {{ openshift.common.client_binary }} --config={{ mktemp.stdout }}/admin.kubeconfig label node {{host}} {{label}}={{value}} --overwrite diff --git a/roles/openshift_logging/tasks/main.yaml b/roles/openshift_logging/tasks/main.yaml index c4ec1b255..4c718805e 100644 --- a/roles/openshift_logging/tasks/main.yaml +++ b/roles/openshift_logging/tasks/main.yaml @@ -3,7 +3,6 @@      msg: Only one Fluentd nodeselector key pair should be provided    when: "{{ openshift_logging_fluentd_nodeselector.keys() | count }} > 1" -  - name: Create temp directory for doing work in    command: mktemp -d /tmp/openshift-logging-ansible-XXXXXX    register: mktemp diff --git a/roles/openshift_logging/templates/curator.j2 b/roles/openshift_logging/templates/curator.j2 index d3b5d33a2..de6258eaa 100644 --- a/roles/openshift_logging/templates/curator.j2 +++ b/roles/openshift_logging/templates/curator.j2 @@ -28,6 +28,12 @@ spec:      spec:        terminationGracePeriod: 600        serviceAccountName: aggregated-logging-curator +{% if curator_node_selector is iterable and curator_node_selector | length > 0 %} +      nodeSelector: +{% for key, value in curator_node_selector.iteritems() %} +        {{key}}: {{value}} +{% endfor %} +{% endif %}        containers:          -            name: "curator" diff --git a/roles/openshift_logging/templates/es.j2 b/roles/openshift_logging/templates/es.j2 index 291589690..ec84c6b76 100644 --- a/roles/openshift_logging/templates/es.j2 +++ b/roles/openshift_logging/templates/es.j2 @@ -30,6 +30,12 @@ spec:        securityContext:          supplementalGroups:          - {{openshift_logging_es_storage_group}} +{% if es_node_selector is iterable and es_node_selector | length > 0 %} +      nodeSelector: +{% for key, value in es_node_selector.iteritems() %} +        {{key}}: {{value}} +{% endfor %} +{% endif %}        containers:          -            name: "elasticsearch" diff --git a/roles/openshift_logging/templates/kibana.j2 b/roles/openshift_logging/templates/kibana.j2 index 1ec97701a..b42f62850 100644 --- a/roles/openshift_logging/templates/kibana.j2 +++ b/roles/openshift_logging/templates/kibana.j2 @@ -27,6 +27,12 @@ spec:          component: "{{component}}"      spec:        serviceAccountName: aggregated-logging-kibana +{% if kibana_node_selector is iterable and kibana_node_selector | length > 0 %} +      nodeSelector: +{% for key, value in kibana_node_selector.iteritems() %} +        {{key}}: {{value}} +{% endfor %} +{% endif %}        containers:          -            name: "kibana" diff --git a/roles/openshift_master_certificates/tasks/main.yml b/roles/openshift_master_certificates/tasks/main.yml index a1688aabc..4620dd877 100644 --- a/roles/openshift_master_certificates/tasks/main.yml +++ b/roles/openshift_master_certificates/tasks/main.yml @@ -105,7 +105,7 @@  - name: Create local temp directory for syncing certs    local_action: command mktemp -d /tmp/openshift-ansible-XXXXXXX -  register: g_master_mktemp +  register: g_master_certs_mktemp    changed_when: False    when: master_certs_missing | bool    delegate_to: localhost @@ -123,7 +123,7 @@  - name: Retrieve the master cert tarball from the master    fetch:      src: "{{ openshift_master_generated_config_dir }}.tgz" -    dest: "{{ g_master_mktemp.stdout }}/" +    dest: "{{ g_master_certs_mktemp.stdout }}/"      flat: yes      fail_on_missing: yes      validate_checksum: yes @@ -138,11 +138,11 @@  - name: Unarchive the tarball on the master    unarchive: -    src: "{{ g_master_mktemp.stdout }}/{{ openshift_master_cert_subdir }}.tgz" +    src: "{{ g_master_certs_mktemp.stdout }}/{{ openshift_master_cert_subdir }}.tgz"      dest: "{{ openshift_master_config_dir }}"    when: master_certs_missing | bool and inventory_hostname != openshift_ca_host -- file: name={{ g_master_mktemp.stdout }} state=absent +- file: name={{ g_master_certs_mktemp.stdout }} state=absent    changed_when: False    when: master_certs_missing | bool    delegate_to: localhost diff --git a/roles/openshift_metrics/README.md b/roles/openshift_metrics/README.md index 0f287e944..a61b0db5e 100644 --- a/roles/openshift_metrics/README.md +++ b/roles/openshift_metrics/README.md @@ -5,6 +5,10 @@ OpenShift Metrics Installation  Requirements  ------------ +This role has the following dependencies: + +- Java is required on the control node to generate keystores for the Java components +- httpd-tools is required on the control node to generate various passwords for the metrics components  The following variables need to be set and will be validated: diff --git a/roles/openshift_metrics/files/import_jks_certs.sh b/roles/openshift_metrics/files/import_jks_certs.sh index bb046df87..f4315ef34 100755 --- a/roles/openshift_metrics/files/import_jks_certs.sh +++ b/roles/openshift_metrics/files/import_jks_certs.sh @@ -114,5 +114,3 @@ function import_certs() {  }  import_certs - -exit 0 diff --git a/roles/openshift_metrics/tasks/import_jks_certs.yaml b/roles/openshift_metrics/tasks/import_jks_certs.yaml index f6bf6c1a6..f5192b005 100644 --- a/roles/openshift_metrics/tasks/import_jks_certs.yaml +++ b/roles/openshift_metrics/tasks/import_jks_certs.yaml @@ -1,76 +1,4 @@  --- -- name: Check for jks-generator service account -  command: > -    {{ openshift.common.client_binary }} -    --config={{ mktemp.stdout }}/admin.kubeconfig -    -n {{openshift_metrics_project}} -    get serviceaccount/jks-generator --no-headers -  register: serviceaccount_result -  ignore_errors: yes -  when: not ansible_check_mode -  changed_when: no - -- name: Create jks-generator service account -  command: > -    {{ openshift.common.client_binary }} -    --config={{ mktemp.stdout }}/admin.kubeconfig -    -n {{openshift_metrics_project}} -    create serviceaccount jks-generator -  when: not ansible_check_mode and "not found" in serviceaccount_result.stderr - -- name: Check for hostmount-anyuid scc entry -  command: > -    {{ openshift.common.client_binary }} -    --config={{ mktemp.stdout }}/admin.kubeconfig -    get scc hostmount-anyuid -    -o jsonpath='{.users}' -  register: scc_result -  when: not ansible_check_mode -  changed_when: no - -- name: Add to hostmount-anyuid scc -  command: > -    {{ openshift.common.admin_binary }} -    --config={{ mktemp.stdout }}/admin.kubeconfig -    -n {{openshift_metrics_project}} -    policy add-scc-to-user hostmount-anyuid -    -z jks-generator -  when: -    - not ansible_check_mode -    - scc_result.stdout.find("system:serviceaccount:{{openshift_metrics_project}}:jks-generator") == -1 - -- name: Copy JKS generation script -  copy: -    src: import_jks_certs.sh -    dest: "{{openshift_metrics_certs_dir}}/import_jks_certs.sh" -  check_mode: no - -- slurp: src={{ openshift_metrics_certs_dir }}/hawkular-metrics-keystore.pwd -  register: metrics_keystore_password - -- slurp: src={{ openshift_metrics_certs_dir }}/hawkular-cassandra-keystore.pwd -  register: cassandra_keystore_password - -- slurp: src={{ openshift_metrics_certs_dir }}/hawkular-jgroups-keystore.pwd -  register: jgroups_keystore_password - -- name: Generate JKS pod template -  template: -    src: jks_pod.j2 -    dest: "{{mktemp.stdout}}/jks_pod.yaml" -  vars: -    metrics_keystore_passwd: "{{metrics_keystore_password.content}}" -    cassandra_keystore_passwd: "{{cassandra_keystore_password.content}}" -    metrics_truststore_passwd: "{{hawkular_truststore_password.content}}" -    cassandra_truststore_passwd: "{{cassandra_truststore_password.content}}" -    jgroups_passwd: "{{jgroups_keystore_password.content}}" -  check_mode: no -  changed_when: no - -- stat: path="{{openshift_metrics_certs_dir}}/hawkular-metrics.keystore" -  register: metrics_keystore -  check_mode: no -  - stat: path="{{openshift_metrics_certs_dir}}/hawkular-cassandra.keystore"    register: cassandra_keystore    check_mode: no @@ -79,6 +7,10 @@    register: cassandra_truststore    check_mode: no +- stat: path="{{openshift_metrics_certs_dir}}/hawkular-metrics.keystore" +  register: metrics_keystore +  check_mode: no +  - stat: path="{{openshift_metrics_certs_dir}}/hawkular-metrics.truststore"    register: metrics_truststore    check_mode: no @@ -87,32 +19,52 @@    register: jgroups_keystore    check_mode: no -- name: create JKS pod -  command: > -    {{ openshift.common.client_binary }} -    --config={{ mktemp.stdout }}/admin.kubeconfig -    -n {{openshift_metrics_project}} -    create -f {{mktemp.stdout}}/jks_pod.yaml -    -o name -  register: podoutput -  check_mode: no -  when: not metrics_keystore.stat.exists or -        not metrics_truststore.stat.exists or -        not cassandra_keystore.stat.exists or -        not cassandra_truststore.stat.exists or -        not jgroups_keystore.stat.exists +- block: +  - slurp: src={{ openshift_metrics_certs_dir }}/hawkular-metrics-keystore.pwd +    register: metrics_keystore_password + +  - slurp: src={{ openshift_metrics_certs_dir }}/hawkular-cassandra-keystore.pwd +    register: cassandra_keystore_password + +  - slurp: src={{ openshift_metrics_certs_dir }}/hawkular-jgroups-keystore.pwd +    register: jgroups_keystore_password + +  - local_action: command mktemp -d +    register: local_tmp +    changed_when: False + +  - fetch: +      dest: "{{local_tmp.stdout}}/" +      src: "{{ openshift_metrics_certs_dir }}/{{item}}" +      flat: yes +    changed_when: False +    with_items: +    - hawkular-metrics.pkcs12 +    - hawkular-cassandra.pkcs12 +    - hawkular-metrics.crt +    - hawkular-cassandra.crt +    - ca.crt + +  - local_action: command {{role_path}}/files/import_jks_certs.sh +    environment: +      CERT_DIR: "{{local_tmp.stdout}}" +      METRICS_KEYSTORE_PASSWD: "{{metrics_keystore_password.content}}" +      CASSANDRA_KEYSTORE_PASSWD: "{{cassandra_keystore_password.content}}" +      METRICS_TRUSTSTORE_PASSWD: "{{hawkular_truststore_password.content}}" +      CASSANDRA_TRUSTSTORE_PASSWD: "{{cassandra_truststore_password.content}}" +      JGROUPS_PASSWD: "{{jgroups_keystore_password.content}}" +    changed_when: False + +  - copy: +      dest: "{{openshift_metrics_certs_dir}}/" +      src: "{{item}}" +    with_fileglob: "{{local_tmp.stdout}}/*.*store" + +  - file: +      path: "{{local_tmp.stdout}}" +      state: absent +    changed_when: False -- command: > -    {{ openshift.common.client_binary }} -    --config={{ mktemp.stdout }}/admin.kubeconfig -    -n {{openshift_metrics_project}} -    get {{podoutput.stdout}} -    -o jsonpath='{.status.phase}' -  register: result -  until: result.stdout.find("Succeeded") != -1 -  retries: 5 -  delay: 10 -  changed_when: no    when: not metrics_keystore.stat.exists or          not metrics_truststore.stat.exists or          not cassandra_keystore.stat.exists or diff --git a/roles/openshift_metrics/tasks/install_metrics.yaml b/roles/openshift_metrics/tasks/install_metrics.yaml index bab37dbfb..ddaa54438 100644 --- a/roles/openshift_metrics/tasks/install_metrics.yaml +++ b/roles/openshift_metrics/tasks/install_metrics.yaml @@ -20,15 +20,23 @@    loop_control:      loop_var: include_file +- find: paths={{ mktemp.stdout }}/templates patterns=*.yaml +  register: object_def_files +  changed_when: no + +- slurp: src={{item.path}} +  register: object_defs +  with_items: "{{object_def_files.files}}" +  changed_when: no +  - name: Create objects    include: oc_apply.yaml    vars:      kubeconfig: "{{ mktemp.stdout }}/admin.kubeconfig"      namespace: "{{ openshift_metrics_project }}" -    file_name: "{{ item }}" -    file_content: "{{ lookup('file',item) | from_yaml }}" -  with_fileglob: -    - "{{ mktemp.stdout }}/templates/*.yaml" +    file_name: "{{ item.source }}" +    file_content: "{{ item.content | b64decode | from_yaml }}" +  with_items: "{{ object_defs.results }}"  - name: Scaling up cluster    include: start_metrics.yaml diff --git a/roles/openshift_metrics/tasks/install_support.yaml b/roles/openshift_metrics/tasks/install_support.yaml index b0e4bec80..cc5acc6e5 100644 --- a/roles/openshift_metrics/tasks/install_support.yaml +++ b/roles/openshift_metrics/tasks/install_support.yaml @@ -1,4 +1,22 @@  --- +- name: Check control node to see if htpasswd is installed +  local_action: command which htpasswd +  register: htpasswd_check +  failed_when: no +  changed_when: no + +- fail: msg="'htpasswd' is unavailable. Please install httpd-tools on the control node" +  when: htpasswd_check.rc  == 1 + +- name: Check control node to see if keytool is installed +  local_action: command which htpasswd +  register: keytool_check +  failed_when: no +  changed_when: no + +- fail: msg="'keytool' is unavailable. Please install java-1.8.0-openjdk-headless on the control node" +  when: keytool_check.rc  == 1 +  - include: generate_certificates.yaml  - include: generate_serviceaccounts.yaml  - include: generate_services.yaml diff --git a/roles/openshift_metrics/tasks/main.yaml b/roles/openshift_metrics/tasks/main.yaml index c42440130..1808db5d5 100644 --- a/roles/openshift_metrics/tasks/main.yaml +++ b/roles/openshift_metrics/tasks/main.yaml @@ -7,6 +7,7 @@  - name: Create temp directory for all our templates    file: path={{mktemp.stdout}}/templates state=directory mode=0755    changed_when: False +  when: "{{ openshift_metrics_install_metrics | bool }}"  - name: Copy the admin client config(s)    command: > @@ -15,8 +16,4 @@    check_mode: no    tags: metrics_init -- include: install_metrics.yaml -  when: openshift_metrics_install_metrics | default(false) | bool - -- include: uninstall_metrics.yaml -  when: not openshift_metrics_install_metrics | default(false) | bool +- include: "{{ (openshift_metrics_install_metrics | bool) | ternary('install_metrics.yaml','uninstall_metrics.yaml') }}" diff --git a/roles/openshift_metrics/templates/jks_pod.j2 b/roles/openshift_metrics/templates/jks_pod.j2 deleted file mode 100644 index e86fe38a4..000000000 --- a/roles/openshift_metrics/templates/jks_pod.j2 +++ /dev/null @@ -1,38 +0,0 @@ -apiVersion: v1 -kind: Pod -metadata: -  labels: -    metrics-infra: support -  generateName: jks-cert-gen- -spec: -  containers: -  - name: jks-cert-gen -    image: {{openshift_metrics_image_prefix}}metrics-deployer:{{openshift_metrics_image_version}} -    imagePullPolicy: Always -    command: ["sh",  "{{openshift_metrics_certs_dir}}/import_jks_certs.sh"] -    securityContext: -      runAsUser: 0 -    volumeMounts: -    - mountPath: {{openshift_metrics_certs_dir}} -      name: certmount -    env: -    - name: CERT_DIR -      value: {{openshift_metrics_certs_dir}} -    - name: METRICS_KEYSTORE_PASSWD -      value: {{metrics_keystore_passwd}} -    - name: CASSANDRA_KEYSTORE_PASSWD -      value: {{cassandra_keystore_passwd}} -    - name: METRICS_TRUSTSTORE_PASSWD -      value: {{metrics_truststore_passwd}} -    - name: CASSANDRA_TRUSTSTORE_PASSWD -      value: {{cassandra_truststore_passwd}} -    - name: hawkular_cassandra_alias -      value: {{cassandra_keystore_passwd}} -    - name: JGROUPS_PASSWD -      value: {{jgroups_passwd}} -  restartPolicy: Never -  serviceAccount: jks-generator -  volumes: -  - hostPath: -      path: "{{openshift_metrics_certs_dir}}" -    name: certmount diff --git a/roles/openshift_node/meta/main.yml b/roles/openshift_node/meta/main.yml index 91f118191..10036abed 100644 --- a/roles/openshift_node/meta/main.yml +++ b/roles/openshift_node/meta/main.yml @@ -17,8 +17,6 @@ dependencies:  - role: openshift_docker  - role: openshift_node_certificates  - role: openshift_cloud_provider -- role: openshift_node_dnsmasq -  when: openshift.common.use_dnsmasq | bool  - role: os_firewall    os_firewall_allow:    - service: Kubernetes kubelet @@ -43,3 +41,5 @@ dependencies:    - service: Kubernetes service NodePort UDP      port: "{{ openshift_node_port_range | default('') }}/udp"    when: openshift_node_port_range is defined +- role: openshift_node_dnsmasq +  when: openshift.common.use_dnsmasq | bool diff --git a/roles/os_firewall/library/os_firewall_manage_iptables.py b/roles/os_firewall/library/os_firewall_manage_iptables.py index 8ba650994..8d4878fa7 100755 --- a/roles/os_firewall/library/os_firewall_manage_iptables.py +++ b/roles/os_firewall/library/os_firewall_manage_iptables.py @@ -223,7 +223,9 @@ class IpTablesManager(object):  # pylint: disable=too-many-instance-attributes      def gen_cmd(self):          cmd = 'iptables' if self.ip_version == 'ipv4' else 'ip6tables' -        return ["/usr/sbin/%s" % cmd] +        # Include -w (wait for xtables lock) in default arguments. +        default_args = ['-w'] +        return ["/usr/sbin/%s" % cmd] + default_args      def gen_save_cmd(self):  # pylint: disable=no-self-use          return ['/usr/libexec/iptables/iptables.init', 'save'] diff --git a/roles/os_firewall/tasks/firewall/firewalld.yml b/roles/os_firewall/tasks/firewall/firewalld.yml index 1101870be..c4db197ca 100644 --- a/roles/os_firewall/tasks/firewall/firewalld.yml +++ b/roles/os_firewall/tasks/firewall/firewalld.yml @@ -1,7 +1,8 @@  ---  - name: Install firewalld packages -  package: name=firewalld state=present -  when: not openshift.common.is_containerized | bool +  package: +    name: firewalld +    state: present  - name: Ensure iptables services are not enabled    systemd: diff --git a/roles/rhel_subscribe/tasks/enterprise.yml b/roles/rhel_subscribe/tasks/enterprise.yml index 291df6822..41673ee40 100644 --- a/roles/rhel_subscribe/tasks/enterprise.yml +++ b/roles/rhel_subscribe/tasks/enterprise.yml @@ -7,7 +7,7 @@    when: deployment_type == 'enterprise'  - set_fact: -    default_ose_version: '3.3' +    default_ose_version: '3.4'    when: deployment_type in ['atomic-enterprise', 'openshift-enterprise']  - set_fact: @@ -16,7 +16,7 @@  - fail:      msg: "{{ ose_version }} is not a valid version for {{ deployment_type }} deployment type"    when: ( deployment_type == 'enterprise' and ose_version not in ['3.0'] ) or -        ( deployment_type in ['atomic-enterprise', 'openshift-enterprise'] and ose_version not in ['3.1', '3.2', '3.3'] ) +        ( deployment_type in ['atomic-enterprise', 'openshift-enterprise'] and ose_version not in ['3.1', '3.2', '3.3', '3.4'] )  - name: Enable RHEL repositories    command: subscription-manager repos \ diff --git a/utils/test-requirements.txt b/utils/test-requirements.txt index f6a7bde10..aebfe7c39 100644 --- a/utils/test-requirements.txt +++ b/utils/test-requirements.txt @@ -13,3 +13,5 @@ pyOpenSSL  yamllint  tox  detox +# Temporary work-around for flake8 vs maccabe version conflict +mccabe==0.5.3 | 
