#!/usr/bin/env python
from __future__ import print_function, unicode_literals

import argparse
import io
import logging
import os
import re
import sys


_DESCRIPTION = """\
Migrate arbitrary `.go` sources (mostly) from the v1 to v2 API.
"""
_MIGRATORS = []


def main(sysargs=sys.argv[:]):
    parser = argparse.ArgumentParser(
        description=_DESCRIPTION,
        formatter_class=argparse.ArgumentDefaultsHelpFormatter)
    parser.add_argument('path', nargs='*',
                        type=os.path.abspath, default=os.getcwd())
    parser.add_argument('-w', '--write', help='write changes back to file',
                        action='store_true', default=False)
    parser.add_argument('-q', '--quiet', help='quiet down the logging',
                        action='store_true', default=False)
    parser.add_argument('-D', '--debug', help='debug up the logging',
                        action='store_true',
                        default=(os.environ.get('DEBUG') != ''))
    parser.add_argument('--selftest', help='run internal tests',
                        action='store_true', default=False)

    args = parser.parse_args(sysargs[1:])

    if args.selftest:
        logging.basicConfig(
            level=logging.WARN,
            format='selftest: %(message)s'
        )
        test_migrators()
        return 0

    level = logging.FATAL if args.quiet else logging.INFO
    level = logging.DEBUG if args.debug else level

    logging.basicConfig(level=level, format='%(message)s')

    paths = args.path
    if len(paths) == 0:
        paths = ['.']

    for filepath in _find_candidate_files(paths):
        updated_source = _update_filepath(filepath)
        if args.write:
            logging.info('Updating %s', filepath)

            with io.open(filepath, 'w', encoding='utf-8') as outfile:
                outfile.write(updated_source)
        else:
            logging.info('// Updated %s:', filepath)
            print(updated_source)

    return 0


def _find_candidate_files(paths):
    for path in paths:
        if not os.path.isdir(path):
            yield path
            continue

        for curdir, dirs, files in os.walk(path):
            for i, dirname in enumerate(dirs[:]):
                if dirname.startswith('.'):
                    dirs.pop(i)

            for filename in files:
                if not filename.decode('utf-8').endswith('.go'):
                    continue

                filepath = os.path.join(curdir, filename)
                if not os.access(filepath, os.R_OK | os.W_OK):
                    continue

                yield filepath


def _update_filepath(filepath):
    with io.open(filepath, encoding='utf-8') as infile:
        return _update_source(infile.read())


def _update_source(source):
    for migrator, func in _MIGRATORS:
        logging.debug('Running %s migrator', migrator)
        source = func(source)
    return source


def _subfmt(pattern, replfmt, source, flags=re.UNICODE):
    def repl(match):
        return replfmt.format(**match.groupdict())
    return re.sub(pattern, repl, source, flags=flags)


def _migrator(func):
    _MIGRATORS.append((func.__name__.strip('_'), func))
    return func


@_migrator
def _slice_pointer_types(source):
    return _subfmt(
        '(?P<prefix>\\[\\])cli\\.(?P<type>Command|Author){',
        '{prefix}*cli.{type}{{', source
    )


@_migrator
def _pointer_type_literal(source):
    return _subfmt(
        '(?P<prefix>\\s+)cli\\.(?P<type>Command|Author){',
        '{prefix}&cli.{type}{{', source
    )


@_migrator
def _slice_types(source):
    return _subfmt(
        '&cli\\.(?P<type>IntSlice|StringSlice){(?P<args>[^}]*)}',
        'cli.New{type}({args})', source, flags=re.DOTALL | re.UNICODE
    )


@_migrator
def _flag_literals(source):
    return _subfmt(
        '(?P<prefix>\\s+)cli\\.(?P<type>\\w+)Flag{',
        '{prefix}&cli.{type}Flag{{', source
    )


@_migrator
def _v1_imports(source):
    return re.sub(
        '"(?:github\\.com|gopkg\\.in)/(?:codegangsta|urfave)/cli(?:\\.v1|)"',
        '"gopkg.in/urfave/cli.v2"', source, flags=re.UNICODE
    )


@_migrator
def _new_exit_error(source):
    return re.sub('cli\\.NewExitError', 'cli.Exit', source, flags=re.UNICODE)


