transaction.py
107 lines
| 3.0 KiB
| text/x-python
|
PythonLexer
/ mercurial / transaction.py
mpm@selenic.com
|
r0 | # transaction.py - simple journalling scheme for mercurial | ||
# | ||||
# This transaction scheme is intended to gracefully handle program | ||||
# errors and interruptions. More serious failures like system crashes | ||||
# can be recovered with an fsck-like tool. As the whole repository is | ||||
# effectively log-structured, this should amount to simply truncating | ||||
# anything that isn't referenced in the changelog. | ||||
# | ||||
Vadim Gelfer
|
r2859 | # Copyright 2005, 2006 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r0 | # | ||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
Vadim Gelfer
|
r2470 | from demandload import demandload | ||
Benoit Boissinot
|
r1400 | from i18n import gettext as _ | ||
Vadim Gelfer
|
r2470 | demandload(globals(), 'os') | ||
mpm@selenic.com
|
r0 | |||
Eric Hopper
|
r1559 | class transaction(object): | ||
benoit.boissinot@ens-lyon.fr
|
r1062 | def __init__(self, report, opener, journal, after=None): | ||
mpm@selenic.com
|
r162 | self.journal = None | ||
# abort here if the journal already exists | ||||
if os.path.exists(journal): | ||||
Benoit Boissinot
|
r1402 | raise AssertionError(_("journal already exists - run hg recover")) | ||
mpm@selenic.com
|
r162 | |||
mason@suse.com
|
r1806 | self.count = 1 | ||
mpm@selenic.com
|
r582 | self.report = report | ||
mpm@selenic.com
|
r0 | self.opener = opener | ||
mpm@selenic.com
|
r95 | self.after = after | ||
mpm@selenic.com
|
r0 | self.entries = [] | ||
mpm@selenic.com
|
r42 | self.map = {} | ||
mpm@selenic.com
|
r0 | self.journal = journal | ||
self.file = open(self.journal, "w") | ||||
def __del__(self): | ||||
mpm@selenic.com
|
r558 | if self.journal: | ||
if self.entries: self.abort() | ||||
self.file.close() | ||||
try: os.unlink(self.journal) | ||||
except: pass | ||||
mpm@selenic.com
|
r0 | |||
Chris Mason
|
r2084 | def add(self, file, offset, data=None): | ||
mpm@selenic.com
|
r42 | if file in self.map: return | ||
Chris Mason
|
r2084 | self.entries.append((file, offset, data)) | ||
self.map[file] = len(self.entries) - 1 | ||||
mpm@selenic.com
|
r0 | # add enough data to the journal to do the truncate | ||
self.file.write("%s\0%d\n" % (file, offset)) | ||||
self.file.flush() | ||||
Chris Mason
|
r2084 | def find(self, file): | ||
if file in self.map: | ||||
return self.entries[self.map[file]] | ||||
return None | ||||
def replace(self, file, offset, data=None): | ||||
if file not in self.map: | ||||
raise KeyError(file) | ||||
index = self.map[file] | ||||
self.entries[index] = (file, offset, data) | ||||
self.file.write("%s\0%d\n" % (file, offset)) | ||||
self.file.flush() | ||||
mason@suse.com
|
r1806 | def nest(self): | ||
self.count += 1 | ||||
return self | ||||
def running(self): | ||||
return self.count > 0 | ||||
mpm@selenic.com
|
r0 | def close(self): | ||
mason@suse.com
|
r1806 | self.count -= 1 | ||
if self.count != 0: | ||||
return | ||||
mpm@selenic.com
|
r0 | self.file.close() | ||
self.entries = [] | ||||
mpm@selenic.com
|
r95 | if self.after: | ||
mpm@selenic.com
|
r785 | self.after() | ||
mpm@selenic.com
|
r95 | else: | ||
os.unlink(self.journal) | ||||
mpm@selenic.com
|
r573 | self.journal = None | ||
mpm@selenic.com
|
r0 | |||
def abort(self): | ||||
if not self.entries: return | ||||
Benoit Boissinot
|
r1402 | self.report(_("transaction abort!\n")) | ||
mpm@selenic.com
|
r0 | |||
Chris Mason
|
r2084 | for f, o, ignore in self.entries: | ||
mpm@selenic.com
|
r108 | try: | ||
self.opener(f, "a").truncate(o) | ||||
except: | ||||
Benoit Boissinot
|
r1402 | self.report(_("failed to truncate %s\n") % f) | ||
mpm@selenic.com
|
r0 | |||
self.entries = [] | ||||
Benoit Boissinot
|
r1402 | self.report(_("rollback completed\n")) | ||
mpm@selenic.com
|
r515 | |||
mpm@selenic.com
|
r162 | def rollback(opener, file): | ||
Chris Mason
|
r2084 | files = {} | ||
mpm@selenic.com
|
r162 | for l in open(file).readlines(): | ||
f, o = l.split('\0') | ||||
Chris Mason
|
r2084 | files[f] = o | ||
for f in files: | ||||
o = files[f] | ||||
mpm@selenic.com
|
r162 | opener(f, "a").truncate(int(o)) | ||
os.unlink(file) | ||||
mpm@selenic.com
|
r0 | |||