#!/usr/bin/env python
"""
 SlipStream Client
 =====
 Copyright (C) 2014 SixSq Sarl (sixsq.com)
 =====
 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

import os
import os.path
import re
import sys
import xml.etree.ElementTree as ET

from slipstream.command.CommandBase import CommandBase
from slipstream.HttpClient import HttpClient
import slipstream.util as util

class MainProgram(CommandBase):
    '''Recursively download SlipStream modules as XML from server.'''

    def __init__(self, argv=None):
        self.module = ''
        self.username = None
        self.password = None
        self.cookie = None
        self.endpoint = None
        super(MainProgram, self).__init__(argv)

    def parse(self):
        usage = '''usage: %prog [options] <module-url>

<module-uri>    Name of the root module.'''

        self.parser.usage = usage

        self.add_authentication_options()
        self.addEndpointOption()

        self.parser.add_option('--remove-cloud-specific', dest='remove_clouds',
                               help='Remove all cloud specific elements (image ids, cloud parameters, ...)',
                               default=False, action='store_true')

        self.parser.add_option('--dump-image-ids', dest='dump_image_ids',
            help='Store image IDs found in image modules into per module files.',
            default=False, action='store_true')

        self.parser.add_option('--dump-image-ids-dir', dest='dump_image_ids_dir',
            help='Path to the directory to store the image IDs files. Default: current directory.',
            default='.')

        self.parser.add_option('--remove-group-members', dest='remove_group_members',
                               help='Remove members of the group in the authorizations',
                               default=False, action='store_true')

        self.parser.add_option('--reset-commit-message', dest='reset_commit_message',
                               help='Replace the commit message by "Initial version of this module"',
                               default=False, action='store_true')

        self.parser.add_option('--flat', dest='flat_export',
                               help='Download without creating subdirectories',
                               default=False, action='store_true')

        self.options, self.args = self.parser.parse_args()

        self._checkArgs()

    def _checkArgs(self):
        if len(self.args) == 1:
            self.module = self.args[0]
        if len(self.args) > 1:
            self.usageExitTooManyArguments()

    def _removeRuns(self, root):
        """Remove the run elements from the given document.
           The runs are dynamic and cannot be pushed back
           into the server.  This modified the given XML
           document!"""
        runs = root.find('runs')
        if runs is not None:
            for run in runs.findall('*'):
                runs.remove(run)

    def _remove_clouds(self, root):
        """Remove the cloudImageIdentifiers, cloudNames and cloud specific parameters element from the given document.
           These elements are not portable between SlipStream deployments."""

        ids = root.find('cloudImageIdentifiers')
        if ids is not None:
            for id in ids.findall('*'):
                ids.remove(id)

        cloud_names = root.find('cloudNames')
        if cloud_names is not None:
            parameters = root.find('parameters')
            if parameters is not None:
                for cloud_name in cloud_names.findall('*'):
                    # The following XPath query doesn't work with Python < 2.7
                    cloud_parameters = parameters.findall("./entry/parameter[@category='%s'].." % cloud_name.text)
                    if cloud_parameters is not None:
                        for cloud_parameter in cloud_parameters:
                            parameters.remove(cloud_parameter)
                        cloud_names.remove(cloud_name)

    def _remove_group_members(self, root):
        authz = root.find('authz')
        if authz is None:
            return

        group_members = authz.find('groupMembers')
        if group_members is None:
            return

        for group_member in group_members.findall('*'):
            group_members.remove(group_member)

    def _reset_commit_message(self, root):
        authz = root.find('authz')

        commit = root.find('commit')
        if commit is None:
            return

        commit.attrib['author'] = (authz is not None) and authz.attrib.get('owner', 'super') or 'super'

        comment = commit.find('comment')
        if comment is None:
            return
        comment.text = 'Initial version of this module'

    def _retrieveModuleAsXml(self, client, module):
        uri = util.MODULE_RESOURCE_PATH
        uri += '/' + module

        url = self.options.endpoint + uri

        _, xml = client.get(url)

        return ET.fromstring(xml)

    def _writeModuleAsXml(self, root_element, module):
        if self.options.flat_export:
            module = module.replace('/', '_')
        else:
            if root_element.attrib.get('category', '').lower().strip() == 'project':
                module = os.path.join(module, os.path.basename(module))
            try:
                os.makedirs(os.path.dirname(module), 0775)
            except OSError as e:
                pass
        ET.ElementTree(root_element).write('%s.xml' % module)

    def _getModuleChildren(self, module, root_element):
        children = []
        for child in root_element.findall('children/item'):
            module_name = child.attrib['name']
            module_path = '%s/%s' % (module, module_name)
            children.append(module_path)
        return children

    @staticmethod
    def _is_image(root):
        return root.tag == 'imageModule'

    def _dump_image_ids(self, root):
        ids = root.find('cloudImageIdentifiers')
        if ids is not None:
            module_name = root.get('shortName')
            module_path = re.sub('^module/', '', root.get('parentUri'))
            module_uri = "%s/%s" % (module_path, module_name)
            cloud_ids = ''
            for _id in ids.findall('*'):
                cloud_ids += "%s = %s:%s\n" % (module_uri, _id.get('cloudServiceName'),
                        _id.get('cloudImageIdentifier'))
            if cloud_ids:
                ids_dir = self.options.dump_image_ids_dir
                if not os.path.exists(ids_dir):
                    os.makedirs(ids_dir)
                fn = '%s_%s' % (module_path.replace('/','_'), module_name)
                ids_file = "%s/%s.conf" % (ids_dir, fn)
                with open(ids_file, 'w') as fh:
                    fh.write(cloud_ids)

    def doWork(self):

        if self.options.remove_clouds and sys.version_info[0:3] < (2, 7, 0):
            print('Error: The use of "--remove-cloud-specific" require Python >= 2.7', file=sys.stderr)
            sys.exit(1)

        client = HttpClient(self.options.username, self.options.password)
        client.verboseLevel = self.verboseLevel

        queue = [self.module]
        while len(queue) > 0:
            module = queue.pop(0)
            print('Processing: %s' % module)

            root = self._retrieveModuleAsXml(client, module)
            self._removeRuns(root)
            if self._is_image(root) and self.options.dump_image_ids:
                self._dump_image_ids(root)
            if self.options.remove_clouds:
                self._remove_clouds(root)
            if self.options.remove_group_members:
                self._remove_group_members(root)
            if self.options.reset_commit_message:
                self._reset_commit_message(root)
            self._writeModuleAsXml(root, module)

            for child in self._getModuleChildren(module, root):
                queue.append(child)


if __name__ == "__main__":
    try:
        MainProgram()
    except KeyboardInterrupt:
        print('\n\nExecution interrupted by the user... goodbye!')
        sys.exit(-1)
