#!/usr/bin/env python import glob import os import sys from waflib import Build, Logs, Options from waflib.extras import autowaf # Library and package version (UNIX style major, minor, micro) # major increment <=> incompatible changes # minor increment <=> compatible changes (additions) # micro increment <=> no interface changes SERD_VERSION = '0.30.11' SERD_MAJOR_VERSION = '0' # Mandatory waf variables APPNAME = 'serd' # Package name for waf dist VERSION = SERD_VERSION # Package version for waf dist top = '.' # Source directory out = 'build' # Build directory # Release variables uri = 'http://drobilla.net/sw/serd' dist_pattern = 'http://download.drobilla.net/serd-%d.%d.%d.tar.bz2' post_tags = ['Hacking', 'RDF', 'Serd'] def options(ctx): ctx.load('compiler_c') ctx.add_flags( ctx.configuration_options(), {'no-utils': 'do not build command line utilities', 'stack-check': 'include runtime stack sanity checks', 'static': 'build static library', 'no-shared': 'do not build shared library', 'static-progs': 'build programs as static binaries', 'largefile': 'build with large file support on 32-bit systems', 'no-posix': 'do not use POSIX functions, even if present'}) def configure(conf): conf.load('compiler_c', cache=True) conf.load('autowaf', cache=True) if conf.env.DOCS: conf.load('sphinx') if not autowaf.set_c_lang(conf, 'c11', mandatory=False): autowaf.set_c_lang(conf, 'c99') if Options.options.strict: # Check for programs used by lint target conf.find_program("flake8", var="FLAKE8", mandatory=False) conf.find_program("clang-tidy", var="CLANG_TIDY", mandatory=False) conf.find_program("iwyu_tool", var="IWYU_TOOL", mandatory=False) if Options.options.ultra_strict: autowaf.add_compiler_flags(conf.env, '*', { 'clang': [ '-Wno-cast-align', '-Wno-cast-qual', '-Wno-conversion', '-Wno-disabled-macro-expansion', '-Wno-double-promotion', '-Wno-format-nonliteral', '-Wno-nullability-extension', '-Wno-nullable-to-nonnull-conversion', '-Wno-padded', '-Wno-reserved-id-macro', '-Wno-sign-conversion', ], 'gcc': [ '-Wno-cast-align', '-Wno-cast-qual', '-Wno-float-conversion', '-Wno-inline', '-Wno-padded', '-Wno-sign-conversion', ], 'msvc': [ '/wd4061', # enumerator in switch is not explicitly handled '/wd4365', # signed/unsigned mismatch '/wd4514', # unreferenced inline function has been removed '/wd4710', # function not inlined '/wd4711', # function selected for automatic inline expansion '/wd4800', # implicit conversion from int to bool '/wd4820', # padding added after construct '/wd4996', # POSIX name for this item is deprecated ], }) autowaf.add_compiler_flags(conf.env, 'c', { 'clang': [ '-Wno-bad-function-cast', ], 'gcc': [ '-Wno-bad-function-cast', ], 'msvc': [ '/wd4706', # assignment within conditional expression '/wd5045', # will insert Spectre mitigation for memory load ], }) if 'mingw' in conf.env.CC[0]: conf.env.append_value('CFLAGS', '-Wno-unused-macros') conf.env.update({ 'BUILD_UTILS': not Options.options.no_utils, 'BUILD_SHARED': not Options.options.no_shared, 'STATIC_PROGS': Options.options.static_progs, 'BUILD_STATIC': (Options.options.static or Options.options.static_progs)}) if not conf.env.BUILD_SHARED and not conf.env.BUILD_STATIC: conf.fatal('Neither a shared nor a static build requested') if Options.options.stack_check: conf.define('SERD_STACK_CHECK', SERD_VERSION) if Options.options.largefile: conf.env.append_unique('DEFINES', ['_FILE_OFFSET_BITS=64']) if not Options.options.no_posix: funcs = {'posix_memalign': ('stdlib.h', 'int', 'void**,size_t,size_t'), 'posix_fadvise': ('fcntl.h', 'int', 'int,off_t,off_t,int'), 'fileno': ('stdio.h', 'int', 'FILE*')} for name, (header, ret, args) in funcs.items(): conf.check_function('c', name, header_name = header, return_type = ret, arg_types = args, define_name = 'HAVE_' + name.upper(), defines = ['_POSIX_C_SOURCE=200809L'], mandatory = False) # Set up environment for building/using as a subproject autowaf.set_lib_env(conf, 'serd', SERD_VERSION, include_path=str(conf.path.find_node('include'))) if conf.env.BUILD_TESTS: serdi_node = conf.path.get_bld().make_node('serdi_static') else: serdi_node = conf.path.get_bld().make_node('serdi') conf.env.SERDI = [serdi_node.abspath()] conf.define('SERD_NO_DEFAULT_CONFIG', 1) autowaf.display_summary( conf, {'Build static library': bool(conf.env['BUILD_STATIC']), 'Build shared library': bool(conf.env['BUILD_SHARED']), 'Build utilities': bool(conf.env['BUILD_UTILS']), 'Build unit tests': bool(conf.env['BUILD_TESTS'])}) lib_headers = ['src/reader.h'] lib_source = ['src/base64.c', 'src/byte_source.c', 'src/env.c', 'src/n3.c', 'src/node.c', 'src/reader.c', 'src/string.c', 'src/system.c', 'src/uri.c', 'src/writer.c'] def build(bld): # C Headers includedir = '${INCLUDEDIR}/serd-%s/serd' % SERD_MAJOR_VERSION bld.install_files(includedir, bld.path.ant_glob('include/serd/*.h')) # Pkgconfig file autowaf.build_pc(bld, 'SERD', SERD_VERSION, SERD_MAJOR_VERSION, [], {'SERD_MAJOR_VERSION': SERD_MAJOR_VERSION}) defines = [] lib_args = {'export_includes': ['include'], 'includes': ['include'], 'cflags': ['-fvisibility=hidden'], 'lib': ['m'], 'vnum': SERD_VERSION, 'install_path': '${LIBDIR}'} if bld.env.MSVC_COMPILER: lib_args['cflags'] = [] lib_args['lib'] = [] defines = [] # Shared Library if bld.env.BUILD_SHARED: bld(features = 'c cshlib', source = lib_source, name = 'libserd', target = 'serd-%s' % SERD_MAJOR_VERSION, defines = defines + ['SERD_INTERNAL'], **lib_args) # Static library if bld.env.BUILD_STATIC: bld(features = 'c cstlib', source = lib_source, name = 'libserd_static', target = 'serd-%s' % SERD_MAJOR_VERSION, defines = defines + ['SERD_STATIC', 'SERD_INTERNAL'], **lib_args) if bld.env.BUILD_TESTS: coverage_flags = [''] if bld.env.NO_COVERAGE else ['--coverage'] test_args = {'includes': ['include'], 'cflags': coverage_flags, 'linkflags': coverage_flags, 'lib': lib_args['lib'], 'install_path': ''} # Profiled static library for test coverage bld(features = 'c cstlib', source = lib_source, name = 'libserd_profiled', target = 'serd_profiled', defines = defines + ['SERD_STATIC', 'SERD_INTERNAL'], **test_args) # Test programs for prog in [('serdi_static', 'src/serdi.c'), ('test_env', 'test/test_env.c'), ('test_free_null', 'test/test_free_null.c'), ('test_node', 'test/test_node.c'), ('test_read_chunk', 'test/test_read_chunk.c'), ('test_reader_writer', 'test/test_reader_writer.c'), ('test_string', 'test/test_string.c'), ('test_uri', 'test/test_uri.c'), ('test_writer', 'test/test_writer.c')]: bld(features = 'c cprogram', source = prog[1], use = 'libserd_profiled', target = prog[0], defines = defines + ['SERD_STATIC'], **test_args) # Utilities if bld.env.BUILD_UTILS: obj = bld(features = 'c cprogram', source = 'src/serdi.c', target = 'serdi', includes = ['include'], use = 'libserd', lib = lib_args['lib'], install_path = '${BINDIR}') if not bld.env.BUILD_SHARED or bld.env.STATIC_PROGS: obj.use = 'libserd_static' if bld.env.STATIC_PROGS: obj.env.SHLIB_MARKER = obj.env.STLIB_MARKER obj.linkflags = ['-static'] # Documentation if bld.env.DOCS: bld.env.SERD_MAJOR_VERSION = SERD_MAJOR_VERSION bld.recurse('doc/c') # Man page bld.install_files('${MANDIR}/man1', 'doc/serdi.1') bld.add_post_fun(autowaf.run_ldconfig) class LintContext(Build.BuildContext): fun = cmd = 'lint' def lint(ctx): "checks code for style issues" import subprocess st = 0 if "FLAKE8" in ctx.env: Logs.info("Running flake8") st = subprocess.call([ctx.env.FLAKE8[0], "wscript", "--ignore", "E101,E129,W191,E221,W504,E251,E241,E741"]) st += subprocess.call([ctx.env.FLAKE8[0], "scripts/serd_bench.py", "--ignore", "E203"]) else: Logs.warn("Not running flake8") if "IWYU_TOOL" in ctx.env: Logs.info("Running include-what-you-use") cmd = [ctx.env.IWYU_TOOL[0], "-o", "clang", "-p", "build"] output = subprocess.check_output(cmd).decode('utf-8') if 'error: ' in output: sys.stdout.write(output) st += 1 else: Logs.warn("Not running include-what-you-use") if "CLANG_TIDY" in ctx.env and "clang" in ctx.env.CC[0]: Logs.info("Running clang-tidy") sources = glob.glob('include/serd/*.h*') sources += glob.glob('src/*.c') sources += glob.glob('test/*.c') sources = list(map(os.path.abspath, sources)) procs = [] for source in sources: cmd = [ctx.env.CLANG_TIDY[0], "--quiet", "-p=.", source] procs += [subprocess.Popen(cmd, cwd="build")] for proc in procs: stdout, stderr = proc.communicate() st += proc.returncode else: Logs.warn("Not running clang-tidy") if st != 0: sys.exit(st) def amalgamate(ctx): "builds single-file amalgamated source" import shutil import re shutil.copy('serd/serd.h', 'build/serd.h') def include_line(line): return (not re.match(r'#include "[^/]*\.h"', line) and not re.match('#include "serd/serd.h"', line)) with open('build/serd.c', 'w') as amalgamation: amalgamation.write('/* This is amalgamated code, do not edit! */\n') amalgamation.write('#include "serd.h"\n\n') for header_path in ['src/serd_internal.h', 'src/system.h', 'src/byte_sink.h', 'src/byte_source.h', 'src/stack.h', 'src/string_utils.h', 'src/uri_utils.h', 'src/reader.h']: with open(header_path) as header: for l in header: if include_line(l): amalgamation.write(l) for f in lib_headers + lib_source: with open(f) as fd: amalgamation.write('\n/**\n @file %s\n*/' % f) for l in fd: if include_line(l): amalgamation.write(l) for i in ['c', 'h']: Logs.info('Wrote build/serd.%s' % i) def test(tst): import tempfile # Create test output directories for i in ['bad', 'good', 'lax', 'TurtleTests', 'NTriplesTests', 'NQuadsTests', 'TriGTests']: try: test_dir = os.path.join('test', i) os.makedirs(test_dir) for i in glob.glob(test_dir + '/*.*'): os.remove(i) except Exception: pass serdi = './serdi_static' srcdir = tst.path.abspath() with tst.group('Unit') as check: check(['./test_env']) check(['./test_free_null']) check(['./test_node']) check(['./test_read_chunk']) check(['./test_reader_writer']) check(['./test_string']) check(['./test_uri']) check(['./test_writer']) def test_syntax_io(check, in_name, check_name, lang): in_path = 'test/good/%s' % in_name out_path = in_path + '.io' check_path = '%s/test/good/%s' % (srcdir, check_name) check([serdi, '-o', lang, '%s/%s' % (srcdir, in_path), in_path], stdout=out_path, name=in_name) check.file_equals(check_path, out_path) with tst.group('ThroughSyntax') as check: test_syntax_io(check, 'base.ttl', 'base.ttl', 'turtle') test_syntax_io(check, 'qualify-in.ttl', 'qualify-out.ttl', 'turtle') with tst.group('GoodCommands') as check: check([serdi, '%s/serd.ttl' % srcdir], stdout=os.devnull) check([serdi, '-li', 'turtle', '%s/test/lax/test-bad-string.ttl' % srcdir], stdout=os.devnull) check([serdi, '-v']) check([serdi, '-h']) check([serdi, '-s', ' a .']) check([serdi, os.devnull]) with tempfile.TemporaryFile(mode='r') as stdin: check([serdi, '-'], stdin=stdin) with tst.group('BadCommands', expected=1, stderr=autowaf.NONEMPTY) as check: check([serdi]) check([serdi, '/no/such/file']) check([serdi, 'ftp://example.org/unsupported.ttl']) check([serdi, '-c']) check([serdi, '-i', 'illegal']) check([serdi, '-i', 'turtle']) check([serdi, '-i']) check([serdi, '-fi']) check([serdi, '-o', 'illegal']) check([serdi, '-o']) check([serdi, '-p']) check([serdi, '-q', '%s/test/bad/bad-base.ttl' % srcdir], stderr=None) check([serdi, '-r']) check([serdi, '-z']) check([serdi, '-s', ' a .']) with tst.group('IoErrors', expected=1) as check: check([serdi, '-e', 'file://%s/' % srcdir], name='Read directory') check([serdi, 'file://%s/' % srcdir], name='Bulk read directory') if os.path.exists('/dev/full'): check([serdi, 'file://%s/test/good/manifest.ttl' % srcdir], stdout='/dev/full', name='Write error') if sys.version_info.major >= 3: from waflib.extras import autoship try: with tst.group('NEWS') as check: news_path = os.path.join(srcdir, 'NEWS') entries = autoship.read_news(top=srcdir) autoship.write_news(entries, 'NEWS.norm') check.file_equals(news_path, 'NEWS.norm') meta_path = os.path.join(srcdir, 'serd.ttl') autoship.write_news(entries, 'NEWS.ttl', format='turtle', template=meta_path) ttl_entries = autoship.read_news('NEWS.ttl', top=srcdir, format='turtle') autoship.write_news(ttl_entries, 'NEWS.round') check.file_equals(news_path, 'NEWS.round') except ImportError: Logs.warn('Failed to import rdflib, not running NEWS tests') run_test_suite = ['../test/run_test_suite.py', '--serdi', './serdi_static'] with tst.group('TestSuites') as check: # Run serd-specific test suites serd_base = 'http://drobilla.net/sw/serd/test/' check(run_test_suite + ['../test/good/manifest.ttl', serd_base + 'good/']) check(run_test_suite + ['../test/bad/manifest.ttl', serd_base + 'bad/']) check(run_test_suite + ['../test/lax/manifest.ttl', serd_base + 'lax/', '--', '-l']) check(run_test_suite + ['../test/lax/manifest.ttl', serd_base + 'lax/']) # Start test report for standard test suites report_filename = 'earl.ttl' with open(report_filename, 'w') as report: report.write('@prefix earl: .\n' '@prefix dc: .\n') with open(os.path.join(srcdir, 'serd.ttl')) as serd_ttl: report.writelines(serd_ttl) # Run standard test suites w3c_base = 'http://www.w3.org/2013/' check(run_test_suite + [ '--syntax', 'Turtle', '--report', report_filename, '../test/TurtleTests/manifest.ttl', w3c_base + 'TurtleTests/']) check(run_test_suite + [ '--syntax', 'NTriples', '--report', report_filename, '../test/NTriplesTests/manifest.ttl', w3c_base + 'NTriplesTests/']) check(run_test_suite + [ '--syntax', 'NQuads', '--report', report_filename, '../test/NQuadsTests/manifest.ttl', w3c_base + 'NQuadsTests/']) check(run_test_suite + [ '--syntax', 'TriG', '--report', report_filename, '../test/TriGTests/manifest.ttl', w3c_base + 'TriGTests/', '--', '-a'])