Browse Source

Merge pull request #8383 from kwoodson/azure_publishing_new

Adding ability to publish images to the azure portal.
Vadim Rutkovsky 6 years ago
parent
commit
6a8605a606

+ 83 - 0
playbooks/azure/openshift-cluster/create_offer.yml

@@ -0,0 +1,83 @@
+---
+- hosts: localhost
+  gather_facts: no
+  tasks:
+  - name: ensure libraries are available
+    import_role:
+      name: lib_utils
+
+  - name: set expire date
+    set_fact:
+      sas_expire: "{{ lookup('pipe', 'date -d \"14 days\" +\"%Y-%m-%dT%H:%M:00Z\"') }}"
+      disk: "{{ image_name }}.vhd"
+
+  - name: fetch the image for the tags
+    command: |
+      az image list --resource-group images  -o json --query "[?name=='{{ image_name }}'].tags"
+    register: imagetags
+
+  - name: generate storage blob url
+    command: |
+      az storage blob url -c {{ openshift_azure_container }} --account-name {{ openshift_azure_storage_account }} -n {{ disk }} --output=json
+    register: bloburl
+
+  - name: generate sas url for the container
+    command: |
+      az storage container generate-sas --name {{ openshift_azure_container }} --account-name {{ openshift_azure_storage_account }} --permissions rl --expiry {{ sas_expire }} --output=json
+    register: sasurl
+
+  - name: set the sas URLS
+    set_fact:
+      openshift_azure_sas_url: "{{ bloburl.stdout|from_json + '?' + sasurl.stdout|from_json }}"
+
+  - name: set the image tags
+    set_fact:
+      image_tags: "{{ (imagetags.stdout | from_json)[0] }}"
+
+  - name: set the image_type
+    set_fact:
+      image_type: "{{ 'rhel' if 'rhel' in image_tags.base_image else 'centos' }}"
+      image_vm_images: |-
+        { "{{ image_tags.openshift | regex_search('([0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3})') }}": {"osVhdUrl": "{{ openshift_azure_sas_url }}" } }
+
+  - name: fetch the current offer and update the versions
+    oo_azure_rm_publish_image_facts:
+      offer: "{{ azure_image_publish[image_type].offer }}"
+    register: offerout
+
+  - when:
+    - offerout['status_code'] != 404
+    block:
+    - debug:
+        msg: "{{ offerout }}"
+        verbosity: 1
+
+    - debug:
+        msg: "{{ offerout['data']['definition']['plans'][0]['microsoft-azure-virtualmachines.vmImages'] }}"
+        verbosity: 1
+
+    - debug:
+        msg: "{{ lookup('template', 'offer.yml.j2') }}"
+      verbosity: 1
+
+    - name: bring along the previous offer versions and combine with incoming
+      yedit:
+        content: "{{ lookup('template', 'offer.yml.j2') }}"
+        key: "definition#plans[0]#microsoft-azure-virtualmachines.vmImages#{{ item.key }}"
+        value: "{{ item.value }}"
+        separator: '#'
+      with_dict: "{{ offerout['data']['definition']['plans'][0]['microsoft-azure-virtualmachines.vmImages'] }}"
+      when:
+      - "'data' in offerout"
+      - "'definition' in offerout['data']"
+      register: yeditout
+
+    - debug:
+        msg: "{{ yeditout }}"
+        verbosity: 1
+
+  - name: create an offer in cloudpartner portal
+    oo_azure_rm_publish_image:
+      offer: "{{ azure_image_publish[image_type].offer }}"
+      offer_data: "{{ (lookup('template', 'offer.yml.j2') | from_yaml) if 'skipped' in yeditout and yeditout.skipped or not yeditout.changed else yeditout.results[0].result[0].edit }}"
+      force: "{{ openshift_azure_offer_force |default(False)}}"

+ 29 - 0
playbooks/azure/openshift-cluster/group_vars/all/image_publish.yml

