#!/usr/bin/env python
# -*- coding: utf-8 -*-
#
# Copyright (C) 2014-2019 Bitergia
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
# Authors:
#     Santiago Dueñas <sduenas@bitergia.com>
#

import argparse
import datetime
import json
import sys

from sortinghat.exceptions import InvalidFormatError
from sortinghat.parsing.gitdm import GitdmParser


GITDM2SH_DESC_MSG = \
"""Export identities information from Gitdm files to Sorting Hat JSON format."""


def main():
    """Export identities information from Gitdm files"""

    args = parse_args()

    try:
        email_validation = not args.no_email_validation
        parser = parse_gitdm_files(args.aliases, args.email_employer,
                                   args.domain_employer, args.source, email_validation)
    except (IOError, UnicodeDecodeError, InvalidFormatError) as e:
        raise RuntimeError(str(e))

    json = to_json(parser.identities, parser.organizations,
                   args.source)

    try:
        args.outfile.write(json)
        args.outfile.write('\n')
    except IOError as e:
        raise RuntimeError(str(e))


def parse_args():
    """Parse arguments from the command line"""

    parser = argparse.ArgumentParser(description=GITDM2SH_DESC_MSG)

    parser.add_argument('-a', '--aliases', dest='aliases',
                        type=argparse.FileType('r'),
                        help='Gitdm aliases mapping file')
    parser.add_argument('-e', '--email-employer', dest='email_employer',
                        type=argparse.FileType('r'),
                        help='Gitdm email to employer mapping file')
    parser.add_argument('-d', '--domain-employer', dest='domain_employer',
                        type=argparse.FileType('r'),
                        help='Gitdm domain to employer mapping file')
    parser.add_argument('-s', '--source', dest='source', required=True,
                        help='name of the source')
    parser.add_argument('-o', '--outfile', nargs='?', type=argparse.FileType('w'),
                        default=sys.stdout,
                        help='Sorting Hat JSON output filename')
    parser.add_argument('--no-email-validation', dest='no_email_validation',
                        action='store_true',
                        help="do not email addresses validation")

    return parser.parse_args()


def parse_gitdm_files(aliases, email_to_employer,
                      domain_to_employer, source, email_validation):
    """Parse Gitdm files"""

    a = None
    ee = None
    de = None

    if aliases:
        a = read_file(aliases)
    if email_to_employer:
        ee = read_file(email_to_employer)
    if domain_to_employer:
        de = read_file(domain_to_employer)

    parser = GitdmParser(aliases=a,
                         email_to_employer=ee,
                         domain_to_employer=de,
                         source=source,
                         email_validation=email_validation)
    return parser


def to_json(uidentities, organizations, source):
    """Convert unique identities to Sorting Hat JSON format"""

    uids = {}
    orgs = {}

    # Convert to dict objects
    for uidentity in uidentities:
        uuid = uidentity.uuid

        uid = uidentity.to_dict()
        uid['identities'].sort(key=lambda x: x['email'] or x['username'])

        enrollments = [rol.to_dict() \
                       for rol in uidentity.enrollments]
        uid['enrollments'] = enrollments

        uids[uuid] = uid

    for organization in organizations:
        domains = [{'domain': dom.domain,
                    'is_top': dom.is_top_domain} \
                   for dom in organization.domains]
        domains.sort(key=lambda x: x['domain'])

        orgs[organization.name] = domains

    # Generate JSON file
    obj = {'time' : str(datetime.datetime.now()),
           'source' : source,
           'blacklist' : [],
           'organizations' : orgs,
           'uidentities' : uids}

    return json.dumps(obj, default=json_encoder,
                      indent=4, sort_keys=True)


def json_encoder(obj):
    """Default JSON encoder"""

    if isinstance(obj, datetime.datetime):
        return obj.isoformat()
    else:
        return json.JSONEncoder.default(obj)


def read_file(f):
    content = f.read()
    return content


if __name__ == '__main__':
    try:
        main()
    except KeyboardInterrupt:
        s = "\n\nReceived Ctrl-C or other break signal. Exiting.\n"
        sys.stdout.write(s)
        sys.exit(0)
    except RuntimeError as e:
        s = "Error: %s\n" % str(e)
        sys.stderr.write(s)
        sys.exit(1)
