/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar

« back to all changes in this revision

Viewing changes to bzrlib/cleanup.py

  • Committer: Jelmer Vernooij
  • Date: 2011-12-19 10:58:39 UTC
  • mfrom: (6383 +trunk)
  • mto: This revision was merged to the branch mainline in revision 6386.
  • Revision ID: jelmer@canonical.com-20111219105839-uji05ck4rkm1mj4j
Merge bzr.dev.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2009, 2010 Canonical Ltd
 
2
#
 
3
# This program is free software; you can redistribute it and/or modify
 
4
# it under the terms of the GNU General Public License as published by
 
5
# the Free Software Foundation; either version 2 of the License, or
 
6
# (at your option) any later version.
 
7
#
 
8
# This program is distributed in the hope that it will be useful,
 
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
11
# GNU General Public License for more details.
 
12
#
 
13
# You should have received a copy of the GNU General Public License
 
14
# along with this program; if not, write to the Free Software
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
16
 
 
17
from __future__ import absolute_import
 
18
 
 
19
"""Helpers for managing cleanup functions and the errors they might raise.
 
20
 
 
21
The usual way to run cleanup code in Python is::
 
22
 
 
23
    try:
 
24
        do_something()
 
25
    finally:
 
26
        cleanup_something()
 
27
 
 
28
However if both `do_something` and `cleanup_something` raise an exception
 
29
Python will forget the original exception and propagate the one from
 
30
cleanup_something.  Unfortunately, this is almost always much less useful than
 
31
the original exception.
 
32
 
 
33
If you want to be certain that the first, and only the first, error is raised,
 
34
then use::
 
35
 
 
36
    operation = OperationWithCleanups(do_something)
 
37
    operation.add_cleanup(cleanup_something)
 
38
    operation.run_simple()
 
39
 
 
40
This is more inconvenient (because you need to make every try block a
 
41
function), but will ensure that the first error encountered is the one raised,
 
42
while also ensuring all cleanups are run.  See OperationWithCleanups for more
 
43
details.
 
44
"""
 
45
 
 
46
 
 
47
from collections import deque
 
48
import sys
 
49
from bzrlib import (
 
50
    debug,
 
51
    trace,
 
52
    )
 
53
 
 
54
def _log_cleanup_error(exc):
 
55
    trace.mutter('Cleanup failed:')
 
56
    trace.log_exception_quietly()
 
57
    if 'cleanup' in debug.debug_flags:
 
58
        trace.warning('bzr: warning: Cleanup failed: %s', exc)
 
59
 
 
60
 
 
61
def _run_cleanup(func, *args, **kwargs):
 
62
    """Run func(*args, **kwargs), logging but not propagating any error it
 
63
    raises.
 
64
 
 
65
    :returns: True if func raised no errors, else False.
 
66
    """
 
67
    try:
 
68
        func(*args, **kwargs)
 
69
    except KeyboardInterrupt:
 
70
        raise
 
71
    except Exception, exc:
 
72
        _log_cleanup_error(exc)
 
73
        return False
 
74
    return True
 
75
 
 
76
 
 
77
def _run_cleanups(funcs):
 
78
    """Run a series of cleanup functions."""
 
79
    for func, args, kwargs in funcs:
 
80
        _run_cleanup(func, *args, **kwargs)
 
81
 
 
82
 
 
83
class ObjectWithCleanups(object):
 
84
    """A mixin for objects that hold a cleanup list.
 
85
 
 
86
    Subclass or client code can call add_cleanup and then later `cleanup_now`.
 
87
    """
 
88
    def __init__(self):
 
89
        self.cleanups = deque()
 
90
 
 
91
    def add_cleanup(self, cleanup_func, *args, **kwargs):
 
92
        """Add a cleanup to run.
 
93
 
 
94
        Cleanups may be added at any time.  
 
95
        Cleanups will be executed in LIFO order.
 
96
        """
 
97
        self.cleanups.appendleft((cleanup_func, args, kwargs))
 
98
 
 
99
    def cleanup_now(self):
 
100
        _run_cleanups(self.cleanups)
 
