#!/usr/bin/env python

import os

from waflib import Logs, Options, Utils
from waflib.extras import autowaf

# Package version
INGEN_VERSION       = '0.5.1'
INGEN_MAJOR_VERSION = '0'

# Mandatory waf variables
APPNAME = 'ingen'        # Package name for waf dist
VERSION = INGEN_VERSION  # Package version for waf dist
top     = '.'            # Source directory
out     = 'build'        # Build directory

line_just = 47


def options(ctx):
    ctx.load('compiler_cxx')
    ctx.load('python')
    ctx.load('lv2')
    ctx.recurse('src/gui')
    opt = ctx.configuration_options()

    opt.add_option('--data-dir', type='string', dest='datadir',
                   help='ingen data install directory [default: PREFIX/share/ingen]')
    opt.add_option('--module-dir', type='string', dest='moduledir',
                   help='ingen module install directory [default: PREFIX/lib/ingen]')

    ctx.add_flags(
        opt,
        {'no-gui': 'do not build GUI',
         'no-client': 'do not build client library (or GUI)',
         'no-jack': 'do not build jack backend (for ingen.lv2 only)',
         'no-plugin': 'do not build ingen.lv2 plugin',
         'no-python': 'do not install Python bindings',
         'no-webkit': 'do not use webkit to display plugin documentation',
         'no-jack-session': 'do not build JACK session support',
         'no-socket': 'do not build Socket interface',
         'debug-urids': 'print a trace of URI mapping',
         'portaudio': 'build PortAudio backend'})


def configure(conf):
    conf.load('compiler_cxx', cache=True)
    conf.load('lv2', cache=True)
    if not Options.options.no_python:
        conf.load('python', cache=True)

    conf.load('autowaf', cache=True)
    autowaf.set_cxx_lang(conf, 'c++11')

    conf.check_cxx(header_name='boost/intrusive/slist.hpp')
    conf.check_cxx(msg='Checking for thread_local keyword',
                   mandatory=False,
                   fragment='thread_local int i = 0; int main() {}',
                   define_name='INGEN_HAVE_THREAD_LOCAL')
    if not conf.is_defined('INGEN_HAVE_THREAD_LOCAL'):
        conf.check_cxx(msg='Checking for __thread keyword',
                       mandatory=False,
                       fragment='__thread int i = 0; int main() {}',
                       define_name='INGEN_HAVE_THREAD_BUILTIN')

    conf.check_pkg('lv2 >= 1.16.0', uselib_store='LV2')
    conf.check_pkg('lilv-0 >= 0.21.5', uselib_store='LILV')
    conf.check_pkg('suil-0 >= 0.8.7', uselib_store='SUIL')
    conf.check_pkg('sratom-0 >= 0.4.6', uselib_store='SRATOM')
    conf.check_pkg('raul-1 >= 1.0.0', uselib_store='RAUL')
    conf.check_pkg('serd-0 >= 0.30.0', uselib_store='SERD', mandatory=False)
    conf.check_pkg('sord-0 >= 0.12.0', uselib_store='SORD', mandatory=False)
    conf.check_pkg('portaudio-2.0', uselib_store='PORTAUDIO', mandatory=False)
    conf.check_pkg('sigc++-2.0', uselib_store='SIGCPP', mandatory=False)

    conf.check_function('cxx', 'posix_memalign',
                        defines     = '_POSIX_C_SOURCE=200809L',
                        header_name = 'stdlib.h',
                        define_name = 'HAVE_POSIX_MEMALIGN',
                        mandatory   = False)

    conf.check_function('cxx', 'isatty',
                        header_name = 'unistd.h',
                        defines     = '_POSIX_C_SOURCE=200809L',
                        define_name = 'HAVE_ISATTY',
                        mandatory   = False)

    conf.check_function('cxx', 'vasprintf',
                        header_name = 'stdio.h',
                        defines     = '_GNU_SOURCE=1',
                        define_name = 'HAVE_VASPRINTF',
                        mandatory   = False)

    conf.check(define_name = 'HAVE_LIBDL',
               lib         = 'dl',
               mandatory   = False)

    if not Options.options.no_socket:
        conf.check_function('cxx', 'socket',
                            header_name   = 'sys/socket.h',
                            define_name   = 'HAVE_SOCKET',
                            mandatory     = False)

    if not Options.options.no_python:
        conf.check_python_version((2, 4, 0), mandatory=False)

    if not Options.options.no_plugin:
        conf.env.INGEN_BUILD_LV2 = 1

    if not Options.options.no_jack:
        conf.check_pkg('jack >= 0.120.0', uselib_store='JACK', mandatory=False)
        conf.check_function('cxx', 'jack_set_property',
                            header_name   = 'jack/metadata.h',
                            define_name   = 'HAVE_JACK_METADATA',
                            uselib        = 'JACK',
                            mandatory     = False)
        conf.check_function('cxx', 'jack_port_rename',
                            header_name   = 'jack/jack.h',
                            define_name   = 'HAVE_JACK_PORT_RENAME',
                            uselib        = 'JACK',
                            mandatory     = False)
        if not Options.options.no_jack_session:
            conf.define('INGEN_JACK_SESSION', 1)

    if Options.options.debug_urids:
        conf.define('INGEN_DEBUG_URIDS', 1)

    conf.env.INGEN_TEST_LINKFLAGS = []
    conf.env.INGEN_TEST_CXXFLAGS  = []
    if conf.env.BUILD_TESTS:
        if not conf.env.NO_COVERAGE:
            conf.env.INGEN_TEST_CXXFLAGS  += ['--coverage']
            conf.env.INGEN_TEST_LINKFLAGS += ['--coverage']

    conf.env.PTHREAD_CFLAGS = []
    conf.env.PTHREAD_LINKFLAGS = []
    if conf.check(cflags=['-pthread'], mandatory=False):
        conf.env.PTHREAD_CFLAGS = ['-pthread']
    if conf.check(linkflags=['-pthread'], mandatory=False):
        if not (conf.env.DEST_OS == 'darwin' and conf.env.CXX_NAME == 'clang'):
            conf.env.PTHREAD_LINKFLAGS += ['-pthread']
    if conf.check(linkflags=['-lpthread'], mandatory=False):
        conf.env.PTHREAD_LINKFLAGS += ['-lpthread']

    conf.define('INGEN_SHARED', 1)
    conf.define('INGEN_VERSION', INGEN_VERSION)

    if conf.env.HAVE_SIGCPP and not Options.options.no_client:
        conf.env.INGEN_BUILD_CLIENT = 1
    else:
        Options.options.no_gui = True

    if not Options.options.no_gui:
        conf.recurse('src/gui')

    conf.env.INGEN_MAJOR_VERSION = INGEN_MAJOR_VERSION

    conf.define('INGEN_DATA_DIR', os.path.join(conf.env.DATADIR, 'ingen'))
    conf.define('INGEN_MODULE_DIR', conf.env.LIBDIR)
    conf.define('INGEN_BUNDLE_DIR', os.path.join(conf.env.LV2DIR, 'ingen.lv2'))

    autowaf.set_lib_env(conf, 'ingen', INGEN_VERSION)
    conf.run_env.append_unique('XDG_DATA_DIRS', [str(conf.path.get_bld())])
    for i in ['src', 'modules']:
        conf.run_env.append_unique(autowaf.lib_path_name, [str(conf.build_path(i))])
    for i in ['src/client', 'src/server', 'src/gui']:
        conf.run_env.append_unique('INGEN_MODULE_PATH', [str(conf.build_path(i))])

    conf.write_config_header('ingen_config.h', remove=False)

    autowaf.display_summary(
        conf,
        {'GUI': bool(conf.env.INGEN_BUILD_GUI),
         'HTML plugin doc support': bool(conf.env.HAVE_WEBKIT),
         'PortAudio driver': bool(conf.env.HAVE_PORTAUDIO),
         'Jack driver': bool(conf.env.HAVE_JACK),
         'Jack session support': conf.is_defined('INGEN_JACK_SESSION'),
         'Jack metadata support': conf.is_defined('HAVE_JACK_METADATA'),
         'LV2 plugin driver': bool(conf.env.INGEN_BUILD_LV2),
         'LV2 bundle': conf.env.INGEN_BUNDLE_DIR,
         'LV2 plugin support': bool(conf.env.HAVE_LILV),
         'Socket interface': conf.is_defined('HAVE_SOCKET')})


