2011-01-07 04:44:19 +03:00
|
|
|
# testutils.py - utility module for psycopg2 testing.
|
2011-02-12 03:40:23 +03:00
|
|
|
|
2011-01-07 04:44:19 +03:00
|
|
|
#
|
|
|
|
# Copyright (C) 2010-2011 Daniele Varrazzo <daniele.varrazzo@gmail.com>
|
|
|
|
#
|
|
|
|
# psycopg2 is free software: you can redistribute it and/or modify it
|
|
|
|
# under the terms of the GNU Lesser General Public License as published
|
|
|
|
# by the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# In addition, as a special exception, the copyright holders give
|
|
|
|
# permission to link this program with the OpenSSL library (or with
|
|
|
|
# modified versions of OpenSSL that use the same license as OpenSSL),
|
|
|
|
# and distribute linked combinations including the two.
|
|
|
|
#
|
|
|
|
# You must obey the GNU Lesser General Public License in all respects for
|
|
|
|
# all of the code used other than OpenSSL.
|
|
|
|
#
|
|
|
|
# psycopg2 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 Lesser General Public
|
|
|
|
# License for more details.
|
|
|
|
|
2010-11-19 06:55:37 +03:00
|
|
|
|
|
|
|
# Use unittest2 if available. Otherwise mock a skip facility with warnings.
|
|
|
|
|
2011-02-11 01:59:31 +03:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
|
2010-11-19 06:55:37 +03:00
|
|
|
try:
|
|
|
|
import unittest2
|
|
|
|
unittest = unittest2
|
|
|
|
except ImportError:
|
|
|
|
import unittest
|
|
|
|
unittest2 = None
|
|
|
|
|
|
|
|
if hasattr(unittest, 'skipIf'):
|
2010-11-19 21:01:29 +03:00
|
|
|
skip = unittest.skip
|
|
|
|
skipIf = unittest.skipIf
|
2010-11-19 06:55:37 +03:00
|
|
|
|
|
|
|
else:
|
|
|
|
import warnings
|
|
|
|
|
|
|
|
def skipIf(cond, msg):
|
|
|
|
def skipIf_(f):
|
|
|
|
def skipIf__(self):
|
|
|
|
if cond:
|
|
|
|
warnings.warn(msg)
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
return skipIf__
|
|
|
|
return skipIf_
|
|
|
|
|
|
|
|
def skip(msg):
|
|
|
|
return skipIf(True, msg)
|
|
|
|
|
|
|
|
def skipTest(self, msg):
|
|
|
|
warnings.warn(msg)
|
|
|
|
return
|
|
|
|
|
|
|
|
unittest.TestCase.skipTest = skipTest
|
|
|
|
|
2011-02-11 01:59:31 +03:00
|
|
|
# Silence warnings caused by the stubborness of the Python unittest maintainers
|
|
|
|
# http://bugs.python.org/issue9424
|
|
|
|
if not hasattr(unittest.TestCase, 'assert_') \
|
|
|
|
or unittest.TestCase.assert_ is not unittest.TestCase.assertTrue:
|
|
|
|
# mavaff...
|
|
|
|
unittest.TestCase.assert_ = unittest.TestCase.assertTrue
|
|
|
|
unittest.TestCase.failUnless = unittest.TestCase.assertTrue
|
|
|
|
unittest.TestCase.assertEquals = unittest.TestCase.assertEqual
|
|
|
|
unittest.TestCase.failUnlessEqual = unittest.TestCase.assertEqual
|
|
|
|
|
2010-11-19 06:55:37 +03:00
|
|
|
|
|
|
|
def decorate_all_tests(cls, decorator):
|
|
|
|
"""Apply *decorator* to all the tests defined in the TestCase *cls*."""
|
|
|
|
for n in dir(cls):
|
|
|
|
if n.startswith('test'):
|
|
|
|
setattr(cls, n, decorator(getattr(cls, n)))
|
2010-11-28 18:03:34 +03:00
|
|
|
|
|
|
|
|
2011-01-18 04:46:10 +03:00
|
|
|
def skip_if_no_uuid(f):
|
|
|
|
"""Decorator to skip a test if uuid is not supported by Py/PG."""
|
|
|
|
def skip_if_no_uuid_(self):
|
|
|
|
try:
|
|
|
|
import uuid
|
|
|
|
except ImportError:
|
|
|
|
return self.skipTest("uuid not available in this Python version")
|
|
|
|
|
|
|
|
try:
|
|
|
|
cur = self.conn.cursor()
|
|
|
|
cur.execute("select typname from pg_type where typname = 'uuid'")
|
|
|
|
has = cur.fetchone()
|
|
|
|
finally:
|
|
|
|
self.conn.rollback()
|
|
|
|
|
|
|
|
if has:
|
|
|
|
return f(self)
|
|
|
|
else:
|
|
|
|
return self.skipTest("uuid type not available on the server")
|
|
|
|
|
|
|
|
return skip_if_no_uuid_
|
|
|
|
|
|
|
|
|
2011-01-10 02:44:58 +03:00
|
|
|
def skip_if_tpc_disabled(f):
|
|
|
|
"""Skip a test if the server has tpc support disabled."""
|
|
|
|
def skip_if_tpc_disabled_(self):
|
|
|
|
from psycopg2 import ProgrammingError
|
|
|
|
cnn = self.connect()
|
|
|
|
cur = cnn.cursor()
|
|
|
|
try:
|
|
|
|
cur.execute("SHOW max_prepared_transactions;")
|
|
|
|
except ProgrammingError:
|
|
|
|
return self.skipTest(
|
|
|
|
"server too old: two phase transactions not supported.")
|
|
|
|
else:
|
|
|
|
mtp = int(cur.fetchone()[0])
|
|
|
|
cnn.close()
|
|
|
|
|
|
|
|
if not mtp:
|
|
|
|
return self.skipTest(
|
|
|
|
"server not configured for two phase transactions. "
|
|
|
|
"set max_prepared_transactions to > 0 to run the test")
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
skip_if_tpc_disabled_.__name__ = f.__name__
|
|
|
|
return skip_if_tpc_disabled_
|
|
|
|
|
|
|
|
|
2011-02-19 03:05:43 +03:00
|
|
|
def skip_if_no_namedtuple(f):
|
|
|
|
def skip_if_no_namedtuple_(self):
|
|
|
|
try:
|
|
|
|
from collections import namedtuple
|
|
|
|
except ImportError:
|
|
|
|
return self.skipTest("collections.namedtuple not available")
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
skip_if_no_namedtuple_.__name__ = f.__name__
|
|
|
|
return skip_if_no_namedtuple_
|
|
|
|
|
|
|
|
|
2011-02-11 01:59:31 +03:00
|
|
|
def skip_if_no_iobase(f):
|
|
|
|
"""Skip a test if io.TextIOBase is not available."""
|
|
|
|
def skip_if_no_iobase_(self):
|
|
|
|
try:
|
|
|
|
from io import TextIOBase
|
|
|
|
except ImportError:
|
|
|
|
return self.skipTest("io.TextIOBase not found.")
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
return skip_if_no_iobase_
|
|
|
|
|
|
|
|
|
2011-02-15 20:11:07 +03:00
|
|
|
def skip_before_postgres(*ver):
|
|
|
|
"""Skip a test on PostgreSQL before a certain version."""
|
|
|
|
ver = ver + (0,) * (3 - len(ver))
|
|
|
|
def skip_before_postgres_(f):
|
|
|
|
def skip_before_postgres__(self):
|
|
|
|
if self.conn.server_version < int("%d%02d%02d" % ver):
|
|
|
|
return self.skipTest("skipped because PostgreSQL %s"
|
|
|
|
% self.conn.server_version)
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
return skip_before_postgres__
|
|
|
|
return skip_before_postgres_
|
|
|
|
|
|
|
|
def skip_after_postgres(*ver):
|
|
|
|
"""Skip a test on PostgreSQL after (including) a certain version."""
|
|
|
|
ver = ver + (0,) * (3 - len(ver))
|
|
|
|
def skip_after_postgres_(f):
|
|
|
|
def skip_after_postgres__(self):
|
|
|
|
if self.conn.server_version >= int("%d%02d%02d" % ver):
|
|
|
|
return self.skipTest("skipped because PostgreSQL %s"
|
|
|
|
% self.conn.server_version)
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
return skip_after_postgres__
|
|
|
|
return skip_after_postgres_
|
|
|
|
|
|
|
|
def skip_before_python(*ver):
|
|
|
|
"""Skip a test on Python before a certain version."""
|
|
|
|
def skip_before_python_(f):
|
|
|
|
def skip_before_python__(self):
|
|
|
|
if sys.version_info[:len(ver)] < ver:
|
|
|
|
return self.skipTest("skipped because Python %s"
|
|
|
|
% ".".join(map(str, sys.version_info[:len(ver)])))
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
return skip_before_python__
|
|
|
|
return skip_before_python_
|
|
|
|
|
|
|
|
def skip_from_python(*ver):
|
|
|
|
"""Skip a test on Python after (including) a certain version."""
|
|
|
|
def skip_from_python_(f):
|
|
|
|
def skip_from_python__(self):
|
|
|
|
if sys.version_info[:len(ver)] >= ver:
|
|
|
|
return self.skipTest("skipped because Python %s"
|
|
|
|
% ".".join(map(str, sys.version_info[:len(ver)])))
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
return skip_from_python__
|
|
|
|
return skip_from_python_
|
2011-02-11 01:59:31 +03:00
|
|
|
|
2013-03-16 15:56:38 +04:00
|
|
|
def skip_if_no_superuser(f):
|
|
|
|
"""Skip a test if the database user running the test is not a superuser"""
|
|
|
|
def skip_if_no_superuser_(self):
|
|
|
|
from psycopg2 import ProgrammingError
|
|
|
|
try:
|
|
|
|
return f(self)
|
|
|
|
except ProgrammingError, e:
|
|
|
|
import psycopg2.errorcodes
|
|
|
|
if e.pgcode == psycopg2.errorcodes.INSUFFICIENT_PRIVILEGE:
|
|
|
|
self.skipTest("skipped because not superuser")
|
|
|
|
else:
|
|
|
|
raise
|
|
|
|
|
|
|
|
return skip_if_no_superuser_
|
|
|
|
|
2013-03-18 14:06:07 +04:00
|
|
|
def skip_copy_if_green(f):
|
|
|
|
def skip_copy_if_green_(self):
|
|
|
|
from testconfig import green
|
|
|
|
if green:
|
|
|
|
return self.skipTest("copy in async mode currently not supported")
|
|
|
|
else:
|
|
|
|
return f(self)
|
|
|
|
|
|
|
|
return skip_copy_if_green_
|
|
|
|
|
2011-02-11 01:59:31 +03:00
|
|
|
|
|
|
|
def script_to_py3(script):
|
|
|
|
"""Convert a script to Python3 syntax if required."""
|
|
|
|
if sys.version_info[0] < 3:
|
|
|
|
return script
|
|
|
|
|
|
|
|
import tempfile
|
2011-02-12 03:40:23 +03:00
|
|
|
f = tempfile.NamedTemporaryFile(suffix=".py", delete=False)
|
2011-02-11 01:59:31 +03:00
|
|
|
f.write(script.encode())
|
|
|
|
f.flush()
|
2011-02-12 03:40:23 +03:00
|
|
|
filename = f.name
|
|
|
|
f.close()
|
2011-02-11 01:59:31 +03:00
|
|
|
|
|
|
|
# 2to3 is way too chatty
|
|
|
|
import logging
|
|
|
|
logging.basicConfig(filename=os.devnull)
|
|
|
|
|
|
|
|
from lib2to3.main import main
|
2011-02-12 03:40:23 +03:00
|
|
|
if main("lib2to3.fixes", ['--no-diffs', '-w', '-n', filename]):
|
2011-02-11 01:59:31 +03:00
|
|
|
raise Exception('py3 conversion failed')
|
|
|
|
|
2011-02-12 03:40:23 +03:00
|
|
|
f2 = open(filename)
|
2011-02-11 01:59:31 +03:00
|
|
|
try:
|
|
|
|
return f2.read()
|
|
|
|
finally:
|
|
|
|
f2.close()
|
2011-02-12 03:40:23 +03:00
|
|
|
os.remove(filename)
|
2011-02-11 01:59:31 +03:00
|
|
|
|