This file is indexed.

/usr/lib/python2.7/dist-packages/transaction/tests/test_register_compat.py is in python-transaction 1.4.3-3.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
##############################################################################
#
# Copyright (c) 2004 Zope Foundation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.1 (ZPL).  A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE.
#
##############################################################################
"""Test backwards compatibility for resource managers using register().

The transaction package supports several different APIs for resource
managers.  The original ZODB3 API was implemented by ZODB.Connection.
The Connection passed persistent objects to a Transaction's register()
method.  It's possible that third-party code also used this API, hence
these tests that the code that adapts the old interface to the current
API works.

These tests use a TestConnection object that implements the old API.
They check that the right methods are called and in roughly the right
order.
"""
import unittest


class BBBTests(unittest.TestCase):

    def setUp(self):
        from transaction import abort
        abort()
    tearDown = setUp

    def test_basic_commit(self):
        import transaction
        cn = TestConnection()
        cn.register(Object())
        cn.register(Object())
        cn.register(Object())
        transaction.commit()
        self.assertEqual(len(cn.committed), 3)
        self.assertEqual(len(cn.aborted), 0)
        self.assertEqual(cn.calls, ['begin', 'vote', 'finish'])

    def test_basic_abort(self):
        # If the application calls abort(), then the transaction never gets
        # into the two-phase commit.  It just aborts each object.
        import transaction
        cn = TestConnection()
        cn.register(Object())
        cn.register(Object())
        cn.register(Object())
        transaction.abort()
        self.assertEqual(len(cn.committed), 0)
        self.assertEqual(len(cn.aborted), 3)
        self.assertEqual(cn.calls, [])

    def test_tpc_error(self):
        # The tricky part of the implementation is recovering from an error
        # that occurs during the two-phase commit.  We override the commit()
        # and abort() methods of Object to cause errors during commit.

        # Note that the implementation uses lists internally, so that objects
        # are committed in the order they are registered.  (In the presence
        # of multiple resource managers, objects from a single resource
        # manager are committed in order.  I'm not sure if this is an
        # accident of the implementation or a feature that should be
        # supported by any implementation.)

        # The order of resource managers depends on sortKey().
        import transaction
        cn = TestConnection()
        cn.register(Object())
        cn.register(CommitError())
        cn.register(Object())
        self.assertRaises(RuntimeError, transaction.commit)
        self.assertEqual(len(cn.committed), 1)
        self.assertEqual(len(cn.aborted), 3)


class Object(object):

    def commit(self):
        pass

    def abort(self):
        pass


class CommitError(Object):

    def commit(self):
        raise RuntimeError("commit")


class AbortError(Object):

    def abort(self):
        raise RuntimeError("abort")


class BothError(CommitError, AbortError):
    pass


class TestConnection(object):

    def __init__(self):
        self.committed = []
        self.aborted = []
        self.calls = []

    def register(self, obj):
        import transaction
        obj._p_jar = self
        transaction.get().register(obj)

    def sortKey(self):
        return str(id(self))

    def tpc_begin(self, txn):
        self.calls.append("begin")

    def tpc_vote(self, txn):
        self.calls.append("vote")

    def tpc_finish(self, txn):
        self.calls.append("finish")

    def tpc_abort(self, txn):
        self.calls.append("abort")

    def commit(self, obj, txn):
        obj.commit()
        self.committed.append(obj)

    def abort(self, obj, txn):
        obj.abort()
        self.aborted.append(obj)

def test_suite():
    return unittest.TestSuite((
        unittest.makeSuite(BBBTests),
    ))