#!/usr/bin/env python
#coding=utf-8

import os, re, datetime, sys, subprocess

DEFAULT_LICENSE = "gpl3"
CONTRIBUTOR_LICENSE = "mit"
LICENSE_DIR = "Documentation/Licenses"

# The following regex parses license comment blocks and its part out of a complete source file.
reParseLicenseCommentBlocks = re.compile(ur'(\/\*\n\s\*\sCopyright \(c\) (?P<startYear>\d\d\d\d)(-(?P<endYear>\d\d\d\d))? (?P<author>[^\n\.]*)\.?\n.\* (?P<license>[^\n]*)\n \* (?P<seeMore>[^\n]+)\n *\*\/)')

class License :
    def __init__(self, name, file) :
        self.name = name
        self.file = file

licenses = {
        "default": License("All rights reserved.", "See the COPYING file for more information."),
        "gpl3" : License("Licensed under the GNU General Public License v3.", "See " + LICENSE_DIR + "/" + "GPLv3.txt" + " for more information."),
        "mit" : License("Licensed under the MIT License.", "See " + LICENSE_DIR + "/" +  "MIT.txt" + " for more information."),
    }

class Copyright :
    def __init__(self, author, year, license) :
        self.author = author
        self.year = year
        self.license = license

    def to_string(self, comment_chars) :
        return "\n".join([
            comment_chars[0],
            comment_chars[1] + " Copyright (c) %(year)s %(name)s" % {"year" : self.year, "name" : self.author },
            comment_chars[1] + licenses[self.license].name,
            comment_chars[1] + licenses[self.license].file,
            comment_chars[2],
            "\n"])
    def __str__(self):
        return """/*
 * Copyright (c) %s %s.
 * %s
 * %s
 */
""" % (self.year, self.author, licenses[self.license].name, licenses[self.license].file)

class ContentRef :
    def __init__(self, begin, end, content):
        self.begin = begin
        self.end = end
        self.content = content

class CopyrightBlock :
    def __init__(self, yearBegin, yearEnd, author, license, seeMore, total):
        self.yearBegin = yearBegin
        self.yearEnd = yearEnd
        self.author = author
        self.license = license
        self.seeMore = seeMore
        self.total = total

def cref_from_group(match, group):
    if match.group(group):
        return ContentRef(match.start(group), match.end(group), match.group(group))
    else :
        return None

def parse_file_new(filename):
    copyrightBlocks = []
    with open(filename, 'r') as file:
        content = file.read()
        for match in re.finditer(reParseLicenseCommentBlocks, content):
            copyrightBlocks.append(CopyrightBlock(
                cref_from_group(match, "startYear"), 
                cref_from_group(match, "endYear"), 
                cref_from_group(match, "author"), 
                cref_from_group(match, "license"), 
                cref_from_group(match, "seeMore"), 
                cref_from_group(match, 0)))
    return copyrightBlocks

def get_userinfo() :
    p = subprocess.Popen("git config user.name", shell=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, close_fds=(os.name != "nt"))
    username = p.stdout.read().rstrip()
    p.stdin.close()
    if p.wait() != 0 :
        return None
    p = subprocess.Popen("git config user.email", shell=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, close_fds=(os.name != "nt"))
    email = p.stdout.read().rstrip()
    p.stdin.close()
    if p.wait() != 0 :
        return None
    return (username, email)

def get_copyright(username, email) :
    if email in ["git@el-tramo.be", "git@kismith.co.uk"] :
        license = DEFAULT_LICENSE
    else :
        license = CONTRIBUTOR_LICENSE
    return Copyright(username, datetime.date.today().strftime("%Y"), license)

def get_copyright_setting(username, email) :
    config = os.getenv("SWIFT_LICENSE_CONFIG")
    if config :
        copyrightHolder, license = config.split("|")
    else :
        if email.endswith("isode.com") or email in ["git@el-tramo.be", "git@kismith.co.uk", "tm@ayena.de"] :
            copyrightHolder, license = "Isode Limited", "default"
        else :
            copyrightHolder, license = username, "mit"
    return Copyright(copyrightHolder, datetime.date.today().year, license)

def check_copyright(filename, hints) :
    copyrightBlocks = parse_file_new(filename)
    if copyrightBlocks :
        # looking for copyright block for current author
        username, email = get_userinfo()
        copyrightSetting = get_copyright_setting(username, email)
        for block in copyrightBlocks :
            if block.author.content == copyrightSetting.author:
                year = block.yearBegin.content if not block.yearEnd else block.yearEnd.content
                if int(year) == copyrightSetting.year:
                    return True
                else :
                    if hints :
                        print "Copyright block for " + copyrightSetting.author + " does not cover current year in: " + filename
                    return False
        if hints :
            print "Missing copyright block for " + copyrightSetting.author + " in: " + filename
        return False
    else :
        if hints :
            print "No copyright found in: " + filename
        return False

def replace_data_in_file(filename, begin, end, replaceWith) :
    with open(filename, 'r') as file:
        content = file.read()
    with open(filename, 'w') as file:
        file.write(content[:begin] + replaceWith + content[end:])

def set_or_update_copyright(filename) :
    if check_copyright(filename, False) :
        print "No update required for file: " + filename
    else :
        copyrightBlocks = parse_file_new(filename)
        username, email = get_userinfo()
        copyrightSetting = get_copyright_setting(username, email)
        lastBlock = 0
        for block in copyrightBlocks :
            if block.author.content == copyrightSetting.author :
                if not block.yearEnd :
                    # replace year with range
                    replace_data_in_file(filename, block.yearBegin.begin, block.yearBegin.end, "%s-%s" % (block.yearBegin.content, str(copyrightSetting.year)))
                else :
                    # replace end of range with current year
                    replace_data_in_file(filename, block.yearEnd.begin, block.yearEnd.end, "%s" % str(copyrightSetting.year))
                return
            lastBlock = block.total.end

        # No copyright block found. Append a new one.
        replace_data_in_file(filename, lastBlock+1, lastBlock+1, "\n" + str(copyrightSetting))

def print_help() :
    print """Usage:
    Copyrighter.py check-copyright $filename
        Cheks for the existence of a copyright comment block.

    Copyrighter.py set-copyright $filename
        Adds or updates the existing copyright comment block.

    License setting:
        A users license configuration can be set via the SWIFT_LICENSE_CONFIG environment variable 
        in the format "$copyright holder|$license", e.g. "Jane Doe|mit". Possible values for 
        $license are default, mit and gpl.
    """

if sys.argv[1] == "check-copyright" :
    file = sys.argv[2]
    if (file.endswith(".cpp") or file.endswith(".h")) and not "3rdParty" in file :
        if not check_copyright(file, True) :
            sys.exit(-1)
elif sys.argv[1] == "set-copyright" :
    file = sys.argv[2]
    set_or_update_copyright(file)
else :
    print "Unknown command: " + sys.argv[1]
    print_help()
    sys.exit(-1)