Edgewall Software
Modify

Opened 16 years ago

Closed 16 years ago

Last modified 16 years ago

#8518 closed defect (fixed)

Corner case failures when running the functional tests with alternate db backend

Reported by: Christian Boos Owned by: Christian Boos
Priority: normal Milestone: 0.11.6
Component: database backend Version: none
Severity: normal Keywords: functional
Cc: Branch:
Release Notes:
API Changes:
Internal Changes:

Description (last modified by Christian Boos)

While testing #8502, I've noticed that when running the functional tests for the first time against a non-existing schema, some tests would fail, due to a bootstrap problem.

The errors were:

$ PYTHONPATH=. python trac/db/tests/functional.py
...
Traceback (most recent call last):
  File "trac/db/tests/functional.py", line 27, in <module>
    unittest.main(defaultTest='functionalSuite')
  File "C:\Dev\Python261\lib\unittest.py", line 817, in __init__
    self.runTests()
  File "C:\Dev\Python261\lib\unittest.py", line 861, in runTests
    result = testRunner.run(self.test)
  File "C:\Dev\Python261\lib\unittest.py", line 753, in run
    test(result)
  File "C:\Dev\Python261\lib\unittest.py", line 464, in __call__
    return self.run(*args, **kwds)
  File "C:\Dev\Python261\lib\unittest.py", line 460, in run
    test(result)
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\test.py", line 137, in __call__
    return self.run(*args, **kwds)
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\test.py", line 124, in run
    self.setUp()
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\tests\functional\__init__.py", line 126, in setUp
    self._testenv = self.env_class(dirname, port, baseurl)
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\tests\functional\testenv.py", line 57, in __init__
    self.destroy()
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\tests\functional\testenv.py", line 67, in destroy
    env.destroy_db()
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\test.py", line 350, in destroy_db
    db = self.get_db_cnx()
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\test.py", line 292, in get_db_cnx
    self.reset_db() # make sure we get rid of garbage from previous run
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\test.py", line 343, in reset_db
    ('database_version', str(db_default.db_version)))
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\db\util.py", line 59, in execute
    return self.cursor.execute(sql_escape_percent(sql), args)
  File "C:\Workspace\src\trac\repos\0.11-stable\trac\db\util.py", line 59, in execute
    return self.cursor.execute(sql_escape_percent(sql), args)
psycopg2.ProgrammingError: relation "system" does not exist

The following fixed it for me:

  • trac/test.py

    diff --git a/trac/test.py b/trac/test.py
    a b  
    276276                return True
    277277        return False
    278278
    279     def get_db_cnx(self):
     279    def get_db_cnx(self, destroying=False):
    280280        if self.db:
    281281            return self.db # in-memory SQLite
    282282
     
    289289        if not dbenv:
    290290            dbenv = EnvironmentStub.dbenv = EnvironmentStub()
    291291            dbenv.config.set('trac', 'database', self.dburi)
    292             self.reset_db() # make sure we get rid of garbage from previous run
     292            if not destroying:
     293                self.reset_db() # make sure we get rid of previous garbage
    293294        return DatabaseManager(dbenv).get_connection()
    294295
    295296    def reset_db(self, default_data=None):
     
    347348        if not (scheme and db_prop):
    348349            scheme, db_prop = _parse_db_str(self.dburi)
    349350
    350         db = self.get_db_cnx()
     351        db = self.get_db_cnx(destroying=True)
    351352        cursor = db.cursor()
    352353        try:
    353354            if scheme == 'postgres' and db.schema:

Remy also had (similar?) problems which were not fixed by the above patch, so we still need to investigate further.

For him, the first errors were:

======================================================================
ERROR: test_delete (trac.tests.attachment.AttachmentTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "/home/joe/src/trac/0.11-stable/trac/tests/attachment.py", line 101, in test_delete
    attachment1.insert('foo.txt', tempfile.TemporaryFile(), 0)
  File "/home/joe/src/trac/0.11-stable/trac/attachment.py", line 202, in insert
    db = self.env.get_db_cnx()
  File "/home/joe/src/trac/0.11-stable/trac/test.py", line 293, in get_db_cnx
    self.reset_db() # make sure we get rid of previous garbage
  File "/home/joe/src/trac/0.11-stable/trac/test.py", line 344, in reset_db
    ('database_version', str(db_default.db_version)))
  File "/home/joe/src/trac/0.11-stable/trac/db/util.py", line 59, in execute
    return self.cursor.execute(sql_escape_percent(sql), args)
  File "/home/joe/src/trac/0.11-stable/trac/db/util.py", line 59, in execute
    return self.cursor.execute(sql_escape_percent(sql), args)
ProgrammingError: relation "system" does not exist