101
        self.cleanups.clear()
 
102
 
 
103
 
 
104
class OperationWithCleanups(ObjectWithCleanups):
 
105
    """A way to run some code with a dynamic cleanup list.
 
106
 
 
107
    This provides a way to add cleanups while the function-with-cleanups is
 
108
    running.
 
109
 
 
110
    Typical use::
 
111
 
 
112
        operation = OperationWithCleanups(some_func)
 
113
        operation.run(args...)
 
114
 
 
115
    where `some_func` is::
 
116
 
 
117
        def some_func(operation, args, ...):
 
118
            do_something()
 
119
            operation.add_cleanup(something)
 
120
            # etc
 
121
 
 
122
    Note that the first argument passed to `some_func` will be the
 
123
    OperationWithCleanups object.  To invoke `some_func` without that, use
 
124
    `run_simple` instead of `run`.
 
125
    """
 
126
 
 
127
    def __init__(self, func):
 
128
        super(OperationWithCleanups, self).__init__()
 
129
        self.func = func
 
130
 
 
131
    def run(self, *args, **kwargs):
 
132
        return _do_with_cleanups(
 
133
            self.cleanups, self.func, self, *args, **kwargs)
 
134
 
 
135
    def run_simple(self, *args, **kwargs):
 
136
        return _do_with_cleanups(
 
137
            self.cleanups, self.func, *args, **kwargs)
 
138
 
 
139
 
 
140
def _do_with_cleanups(cleanup_funcs, func, *args, **kwargs):
 
141
    """Run `func`, then call all the cleanup_funcs.
 
142
 
 
143
    All the cleanup_funcs are guaranteed to be run.  The first exception raised
 
144
    by func or any of the cleanup_funcs is the one that will be propagted by
 
145
    this function (subsequent errors are caught and logged).
 
146
 
 
147
    Conceptually similar to::
 
148
 
 
149
        try:
 
150
            return func(*args, **kwargs)
 
151
        finally:
 
152
            for cleanup, cargs, ckwargs in cleanup_funcs:
 
153
                cleanup(*cargs, **ckwargs)
 
154
 
 
155
    It avoids several problems with using try/finally directly:
 
156
     * an exception from func will not be obscured by a subsequent exception
 
157
       from a cleanup.
 
158
     * an exception from a cleanup will not prevent other cleanups from
 
159
       running (but the first exception encountered is still the one
 
160
       propagated).
 
161
 
 
162
    Unike `_run_cleanup`, `_do_with_cleanups` can propagate an exception from a
 
163
    cleanup, but only if there is no exception from func.
 
164
    """
 
165
    # As correct as Python 2.4 allows.
 
166
    try:
 
167
        result = func(*args, **kwargs)
 
168
    except:
 
169
        # We have an exception from func already, so suppress cleanup errors.
 
170
        _run_cleanups(cleanup_funcs)
 
171
        raise
 
172
    else:
 
173
        # No exception from func, so allow the first exception from
 
174
        # cleanup_funcs to propagate if one occurs (but only after running all
 
175
        # of them).
 
176
        exc_info = None
 
177
        for cleanup, c_args, c_kwargs in cleanup_funcs:
 
178
            # XXX: Hmm, if KeyboardInterrupt arrives at exactly this line, we
 
179
            # won't run all cleanups... perhaps we should temporarily install a
 
180
            # SIGINT handler?
 
181
            if exc_info is None:
 
182
                try:
 
183
                    cleanup(*c_args, **c_kwargs)
 
184
                except:
 
185
                    # This is the first cleanup to fail, so remember its
 
186
                    # details.
 
187
                    exc_info = sys.exc_info()
 
188
            else:
 
189
                # We already have an exception to propagate, so log any errors
 
190
                # but don't propagate them.
 
191
                _run_cleanup(cleanup, *c_args, **kwargs)
 
192
        if exc_info is not None:
 
193
            try:
 
194
                raise exc_info[0], exc_info[1], exc_info[2]
 
195
            finally:
 
196
                del exc_info
 
197
        # No error, so we can return the result
 
198
        return result
 
199
 
 
200