@@ -0,0 +1,29 @@
+---
+azure_image_publish:
+  rhel:
+    offer: acsengine
+    offer_force: True
+    templ_display_text: Openshift for acsengine
+    templ_sku_summary: Openshift on Red Hat Enterprise Linux on Azure
+    templ_sku_description: Openshift on Red Hat Enterprise Linux
+    templ_offer_id: acsengine
+    templ_plan_id: rhel74
+    templ_title: Openshift on Red Hat Enterprise Linux
+    templ_os: rhel7
+    templ_description: Openshift image baked on Rhel
+    templ_summary: Openshift for Rhel
+    templ_marketing_url_id: openshift-acsengine
+
+  centos:
+    offer: origin-acsengine
+    offer_force: True
+    templ_display_text: origin for acsengine
+    templ_sku_description: Openshift Origin on CentOS 7
+    templ_sku_summary: Openshift Origin on CentOS on Azure
+    templ_offer_id: origin-acsengine
+    templ_plan_id: centos7
+    templ_title: CentOS 7 for Origin
+    templ_os: CentOS
+    templ_description: Openshift Origin on CentOS
+    templ_summary: Openshift Origin on CentOS
+    templ_marketing_url_id: origin-acsengine

File diff suppressed because it is too large
+ 59 - 0
playbooks/azure/openshift-cluster/templates/offer.yml.j2


+ 247 - 0
roles/lib_utils/library/oo_azure_rm_publish_image.py