======================================================================
ERROR: test_delete (trac.tests.attachment.AttachmentTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "/home/joe/src/trac/0.11-stable/trac/tests/attachment.py", line 43, in tearDown
    self.env.reset_db()
  File "/home/joe/src/trac/0.11-stable/trac/test.py", line 331, in reset_db
    DatabaseManager(EnvironmentStub.dbenv).init_db()
  File "/home/joe/src/trac/0.11-stable/trac/db/api.py", line 81, in init_db
    connector.init_db(**args)
  File "/home/joe/src/trac/0.11-stable/trac/db/postgres_backend.py", line 76, in init_db
    cursor.execute('CREATE SCHEMA "%s"' % cnx.schema)
  File "/home/joe/src/trac/0.11-stable/trac/db/util.py", line 60, in execute
    return self.cursor.execute(sql)
ProgrammingError: schema "tractest-pat-011" already exists


======================================================================
ERROR: test_delete_file_gone (trac.tests.attachment.AttachmentTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "/home/joe/src/trac/0.11-stable/trac/tests/attachment.py", line 123, in test_delete_file_gone
    attachment.insert('foo.txt', tempfile.TemporaryFile(), 0)
  File "/home/joe/src/trac/0.11-stable/trac/attachment.py", line 234, in insert
    self.author, self.ipnr))
  File "/home/joe/src/trac/0.11-stable/trac/db/util.py", line 59, in execute
    return self.cursor.execute(sql_escape_percent(sql), args)
  File "/home/joe/src/trac/0.11-stable/trac/db/util.py", line 59, in execute
    return self.cursor.execute(sql_escape_percent(sql), args)
ProgrammingError: relation "attachment" does not exist

… whether the above patch was applied or not.

This was obtained while running the unit tests.

Attachments (0)

Change History (9)

comment:1 by Christian Boos, 16 years ago

I've applied the above patch in r8381 as it fixes the issue for me. Remy could you please try again at that revision?

in reply to:  1 comment:2 by Remy Blank, 16 years ago

Replying to cboos:

Remy could you please try again at that revision?

No, still no luck. Just in case, here's the script I use to run the tests (the DB is on another machine):

#!/bin/bash

rm -rf testenv
export TRAC_TEST_DB_URI=postgres://trac:password@mac.vathome:5432/trac?schema=tractest-pat-011
PYTHONPATH=. python trac/test.py

comment:3 by Christian Boos, 16 years ago

Ah yes, forgot to remove testenv ;-)

I have the error as well now.

comment:4 by Christian Boos, 16 years ago

Description: modified (diff)

Well, removing testenv or not was not the problem, I was just wrong and r8381 doesn't fix the unit-tests for me either. What it fixes arethe functional tests only.

Description updated accordingly.

comment:5 by Christian Boos, 16 years ago

Hm, I have a fix now, but I don't understand it ;-)

  • trac/test.py

     
    328328
    329329            if not tables:
    330330                del db
    331                 DatabaseManager(EnvironmentStub.dbenv).init_db()
     331                dm = DatabaseManager(EnvironmentStub.dbenv)
     332                dm.init_db()
     333                # we need to make sure the next get_db_cnx() will re-create
     334                # a new connection aware of the new data model - see #8518.
     335                dm.shutdown()
    332336
    333337        db = self.get_db_cnx()
    334338        cursor = db.cursor()

The idea I had here was that in reset_db(), we should make sure that after init_db(), the get_db_cnx() call that follows doesn't give us back a connection from the pool that existed before the tables were created, as it appears that for PosgreSQL at least, this old connection wouldn't see the newly created tables. So we force a flush of the pool using shutdown().

The problem is, I don't see any reference to that behavior in the PostgreSQL manual or elsewhere on the web, and interactive testing with psql show that this shouldn't be a problem (well, unless psql do some tricks that psycopg2 doesn't).

comment:6 by Remy Blank, 16 years ago

The patch works fine here. Maybe this is again an issue due to not rolling back connections before releasing them to the pool?

comment:7 by Christian Boos, 16 years ago

We normally do a rollback before putting connections back into the _pool LRU list.

I've just verified and that rollback is indeed called on the connection, just after the del db before init_db(). So it must be something else.

comment:8 by Christian Boos, 16 years ago

Resolution: fixed
Status: newclosed

Ok, for now I've applied the fix from comment:5, in r8416.

One might revisit this issue one day in order to understand it better, but enough for now ;-)

comment:9 by Christian Boos, 16 years ago

Owner: set to Christian Boos

Modify Ticket

Change Properties
Set your email in Preferences
Action
as closed The owner will remain Christian Boos.
The resolution will be deleted. Next status will be 'reopened'.
to The owner will be changed from Christian Boos to the specified user.

Add Comment


E-mail address and name can be saved in the Preferences .
 
Note: See TracTickets for help on using tickets.