@_migrator
def _bool_t_flag(source):
    return _subfmt(
        'cli\\.BoolTFlag{(?P<args>[^}]*)}',
        'cli.BoolFlag{{Value: true,{args}}}',
        source, flags=re.DOTALL | re.UNICODE
    )


@_migrator
def _context_args_len(source):
    return _subfmt(
        'len\\((?P<prefix>\\S+)\\.Args\\(\\)\\)',
        '{prefix}.Args().Len()', source
    )


@_migrator
def _context_args_index(source):
    return _subfmt(
        '\\.Args\\(\\)\\[(?P<index>\\d+)\\]',
        '.Args().Get({index})', source
    )


@_migrator
def _envvar_string(source):
    return re.sub(
        'EnvVar:(?P<ws>\\s+)"(?P<string>[^"]+)"',
        _envvar_string_repl, source, flags=re.UNICODE
    )


def _envvar_string_repl(match):
    return 'EnvVars:{ws}[]string{{{value}}}'.format(
        value=', '.join([
            '"{}"'.format(s) for s in
            re.split(
                '\\s*,\\s*', match.groupdict()['string'],
                flags=re.UNICODE
            )
        ]),
        **match.groupdict()
    )


@_migrator
def _flag_name_stringly(source):
    return re.sub(
        '(?P<prefix>\\s+)Name:(?P<ws>\\s+)"(?P<string>[^"]+)"',
        _flag_name_stringly_repl, source, flags=re.UNICODE
    )


def _flag_name_stringly_repl(match):
    revars = dict(match.groupdict())

    string = revars['string']
    parts = list(
        reversed(
            sorted(
                filter(lambda s: len(s.strip()) > 0, [
                    part.strip() for part in string.split(',')
                ]), key=len
            )
        )
    )

    if len(parts) == 1:
        return '{prefix}Name:{ws}"{string}"'.format(**revars)

    return (
        '{prefix}Name:{ws}"{name}", Aliases: []string{{{aliases}}}'
    ).format(
        name=parts[0],
        aliases=', '.join(['"{}"'.format(s) for s in parts[1:]]),
        **revars
    )


@_migrator
def _commands_opaque_type(source):
    return re.sub('cli\\.Commands', '[]*cli.Command', source, flags=re.UNICODE)


@_migrator
def _flag_names(source):
    return re.sub('\\.GetName\\(\\)', '.Names()[0]', source, flags=re.UNICODE)


@_migrator
def _app_categories(source):
    source = _subfmt(
        '(?P<prefix>range\\s+\\S+)\\.App\\.Categories\\(\\)',
        '{prefix}.App.Categories.Categories()', source
    )

    return re.sub(
        '\\.App\\.Categories\\(\\)', '.App.Categories',
        source, flags=re.UNICODE
    )


@_migrator
def _command_category_commands(source):
    # XXX: brittle
    return _subfmt(
        '(?P<prefix>\\s+category\\.)Commands(?P<suffix>[^(])',
        '{prefix}VisibleCommands(){suffix}', source
    )


@_migrator
def _context_bool_t(source):
    # XXX: probably brittle
    return _subfmt(
        '(?P<prefix>\\S+)(?:Global|)BoolT\\(',
        '!{prefix}Bool(', source
    )


@_migrator
def _context_global_methods(source):
    return _subfmt(
        '\\.Global(?P<method>'
        'Bool|Duration|Float64|Generic|Int|IntSlice|String|StringSlice|'
        'FlagNames|IsSet|Set'
        ')\\(',
        '.{method}(', source
    )


@_migrator
def _context_parent(source):
    # XXX: brittle
    return re.sub('\\.Parent\\(\\)', '.Lineage()[1]', source, flags=re.UNICODE)


@_migrator
def _app_init(source):
    return re.sub(
        'cli\\.NewApp\\(\\)', '(&cli.App{})', source, flags=re.UNICODE
    )


def test_migrators():
    import difflib

    for i, (source, expected) in enumerate(_MIGRATOR_TESTS):
        actual = _update_source(source)
        if expected != actual:
            udiff = difflib.unified_diff(
                expected.splitlines(), actual.splitlines(),
                fromfile='a/source.go', tofile='b/source.go', lineterm=''
            )
            for line in udiff:
                print(line)
            raise AssertionError('migrated source does not match expected')
        logging.warn('Test case %d/%d OK', i+1, len(_MIGRATOR_TESTS))