unit_tests = ['tst_FilePath']


def build(bld):
    opts           = Options.options
    opts.datadir   = opts.datadir or bld.env.PREFIX + 'share'
    opts.moduledir = opts.moduledir or bld.env.PREFIX + 'lib/ingen'

    # Headers
    for i in ['', 'client']:
        bld.install_files('${INCLUDEDIR}/ingen/%s' % i,
                          bld.path.ant_glob('ingen/%s/*' % i))

    # Python modules
    if bld.env.PYTHONDIR:
        bld.install_files('${PYTHONDIR}/', 'scripts/ingen.py')

    # Modules
    bld.recurse('src')
    bld.recurse('src/server')

    if bld.env.INGEN_BUILD_CLIENT:
        bld.recurse('src/client')

    if bld.env.INGEN_BUILD_GUI:
        bld.recurse('src/gui')

    # Program
    bld(features     = 'c cxx cxxprogram',
        source       = 'src/ingen/ingen.cpp',
        target       = 'ingen',
        includes     = ['.'],
        use          = 'libingen',
        uselib       = 'SERD SORD SRATOM RAUL LILV LV2',
        install_path = '${BINDIR}')

    # Test program
    if bld.env.BUILD_TESTS:
        for i in ['ingen_test', 'ingen_bench'] + unit_tests:
            bld(features     = 'cxx cxxprogram',
                source       = 'tests/%s.cpp' % i,
                target       = 'tests/%s' % i,
                includes     = ['.'],
                use          = 'libingen',
                uselib       = 'SERD SORD SRATOM RAUL LILV LV2',
                install_path = '',
                cxxflags     = bld.env.INGEN_TEST_CXXFLAGS,
                linkflags    = bld.env.INGEN_TEST_LINKFLAGS)

    bld.install_files('${DATADIR}/applications', 'src/ingen/ingen.desktop')
    bld.install_files('${BINDIR}', 'scripts/ingenish', chmod=Utils.O755)
    bld.install_files('${BINDIR}', 'scripts/ingenams', chmod=Utils.O755)

    # Code documentation
    autowaf.build_dox(bld, 'INGEN', INGEN_VERSION, top, out)

    # Ontology documentation
    if bld.env.DOCS:
        bld(rule='lv2specgen.py ${SRC} ${TGT} -i -p ingen --copy-style --list-email ingen@drobilla.net --list-page http://lists.drobilla.net/listinfo.cgi/ingen-drobilla.net',
            source = 'bundles/ingen.lv2/ingen.ttl',
            target = 'ingen.lv2/ingen.html')

    # Man page
    bld.install_files('${MANDIR}/man1', 'doc/ingen.1')

    # Icons
    icon_dir   = os.path.join(bld.env.DATADIR, 'icons', 'hicolor')
    icon_sizes = [16, 22, 24, 32, 48, 64, 128, 256]
    for s in icon_sizes:
        d = '%dx%d' % (s, s)
        bld.install_as(
            os.path.join(icon_dir, d, 'apps', 'ingen.png'),
            os.path.join('icons', d, 'ingen.png'))

    bld.install_as(
        os.path.join(icon_dir, 'scalable', 'apps', 'ingen.svg'),
        os.path.join('icons', 'scalable', 'ingen.svg'))

    bld.install_files('${LV2DIR}/ingen.lv2/',
                      bld.path.ant_glob('bundles/ingen.lv2/*'))

    # Install template graph bundles
    for c in ['Stereo', 'Mono']:
        for t in ['Effect', 'Instrument']:
            bundle = '%s%s.ingen' % (c, t)
            bld.install_files('${LV2DIR}/%s/' % bundle,
                              bld.path.ant_glob('bundles/%s/*' % bundle))

    bld.add_post_fun(autowaf.run_ldconfig)