@@ -0,0 +1,247 @@
+#!/usr/bin/env python
+# pylint: disable=missing-docstring
+# Copyright 2018 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.
+#
+
+from __future__ import print_function  # noqa: F401
+# import httplib
+import json
+import os
+import requests
+
+from ansible.module_utils.basic import AnsibleModule
+
+
+class AzurePublisherException(Exception):
+    '''Exception class for AzurePublisher'''
+    pass
+
+
+class AzurePublisher(object):
+    '''Python class to represent the Azure Publishing portal https://cloudpartner.azure.com'''
+
+    # pylint: disable=too-many-arguments
+    def __init__(self,
+                 publisher_id,
+                 client_info,
+                 ssl_verify=True,
+                 api_version='2017-10-31',
+                 debug=False):
+        '''
+          :publisher_id: string of the publisher id
+          :client_info: a dict containing the client_id, client_secret to get an access_token
+        '''
+        self._azure_server = 'https://cloudpartner.azure.com/api/publishers/{}'.format(publisher_id)
+        self.client_info = client_info
+        self.ssl_verify = ssl_verify
+        self.api_version = 'api-version={}'.format(api_version)
+        self.debug = debug
+        # if self.debug:
+        # httplib.HTTPSConnection.debuglevel = 1
+        # httplib.HTTPConnection.debuglevel = 1
+
+        self._access_token = None
+
+    @property
+    def server(self):
+        '''property for  server url'''
+        return self._azure_server
+
+    @property
+    def token(self):
+        '''property for the access_token
+            curl -d \
+            'client_id=<id>&client_secret=<sec>&grant_type=client_credentials&resource=https://cloudpartner.azure.com' \
+            https://login.microsoftonline.com/72f988bf-86f1-41af-91ab-2d7cd011db47/oauth2/token
+        '''
+        if self._access_token is None:
+            url = 'https://login.microsoftonline.com/{}/oauth2/token'.format(self.client_info['tenant_id'])
+            data = {
+                'client_id': {self.client_info['client_id']},
+                'client_secret': self.client_info['client_secret'],
+                'grant_type': 'client_credentials',
+                'resource': 'https://cloudpartner.azure.com'
+            }
+
+            results = AzurePublisher.request('POST', url, data, {})
+            jres = results.json()
+            self._access_token = jres['access_token']
+
+        return self._access_token
+
+    def get_offers(self, offer=None, version=None, slot='preview'):
+        ''' fetch all offers by publisherid '''
+        url = '/offers'
+
+        if offer is not None:
+            url += '/{}'.format(offer)
+            if version is not None:
+                url += '/versions/{}'.format(version)
+            if slot == 'preview':
+                url += '/slot/{}'.format(slot)
+
+        url += '?{}'.format(self.api_version)
+
+        return self.prepare_action(url)
+
+    def get_operations(self, offer, operation=None, status=None):
+        ''' create or modify an offer '''
+        url = '/offers/{0}/submissions'.format(offer)
+
+        if operation is not None:
+            url += '/operations/{0}'.format(operation)
+
+        if not url.endswith('/'):
+            url += '/'
+
+        url += '?{0}'.format(self.api_version)
+
+        if status is not None:
+            url += '&status={0}'.format(status)
+
+        return self.prepare_action(url, 'GET')
+
+    def cancel_operation(self, offer):
+        ''' create or modify an offer '''
+        url = '/offers/{0}/cancel?{1}'.format(offer, self.api_version)
+
+        return self.prepare_action(url, 'POST')
+
+    def go_live(self, offer):
+        ''' create or modify an offer '''
+        url = '/offers/{0}/golive?{1}'.format(offer, self.api_version)
+
+        return self.prepare_action(url, 'POST')
+
+    def create_or_modify_offer(self, offer, data=None, modify=False):
+        ''' create or modify an offer '''
+        url = '/offers/{0}?{1}'.format(offer, self.api_version)
+
+        headers = None
+
+        if modify:
+            headers = {
+                'If-Match': '*',
+            }
+
+        return self.prepare_action(url, 'PUT', data=data, add_headers=headers)
+
+    def prepare_action(self, url, action='GET', data=None, add_headers=None):
+        '''perform the http request
+
+           :action: string of either GET|POST
+        '''
+        headers = {
+            'Content-Type': 'application/json',
+            'Accept': 'application/json',
+            'Authorization': 'Bearer {}'.format(self.token)
+        }
+
+        if add_headers is not None:
+            headers.update(add_headers)
+
+        if data is None:
+            data = ''
+        else:
+            data = json.dumps(data)
+
+        return AzurePublisher.request(action.upper(), self.server + url, data, headers)
+
+    def manage_offer(self, params):
+        ''' handle creating or modifying offers'''
+        # fetch the offer to verify it exists:
+        results = self.get_offers(offer=params['offer'])
+
+        if (results.status_code == 200 or results.status_code == 404) and params['force']:
+            return self.create_or_modify_offer(offer=params['offer'], data=params['offer_data'], modify=True)
+
+        return self.create_or_modify_offer(offer=params['offer'], data=params['offer_data'])
+
+    @staticmethod
+    def request(action, url, data=None, headers=None, ssl_verify=True):
+        req = requests.Request(action.upper(), url, data=data, headers=headers)
+
+        session = requests.Session()
+        req_prep = session.prepare_request(req)
+        response = session.send(req_prep, verify=ssl_verify)
+
+        return response
+
+    @staticmethod
+    def run_ansible(params):
+        '''perform the ansible operations'''
+        client_info = {
+            'tenant_id': params['tenant_id'],
+            'client_id': params['client_id'],
+            'client_secret': params['client_secret']}
+
+        apc = AzurePublisher(params['publisher'],
+                             client_info,
+                             debug=params['debug'])
+
+        if params['state'] == 'offer':
+            results = apc.manage_offer(params)
+        elif params['state'] == 'cancel_op':
+            results = apc.cancel_operation(offer=params['offer'])
+        elif params['state'] == 'go_live':
+            results = apc.go_live(offer=params['offer'])
+        else:
+            raise AzurePublisherException('Unsupported query type: {}'.format(params['query']))
+
+        changed = False
+
+        if results.status_code in [200, 201]:
+            changed = True
+
+        return {'data': results.json(), 'changed': changed, 'status_code': results.status_code}
+
+
+# pylint: disable=too-many-branches
+def main():
+    ''' ansible oc module for secrets '''
+
+    module = AnsibleModule(
+        argument_spec=dict(
+            state=dict(default='offer', choices=['offer', 'cancel_op', 'go_live']),
+            force=dict(default=False, type='bool'),
+            publisher=dict(default='redhat', type='str'),
+            debug=dict(default=False, type='bool'),
+            tenant_id=dict(default=os.environ.get('AZURE_TENANT_ID'), type='str'),
+            client_id=dict(default=os.environ.get('AZURE_CLIENT_ID'), type='str'),
+            client_secret=dict(default=os.environ.get('AZURE_CLIENT_SECRET'), type='str'),
+            offer=dict(default=None, type='str'),
+            offer_data=dict(required=True, type='dict'),
+        ),
+    )
+
+    # Verify we recieved either a valid key or edits with valid keys when receiving a src file.
+    # A valid key being not None or not ''.
+    if (module.params['tenant_id'] is None or module.params['client_id'] is None or
+            module.params['client_secret'] is None):
+        return module.fail_json(**{'failed': True,
+                                   'msg': 'Please specify tenant_id, client_id, and client_secret'})
+
+    rval = AzurePublisher.run_ansible(module.params)
+
+    if int(rval['status_code']) >= 300:
+        rval['msg'] = 'Failed. status_code {}'.format(rval['status_code'])
+        return module.fail_json(**rval)
+
+    return module.exit_json(**rval)
+
+
+if __name__ == '__main__':
+    main()