_MIGRATOR_TESTS = (
    ("""
\t\t\t&cli.StringSlice{"a", "b", "c"},
""", """
\t\t\tcli.NewStringSlice("a", "b", "c"),
"""),
    ("""
\t\tcli.IntFlag{
\t\t\tName:  "yep",
\t\t\tValue: 3,
\t\t}
""", """
\t\t&cli.IntFlag{
\t\t\tName:  "yep",
\t\t\tValue: 3,
\t\t}
"""),
    ("""
\t\tapp.Commands = []cli.Command{
\t\t\t{
\t\t\t\tName: "whatebbs",
\t\t\t},
\t\t}
""", """
\t\tapp.Commands = []*cli.Command{
\t\t\t{
\t\t\t\tName: "whatebbs",
\t\t\t},
\t\t}
"""),
    ("""
\t\tapp.Commands = []cli.Command{
\t\t\tcli.Command{
\t\t\t\tName: "whatebbs",
\t\t\t},
\t\t}
""", """
\t\tapp.Commands = []*cli.Command{
\t\t\t&cli.Command{
\t\t\t\tName: "whatebbs",
\t\t\t},
\t\t}
"""),
    ("""
\t"github.com/codegangsta/cli"
\t"github.com/urfave/cli"
\t"gopkg.in/codegangsta/cli"
\t"gopkg.in/codegangsta/cli.v1"
\t"gopkg.in/urfave/cli"
\t"gopkg.in/urfave/cli.v1"
""", """
\t"gopkg.in/urfave/cli.v2"
\t"gopkg.in/urfave/cli.v2"
\t"gopkg.in/urfave/cli.v2"
\t"gopkg.in/urfave/cli.v2"
\t"gopkg.in/urfave/cli.v2"
\t"gopkg.in/urfave/cli.v2"
"""),
    ("""
\t\t\t\treturn cli.NewExitError("foo whatebber", 9)
""", """
\t\t\t\treturn cli.Exit("foo whatebber", 9)
"""),
    ("""
\t\t\tapp.Flags = []cli.Flag{
\t\t\t\tcli.StringFlag{
\t\t\t\t\tName: "aha",
\t\t\t\t},
\t\t\t\tcli.BoolTFlag{
\t\t\t\t\tName: "blurp",
\t\t\t\t},
\t\t\t}
""", """
\t\t\tapp.Flags = []cli.Flag{
\t\t\t\t&cli.StringFlag{
\t\t\t\t\tName: "aha",
\t\t\t\t},
\t\t\t\t&cli.BoolFlag{Value: true,
\t\t\t\t\tName: "blurp",
\t\t\t\t},
\t\t\t}
"""),
    ("""
\t\t\tAction = func(c *cli.Context) error {
\t\t\t\tif c.Args()[4] == "meep" {
\t\t\t\t\treturn nil
\t\t\t\t}
\t\t\t\treturn errors.New("mope")
\t\t\t}
""", """
\t\t\tAction = func(c *cli.Context) error {
\t\t\t\tif c.Args().Get(4) == "meep" {
\t\t\t\t\treturn nil
\t\t\t\t}
\t\t\t\treturn errors.New("mope")
\t\t\t}
"""),
    ("""
\t\tapp.Flags = []cli.Flag{
\t\t\tcli.StringFlag{
\t\t\t\tName:   "toots",
\t\t\t\tEnvVar: "TOOTS,TOOTERS",
\t\t\t},
\t\t}
""", """
\t\tapp.Flags = []cli.Flag{
\t\t\t&cli.StringFlag{
\t\t\t\tName:   "toots",
\t\t\t\tEnvVars: []string{"TOOTS", "TOOTERS"},
\t\t\t},
\t\t}
"""),
    ("""
\t\tapp.Flags = []cli.Flag{
\t\t\tcli.StringFlag{
\t\t\t\tName:   "t, tootles, toots",
\t\t\t},
\t\t}
""", """
\t\tapp.Flags = []cli.Flag{
\t\t\t&cli.StringFlag{
\t\t\t\tName:   "tootles", Aliases: []string{"toots", "t"},
\t\t\t},
\t\t}
"""),
    ("""
\t\tapp := cli.NewApp()
\t\tapp.HideHelp = true
""", """
\t\tapp := (&cli.App{})
\t\tapp.HideHelp = true
""")
)


if __name__ == '__main__':
    sys.exit(main())