def lint(ctx):
    "checks code for style issues"
    import subprocess

    status = 0

    # Check Python style with flake8
    try:
        for i in ["src/client/wscript",
                  "src/gui/wscript",
                  "src/server/wscript",
                  "src/wscript",
                  "scripts/ingen.py",
                  "scripts/ingenish",
                  "scripts/ingenams",
                  "wscript"]:
            status += subprocess.call(["flake8",
                                       "--ignore", "E221,W504,E251,E501",
                                       i])
    except Exception:
        Logs.warn('warning: Failed to call flake8')

    cmd = ("clang-tidy -p=. -header-filter=ingen/ -checks=\"*," +
           "-clang-analyzer-alpha.*," +
           "-cppcoreguidelines-*," +
           "-cppcoreguidelines-pro-type-union-access," +
           "-google-build-using-namespace," +
           "-google-readability-casting," +
           "-google-readability-todo," +
           "-llvm-header-guard," +
           "-llvm-include-order," +
           "-llvm-namespace-comment," +
           "-misc-unused-parameters," +
           "-readability-else-after-return," +
           "-readability-implicit-bool-cast," +
           "-readability-named-parameter\" " +
           "$(find .. -name '*.cpp')")
    subprocess.call(cmd, cwd='build', shell=True)

    if status != 0:
        ctx.fatal("Lint checks failed")


def upload_docs(ctx):
    # Ontology documentation
    os.system('rsync -avz -e ssh bundles/ingen.lv2/ingen.ttl drobilla@drobilla.net:~/drobilla.net/ns/')
    os.system('rsync -avz -e ssh build/ingen.lv2/ingen.html drobilla@drobilla.net:~/drobilla.net/ns/')
    os.system('rsync -avz -e ssh build/ingen.lv2/style.css drobilla@drobilla.net:~/drobilla.net/ns/')

    # Doxygen documentation
    os.system('rsync -ravz --delete -e ssh build/doc/html/* drobilla@drobilla.net:~/drobilla.net/docs/ingen/')


def test(tst):
    with tst.group('unit') as check:
        for i in unit_tests:
            check(['./tests/' + i])

    with tst.group('integration') as check:
        empty      = tst.src_path('tests/empty.ingen')
        empty_main = os.path.join(empty, 'main.ttl')
        for i in tst.path.ant_glob('tests/*.ttl'):
            base = os.path.basename(i.abspath().replace('.ttl', ''))

            # Run test
            check(['./tests/ingen_test',
                   '--load', empty,
                   '--execute', os.path.relpath(i.abspath(), os.getcwd())])

            # Check undo output for changes
            check.file_equals(empty_main, base + '.undo.ingen/main.ttl')

            # Check redo output for changes
            check.file_equals(base + '.out.ingen/main.ttl',
                              base + '.redo.ingen/main.ttl')