diff --git a/configure.ac b/configure.ac index 0dfe64195e204a9c4610ffbee8deb15d8a61b1b2..40a73acafe8a8fd3f95dd1e40693eb14010ab418 100644 --- a/configure.ac +++ b/configure.ac @@ -3,7 +3,7 @@ AC_PREREQ(2.60) # Version/identifiers m4_define([VERSION_MAJOR],[0]) m4_define([VERSION_MINOR],[0]) -m4_define([VERSION_MICRO],[1]) +m4_define([VERSION_MICRO],[2]) AC_INIT(knot-resolver, m4_defn([VERSION_MAJOR]).m4_defn([VERSION_MINOR]).m4_defn([VERSION_MICRO]), knot-dns@lists.nic.cz, knot-resolver) AM_INIT_AUTOMAKE([gnits subdir-objects dist-xz -Wall -Werror]) AM_SILENT_RULES([yes]) @@ -21,9 +21,7 @@ CFLAGS="$CFLAGS -Wall -Werror=format-security" # Checks for programs. m4_ifdef([AM_PROG_AR], [AM_PROG_AR]) # call AM_PROG_AR only if available -# Initialize libtool -AC_DISABLE_STATIC -AC_PROG_LIBTOOL +# Linker LT_INIT # Use pkg-config @@ -34,7 +32,14 @@ AC_CHECK_FUNCS([nftw, mkdtemp]) PKG_CHECK_MODULES([libknot], [libknot]) PKG_CHECK_MODULES([libuv], [libuv], [build_daemon=yes], [build_daemon=no]) PKG_CHECK_MODULES([cmocka], [cmocka], [build_tests=yes], [build_tests=no]) + +# Check for python (integration tests) +AX_PYTHON_DEVEL([>= '2.5.0']) +if test "x$PYTHON" != x ; then build_integration=yes; else build_integration=no; fi + +# Set up conditionals AM_CONDITIONAL([BUILD_TESTS], [test "$build_tests" == "yes"]) +AM_CONDITIONAL([BUILD_INTEGRATION], [test "$build_integration" == "yes"]) AM_CONDITIONAL([BUILD_DAEMON], [test "$build_daemon" == "yes"]) # Search other libraries @@ -55,7 +60,7 @@ AC_MSG_RESULT([ Features: --------- - Build tests: ${build_tests} + Build tests: ${build_tests} (integration ${build_integration}) Build daemon: ${build_daemon} Continue with 'make' command diff --git a/knot-resolver.files b/knot-resolver.files index 70be30723cbe1fa185a5003b57c9e9c7dfd6f3bd..469648cce69e664241c9be4ab5692126e9ec4f7b 100644 --- a/knot-resolver.files +++ b/knot-resolver.files @@ -37,4 +37,5 @@ tests/Makefile.am tests/test.h tests/test_cache.c tests/test_context.c +tests/test_integration.c tests/test_resolve.c diff --git a/m4/ax_python_devel.m4 b/m4/ax_python_devel.m4 new file mode 100644 index 0000000000000000000000000000000000000000..59a2ff0903022d90bd0f6f60ddcbc7951ff7276b --- /dev/null +++ b/m4/ax_python_devel.m4 @@ -0,0 +1,324 @@ +# =========================================================================== +# http://www.gnu.org/software/autoconf-archive/ax_python_devel.html +# =========================================================================== +# +# SYNOPSIS +# +# AX_PYTHON_DEVEL([version]) +# +# DESCRIPTION +# +# Note: Defines as a precious variable "PYTHON_VERSION". Don't override it +# in your configure.ac. +# +# This macro checks for Python and tries to get the include path to +# 'Python.h'. It provides the $(PYTHON_CPPFLAGS) and $(PYTHON_LDFLAGS) +# output variables. It also exports $(PYTHON_EXTRA_LIBS) and +# $(PYTHON_EXTRA_LDFLAGS) for embedding Python in your code. +# +# You can search for some particular version of Python by passing a +# parameter to this macro, for example ">= '2.3.1'", or "== '2.4'". Please +# note that you *have* to pass also an operator along with the version to +# match, and pay special attention to the single quotes surrounding the +# version number. Don't use "PYTHON_VERSION" for this: that environment +# variable is declared as precious and thus reserved for the end-user. +# +# This macro should work for all versions of Python >= 2.1.0. As an end +# user, you can disable the check for the python version by setting the +# PYTHON_NOVERSIONCHECK environment variable to something else than the +# empty string. +# +# If you need to use this macro for an older Python version, please +# contact the authors. We're always open for feedback. +# +# LICENSE +# +# Copyright (c) 2009 Sebastian Huber <sebastian-huber@web.de> +# Copyright (c) 2009 Alan W. Irwin +# Copyright (c) 2009 Rafael Laboissiere <rafael@laboissiere.net> +# Copyright (c) 2009 Andrew Collier +# Copyright (c) 2009 Matteo Settenvini <matteo@member.fsf.org> +# Copyright (c) 2009 Horst Knorr <hk_classes@knoda.org> +# Copyright (c) 2013 Daniel Mullner <muellner@math.stanford.edu> +# +# 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/>. +# +# As a special exception, the respective Autoconf Macro's copyright owner +# gives unlimited permission to copy, distribute and modify the configure +# scripts that are the output of Autoconf when processing the Macro. You +# need not follow the terms of the GNU General Public License when using +# or distributing such scripts, even though portions of the text of the +# Macro appear in them. The GNU General Public License (GPL) does govern +# all other use of the material that constitutes the Autoconf Macro. +# +# This special exception to the GPL applies to versions of the Autoconf +# Macro released by the Autoconf Archive. When you make and distribute a +# modified version of the Autoconf Macro, you may extend this special +# exception to the GPL to apply to your modified version as well. + +#serial 17 + +AU_ALIAS([AC_PYTHON_DEVEL], [AX_PYTHON_DEVEL]) +AC_DEFUN([AX_PYTHON_DEVEL],[ + # + # Allow the use of a (user set) custom python version + # + AC_ARG_VAR([PYTHON_VERSION],[The installed Python + version to use, for example '2.3'. This string + will be appended to the Python interpreter + canonical name.]) + + AC_PATH_PROG([PYTHON],[python[$PYTHON_VERSION]]) + if test -z "$PYTHON"; then + AC_MSG_ERROR([Cannot find python$PYTHON_VERSION in your system path]) + PYTHON_VERSION="" + fi + + # + # Check for a version of Python >= 2.1.0 + # + AC_MSG_CHECKING([for a version of Python >= '2.1.0']) + ac_supports_python_ver=`$PYTHON -c "import sys; \ + ver = sys.version.split ()[[0]]; \ + print (ver >= '2.1.0')"` + if test "$ac_supports_python_ver" != "True"; then + if test -z "$PYTHON_NOVERSIONCHECK"; then + AC_MSG_RESULT([no]) + AC_MSG_FAILURE([ +This version of the AC@&t@_PYTHON_DEVEL macro +doesn't work properly with versions of Python before +2.1.0. You may need to re-run configure, setting the +variables PYTHON_CPPFLAGS, PYTHON_LDFLAGS, PYTHON_SITE_PKG, +PYTHON_EXTRA_LIBS and PYTHON_EXTRA_LDFLAGS by hand. +Moreover, to disable this check, set PYTHON_NOVERSIONCHECK +to something else than an empty string. +]) + else + AC_MSG_RESULT([skip at user request]) + fi + else + AC_MSG_RESULT([yes]) + fi + + # + # if the macro parameter ``version'' is set, honour it + # + if test -n "$1"; then + AC_MSG_CHECKING([for a version of Python $1]) + ac_supports_python_ver=`$PYTHON -c "import sys; \ + ver = sys.version.split ()[[0]]; \ + print (ver $1)"` + if test "$ac_supports_python_ver" = "True"; then + AC_MSG_RESULT([yes]) + else + AC_MSG_RESULT([no]) + AC_MSG_ERROR([this package requires Python $1. +If you have it installed, but it isn't the default Python +interpreter in your system path, please pass the PYTHON_VERSION +variable to configure. See ``configure --help'' for reference. +]) + PYTHON_VERSION="" + fi + fi + + # + # Check if you have distutils, else fail + # + AC_MSG_CHECKING([for the distutils Python package]) + ac_distutils_result=`$PYTHON -c "import distutils" 2>&1` + if test -z "$ac_distutils_result"; then + AC_MSG_RESULT([yes]) + else + AC_MSG_RESULT([no]) + AC_MSG_ERROR([cannot import Python module "distutils". +Please check your Python installation. The error was: +$ac_distutils_result]) + PYTHON_VERSION="" + fi + + # + # Check for Python include path + # + AC_MSG_CHECKING([for Python include path]) + if test -z "$PYTHON_CPPFLAGS"; then + python_path=`$PYTHON -c "import distutils.sysconfig; \ + print (distutils.sysconfig.get_python_inc ());"` + plat_python_path=`$PYTHON -c "import distutils.sysconfig; \ + print (distutils.sysconfig.get_python_inc (plat_specific=1));"` + if test -n "${python_path}"; then + if test "${plat_python_path}" != "${python_path}"; then + python_path="-I$python_path -I$plat_python_path" + else + python_path="-I$python_path" + fi + fi + PYTHON_CPPFLAGS=$python_path + fi + AC_MSG_RESULT([$PYTHON_CPPFLAGS]) + AC_SUBST([PYTHON_CPPFLAGS]) + + # + # Check for Python library path + # + AC_MSG_CHECKING([for Python library path]) + if test -z "$PYTHON_LDFLAGS"; then + # (makes two attempts to ensure we've got a version number + # from the interpreter) + ac_python_version=`cat<<EOD | $PYTHON - + +# join all versioning strings, on some systems +# major/minor numbers could be in different list elements +from distutils.sysconfig import * +e = get_config_var('VERSION') +if e is not None: + print(e) +EOD` + + if test -z "$ac_python_version"; then + if test -n "$PYTHON_VERSION"; then + ac_python_version=$PYTHON_VERSION + else + ac_python_version=`$PYTHON -c "import sys; \ + print (sys.version[[:3]])"` + fi + fi + + # Make the versioning information available to the compiler + AC_DEFINE_UNQUOTED([HAVE_PYTHON], ["$ac_python_version"], + [If available, contains the Python version number currently in use.]) + + # First, the library directory: + ac_python_libdir=`cat<<EOD | $PYTHON - + +# There should be only one +import distutils.sysconfig +e = distutils.sysconfig.get_config_var('LIBDIR') +if e is not None: + print (e) +EOD` + + # Now, for the library: + ac_python_library=`cat<<EOD | $PYTHON - + +import distutils.sysconfig +c = distutils.sysconfig.get_config_vars() +if 'LDVERSION' in c: + print ('python'+c[['LDVERSION']]) +else: + print ('python'+c[['VERSION']]) +EOD` + + # This small piece shamelessly adapted from PostgreSQL python macro; + # credits goes to momjian, I think. I'd like to put the right name + # in the credits, if someone can point me in the right direction... ? + # + if test -n "$ac_python_libdir" -a -n "$ac_python_library" + then + # use the official shared library + ac_python_library=`echo "$ac_python_library" | sed "s/^lib//"` + PYTHON_LDFLAGS="-L$ac_python_libdir -l$ac_python_library" + else + # old way: use libpython from python_configdir + ac_python_libdir=`$PYTHON -c \ + "from distutils.sysconfig import get_python_lib as f; \ + import os; \ + print (os.path.join(f(plat_specific=1, standard_lib=1), 'config'));"` + PYTHON_LDFLAGS="-L$ac_python_libdir -lpython$ac_python_version" + fi + + if test -z "PYTHON_LDFLAGS"; then + AC_MSG_ERROR([ + Cannot determine location of your Python DSO. Please check it was installed with + dynamic libraries enabled, or try setting PYTHON_LDFLAGS by hand. + ]) + fi + fi + AC_MSG_RESULT([$PYTHON_LDFLAGS]) + AC_SUBST([PYTHON_LDFLAGS]) + + # + # Check for site packages + # + AC_MSG_CHECKING([for Python site-packages path]) + if test -z "$PYTHON_SITE_PKG"; then + PYTHON_SITE_PKG=`$PYTHON -c "import distutils.sysconfig; \ + print (distutils.sysconfig.get_python_lib(0,0));"` + fi + AC_MSG_RESULT([$PYTHON_SITE_PKG]) + AC_SUBST([PYTHON_SITE_PKG]) + + # + # libraries which must be linked in when embedding + # + AC_MSG_CHECKING(python extra libraries) + if test -z "$PYTHON_EXTRA_LIBS"; then + PYTHON_EXTRA_LIBS=`$PYTHON -c "import distutils.sysconfig; \ + conf = distutils.sysconfig.get_config_var; \ + print (conf('LIBS') + ' ' + conf('SYSLIBS'))"` + fi + AC_MSG_RESULT([$PYTHON_EXTRA_LIBS]) + AC_SUBST(PYTHON_EXTRA_LIBS) + + # + # linking flags needed when embedding + # + AC_MSG_CHECKING(python extra linking flags) + if test -z "$PYTHON_EXTRA_LDFLAGS"; then + PYTHON_EXTRA_LDFLAGS=`$PYTHON -c "import distutils.sysconfig; \ + conf = distutils.sysconfig.get_config_var; \ + print (conf('LINKFORSHARED'))"` + fi + AC_MSG_RESULT([$PYTHON_EXTRA_LDFLAGS]) + AC_SUBST(PYTHON_EXTRA_LDFLAGS) + + # + # final check to see if everything compiles alright + # + AC_MSG_CHECKING([consistency of all components of python development environment]) + # save current global flags + ac_save_LIBS="$LIBS" + ac_save_CPPFLAGS="$CPPFLAGS" + LIBS="$ac_save_LIBS $PYTHON_LDFLAGS $PYTHON_EXTRA_LDFLAGS $PYTHON_EXTRA_LIBS" + CPPFLAGS="$ac_save_CPPFLAGS $PYTHON_CPPFLAGS" + AC_LANG_PUSH([C]) + AC_LINK_IFELSE([ + AC_LANG_PROGRAM([[#include <Python.h>]], + [[Py_Initialize();]]) + ],[pythonexists=yes],[pythonexists=no]) + AC_LANG_POP([C]) + # turn back to default flags + CPPFLAGS="$ac_save_CPPFLAGS" + LIBS="$ac_save_LIBS" + + AC_MSG_RESULT([$pythonexists]) + + if test ! "x$pythonexists" = "xyes"; then + AC_MSG_FAILURE([ + Could not link test program to Python. Maybe the main Python library has been + installed in some non-standard library path. If so, pass it to configure, + via the LDFLAGS environment variable. + Example: ./configure LDFLAGS="-L/usr/non-standard-path/python/lib" + ============================================================================ + ERROR! + You probably have to install the development version of the Python package + for your distribution. The exact name of this package varies among them. + ============================================================================ + ]) + PYTHON_VERSION="" + fi + + # + # all done! + # +]) diff --git a/tests/Makefile.am b/tests/Makefile.am index 3069996ca6d10712c5d3b7f4decddeaecf46d78e..cbdb2a305fa1ca2f8ae3684d909a0e083a61bfac 100644 --- a/tests/Makefile.am +++ b/tests/Makefile.am @@ -1,23 +1,54 @@ +# Unit tests if BUILD_TESTS AM_CPPFLAGS = \ -include $(top_builddir)/config.h \ - -I$(top_srcdir)/lib + -I$(top_srcdir)/lib \ + $(libknot_CFLAGS) \ + $(cmocka_CFLAGS) LDADD = \ $(top_builddir)/lib/libknotresolve.la \ - $(libuv_LIBS) \ $(libknot_LIBS) \ $(cmocka_LIBS) + check_PROGRAMS = \ test_cache \ test_context \ test_resolve check-compile-only: $(check_PROGRAMS) - -check-local: $(check_PROGRAMS) +check-local-exec: $(check_PROGRAMS) + @echo "---- Executing unit tests ----" $(top_builddir)/tests/runtests -b $(top_builddir)/tests $(check_PROGRAMS) endif + +# Integration tests +if BUILD_INTEGRATION + +noinst_LTLIBRARIES = _test_integration.la +_test_integration_la_SOURCES = test_integration.c +_test_integration_la_CPPFLAGS = $(PYTHON_CPPFLAGS) $(AM_CPPFLAGS) +_test_integration_la_LDFLAGS = -rpath $(abs_builddir) -module -export-dynamic -shared -avoid-version \ + --wrap=gettimeofday $(PYTHON_LDFLAGS) +_test_integration_la_LIBADD = $(LDADD) + +convenience-link: $(noinst_LTLIBRARIES) + @for soname in `echo | $(EGREP) "^dlname=" $(noinst_LTLIBRARIES) | $(SED) -e "s|^dlname='\(.*\)'|\1|"`; do \ + rm -f $(abs_builddir)/$$soname; $(LN_S) $(abs_builddir)/.libs/$$soname $$soname || true;\ + done + +check-local-integration: convenience-link + @echo "---- Executing integration tests ----" + @$(LIBTOOL) --mode=execute -dlopen $(top_builddir)/lib/libknotresolve.la ./test_integration.py testdata + +clean-local: + @for soname in `echo | $(EGREP) "^dlname=" $(noinst_LTLIBRARIES) | $(SED) -e "s|^dlname='\(.*\)'|\1|"`; do \ + test -L $(abs_builddir)/$$soname && rm -f $(abs_builddir)/$$soname || true; \ + done + +endif + +check-local: check-local-exec check-local-integration diff --git a/tests/test_cache.c b/tests/test_cache.c index 136172bc69903a35939730e60af4580ec932b490..fb4444adb235421b6c83db8bf7c7f9bc579d12f9 100644 --- a/tests/test_cache.c +++ b/tests/test_cache.c @@ -31,7 +31,6 @@ const char *global_env; /* Test cache open */ static void test_open(void **state) { - printf("OPENSDFDSFSDFSFSDFSDF\n"); *state = kr_cache_open(global_env, &global_mm, CACHE_SIZE); assert_non_null(*state); } diff --git a/tests/test_integration.c b/tests/test_integration.c new file mode 100644 index 0000000000000000000000000000000000000000..1e031bbd940b677de8e1a76337f986e2ff9e5607 --- /dev/null +++ b/tests/test_integration.c @@ -0,0 +1,179 @@ +/* Copyright (C) 2014 CZ.NIC, z.s.p.o. <knot-dns@labs.nic.cz> + + 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/>. + */ + +#include <Python.h> +#include <libknot/descriptor.h> +#include <libknot/packet/pkt.h> +#include <libknot/internal/net.h> + +#include "tests/test.h" +#include "lib/resolve.h" + +/* + * Globals + */ +mm_ctx_t global_mm; /* Test memory context */ +struct kr_context global_context; /* Resolution context */ +const char *global_tmpdir = NULL; /* Temporary directory */ +struct timeval _mock_time; /* Mocked system time */ +int _mock_fd; /* Mocked endpoint for recursive queries */ + +/* + * PyModule implementation. + */ + +static PyObject* init(PyObject* self, PyObject* args) +{ + #define CACHE_SIZE 100*1024 + test_mm_ctx_init(&global_mm); + kr_context_init(&global_context, &global_mm); + + global_tmpdir = test_tmpdir_create(); + global_context.cache = kr_cache_open(global_tmpdir, &global_mm, CACHE_SIZE); + if (global_context.cache == NULL) { + test_tmpdir_remove(global_tmpdir); + kr_context_deinit(&global_context); + return NULL; + } + + return Py_BuildValue("s", PACKAGE_STRING " (integration tests)"); +} + +static PyObject* deinit(PyObject* self, PyObject* args) +{ + if (global_tmpdir == NULL) { + return NULL; + } + + kr_cache_close(global_context.cache); + kr_context_deinit(&global_context); + test_tmpdir_remove(global_tmpdir); + global_tmpdir = NULL; + + return Py_BuildValue(""); +} + +static PyObject* resolve(PyObject *self, PyObject *args) +{ + const char *query_wire = NULL; + size_t query_size = 0; + if (!PyArg_ParseTuple(args, "s#", &query_wire, &query_size)) { + return NULL; + } + + /* Prepare input */ + knot_pkt_t *query = knot_pkt_new((uint8_t *)query_wire, query_size, &global_mm); + assert(query); + int ret = knot_pkt_parse(query, 0); + if (ret != KNOT_EOK) { + knot_pkt_free(&query); + return NULL; + } + + /* Resolve query */ + knot_pkt_t *answer = knot_pkt_new(NULL, KNOT_WIRE_MAX_PKTSIZE, &global_mm); + assert(answer); + knot_pkt_init_response(answer, query); + ret = kr_resolve(&global_context, answer, knot_pkt_qname(query), + knot_pkt_qclass(query), knot_pkt_qtype(query)); + + /* Return wire and cleanup. */ + PyObject *out = Py_BuildValue("s#", answer->wire, answer->size); + knot_pkt_free(&answer); + knot_pkt_free(&query); + return out; +} + +static PyObject* set_time(PyObject *self, PyObject *args) +{ + unsigned long arg_time = 0; + if (!PyArg_ParseTuple(args, "k", &arg_time)) { + return NULL; + } + + _mock_time.tv_sec = arg_time; + _mock_time.tv_usec = 0; + + return Py_BuildValue(""); +} + +static PyObject* set_endpoint(PyObject *self, PyObject *args) +{ + int fd = PyObject_AsFileDescriptor(args); + if (fd < 0) { + return NULL; + } + + _mock_fd = fd; + return Py_BuildValue(""); +} + +static PyMethodDef module_methods[] = { + {"init", init, METH_VARARGS, "Initialize resolution context."}, + {"deinit", deinit, METH_VARARGS, "Clean up resolution context."}, + {"resolve", resolve, METH_VARARGS, "Resolve query."}, + {"set_time", set_time, METH_VARARGS, "Set mock system time."}, + {"set_endpoint", set_endpoint, METH_VARARGS, "Set endpoint for recursive queries."}, + {NULL, NULL, 0, NULL} +}; + +PyMODINIT_FUNC init_test_integration(void) +{ + (void) Py_InitModule("_test_integration", module_methods); +} + +/* + * Mock symbol reimplementation. + * These effectively allow to manipulate time/networking during resolution. + */ + +int __wrap_gettimeofday(struct timeval *tv, struct timezone *tz) +{ + memcpy(tv, &_mock_time, sizeof(struct timeval)); + return 0; +} + +int net_unbound_socket(int type, const struct sockaddr_storage *ss) +{ + char addr_str[SOCKADDR_STRLEN]; + sockaddr_tostr(addr_str, sizeof(addr_str), ss); + fprintf(stderr, "%s (%d, %s)\n", __func__, type, addr_str); + return _mock_fd; +} + +int net_bound_socket(int type, const struct sockaddr_storage *ss) +{ + char addr_str[SOCKADDR_STRLEN]; + sockaddr_tostr(addr_str, sizeof(addr_str), ss); + fprintf(stderr, "%s (%d, %s)\n", __func__, type, addr_str); + return _mock_fd; +} + +int net_connected_socket(int type, const struct sockaddr_storage *dst_addr, + const struct sockaddr_storage *src_addr, unsigned flags) +{ + char dst_addr_str[SOCKADDR_STRLEN], src_addr_str[SOCKADDR_STRLEN]; + sockaddr_tostr(dst_addr_str, sizeof(dst_addr_str), dst_addr); + sockaddr_tostr(src_addr_str, sizeof(src_addr_str), src_addr); + fprintf(stderr, "%s (%d, %s, %s, %u)\n", __func__, type, dst_addr_str, src_addr_str, flags); + return _mock_fd; +} + +int net_is_connected(int fd) +{ + fprintf(stderr, "%s (%d)\n", __func__, fd); + return true; +} diff --git a/tests/test_integration.py b/tests/test_integration.py new file mode 100755 index 0000000000000000000000000000000000000000..bb1087ea37a1e89ab06f6fc201fbff9f8a58d63d --- /dev/null +++ b/tests/test_integration.py @@ -0,0 +1,7 @@ +#!/usr/bin/env python +import sys +import _test_integration + +print dir(_test_integration) +print sys.argv +print _test_integration.init()