+ 240 - 0
roles/lib_utils/library/oo_azure_rm_publish_image_facts.py

@@ -0,0 +1,240 @@
+#!/usr/bin/env python
+# pylint: disable=missing-docstring
+# Copyright 2018 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.
+#
+
+from __future__ import print_function  # noqa: F401
+# import httplib
+import json
+import os
+import requests
+
+from ansible.module_utils.basic import AnsibleModule
+
+
+class AzurePublisherException(Exception):
+    '''Exception class for AzurePublisher'''
+    pass
+
+
+class AzurePublisher(object):
+    '''Python class to represent the Azure Publishing portal https://cloudpartner.azure.com'''
+
+    # pylint: disable=too-many-arguments
+    def __init__(self,
+                 publisher_id,
+                 client_info,
+                 ssl_verify=True,
+                 api_version='2017-10-31',
+                 debug=False):
+        '''
+          :publisher_id: string of the publisher id
+          :client_info: a dict containing the client_id, client_secret to get an access_token
+        '''
+        self._azure_server = 'https://cloudpartner.azure.com/api/publishers/{}'.format(publisher_id)
+        self.client_info = client_info
+        self.ssl_verify = ssl_verify
+        self.api_version = 'api-version={}'.format(api_version)
+        self.debug = debug
+        # if self.debug:
+        # httplib.HTTPSConnection.debuglevel = 1
+        # httplib.HTTPConnection.debuglevel = 1
+
+        self._access_token = None
+
+    @property
+    def server(self):
+        '''property for  server url'''
+        return self._azure_server
+
+    @property
+    def token(self):
+        '''property for the access_token
+            curl -d \
+            'client_id=<id>&client_secret=<sec>&grant_type=client_credentials&resource=https://cloudpartner.azure.com' \
+            https://login.microsoftonline.com/72f988bf-86f1-41af-91ab-2d7cd011db47/oauth2/token
+        '''
+        if self._access_token is None:
+            url = 'https://login.microsoftonline.com/{}/oauth2/token'.format(self.client_info['tenant_id'])
+            data = {
+                'client_id': {self.client_info['client_id']},
+                'client_secret': self.client_info['client_secret'],
+                'grant_type': 'client_credentials',
+                'resource': 'https://cloudpartner.azure.com'
+            }
+
+            results = AzurePublisher.request('POST', url, data, {})
+            jres = results.json()
+            self._access_token = jres['access_token']
+
+        return self._access_token
+
+    def get_offers(self, offer=None, version=None, slot=''):
+        ''' fetch all offers by publisherid '''
+        url = '/offers'
+
+        if offer is not None:
+            url += '/{}'.format(offer)
+            if version is not None:
+                url += '/versions/{}'.format(version)
+            if slot != '':
+                url += '/slot/{}'.format(slot)
+
+        url += '?{}'.format(self.api_version)
+
+        return self.prepare_action(url)
+
+    def get_operations(self, offer, operation=None, status=None):
+        ''' create or modify an offer '''
+        url = '/offers/{0}/submissions'.format(offer)
+
+        if operation is not None:
+            url += '/operations/{0}'.format(operation)
+
+        if not url.endswith('/'):
+            url += '/'
+
+        url += '?{0}'.format(self.api_version)
+
+        if status is not None:
+            url += '&status={0}'.format(status)
+
+        return self.prepare_action(url, 'GET')
+
+    def cancel_operation(self, offer):
+        ''' create or modify an offer '''
+        url = '/offers/{0}/cancel?{1}'.format(offer, self.api_version)
+
+        return self.prepare_action(url, 'POST')
+
+    def go_live(self, offer):
+        ''' create or modify an offer '''
+        url = '/offers/{0}/golive?{1}'.format(offer, self.api_version)
+
+        return self.prepare_action(url, 'POST')
+
+    def create_or_modify_offer(self, offer, data=None, modify=False):
+        ''' create or modify an offer '''
+        url = '/offers/{0}?{1}'.format(offer, self.api_version)
+
+        headers = None
+
+        if modify:
+            headers = {
+                'If-Match': '*',
+            }
+
+        return self.prepare_action(url, 'PUT', data=data, add_headers=headers)
+
+    def prepare_action(self, url, action='GET', data=None, add_headers=None):
+        '''perform the http request
+
+           :action: string of either GET|POST
+        '''
+        headers = {
+            'Content-Type': 'application/json',
+            'Accept': 'application/json',
+            'Authorization': 'Bearer {}'.format(self.token)
+        }
+
+        if add_headers is not None:
+            headers.update(add_headers)
+
+        if data is None:
+            data = ''
+        else:
+            data = json.dumps(data)
+
+        return AzurePublisher.request(action.upper(), self.server + url, data, headers)
+
+    def manage_offer(self, params):
+        ''' handle creating or modifying offers'''
+        # fetch the offer to verify it exists:
+        results = self.get_offers(offer=params['offer'])
+
+        if results.status_code == 200 and params['force']:
+            return self.create_or_modify_offer(offer=params['offer'], data=params['offer_data'], modify=True)
+
+        return self.create_or_modify_offer(offer=params['offer'], data=params['offer_data'])
+
+    @staticmethod
+    def request(action, url, data=None, headers=None, ssl_verify=True):
+        req = requests.Request(action.upper(), url, data=data, headers=headers)
+
+        session = requests.Session()
+        req_prep = session.prepare_request(req)
+        response = session.send(req_prep, verify=ssl_verify)
+
+        return response
+
+    @staticmethod
+    def run_ansible(params):
+        '''perform the ansible operations'''
+        client_info = {
+            'tenant_id': params['tenant_id'],
+            'client_id': params['client_id'],
+            'client_secret': params['client_secret']}
+
+        apc = AzurePublisher(params['publisher'],
+                             client_info,
+                             debug=params['debug'])
+
+        if params['query'] == 'offer':
+            results = apc.get_offers(offer=params['offer'])
+        elif params['query'] == 'operation':
+            results = apc.get_operations(offer=params['offer'], operation=params['operation'], status=params['status'])
+        else:
+            raise AzurePublisherException('Unsupported query type: {}'.format(params['query']))
+
+        return {'data': results.json(), 'status_code': results.status_code}
+
+
+def main():
+    ''' ansible oc module for secrets '''
+
+    module = AnsibleModule(
+        argument_spec=dict(
+            query=dict(default='offer', choices=['offer', 'operation']),
+            publisher=dict(default='redhat', type='str'),
+            debug=dict(default=False, type='bool'),
+            tenant_id=dict(default=os.environ.get('AZURE_TENANT_ID'), type='str'),
+            client_id=dict(default=os.environ.get('AZURE_CLIENT_ID'), type='str'),
+            client_secret=dict(default=os.environ.get('AZURE_CLIENT_SECRET'), type='str'),
+            offer=dict(default=None, type='str'),
+            operation=dict(default=None, type='str'),
+            status=dict(default=None, type='str'),
+        ),
+    )
+
+    # Verify we recieved either a valid key or edits with valid keys when receiving a src file.
+    # A valid key being not None or not ''.
+    if (module.params['tenant_id'] is None or module.params['client_id'] is None or
+            module.params['client_secret'] is None):
+        return module.fail_json(**{'failed': True,
+                                   'msg': 'Please specify tenant_id, client_id, and client_secret'})
+
+    rval = AzurePublisher.run_ansible(module.params)
+    if int(rval['status_code']) == 404:
+        rval['msg'] = 'Offer does not exist.'
+    elif int(rval['status_code']) >= 300:
+        rval['msg'] = 'Error.'
+        return module.fail_json(**rval)
+
+    return module.exit_json(**rval)
+
+
+if __name__ == '__main__':
+    main()