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()