transaction.py
78 lines
| 2.2 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. | ||||
# | ||||
# Copyright 2005 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
import os | ||||
mpm@selenic.com
|
r421 | import util | ||
mpm@selenic.com
|
r0 | |||
class transaction: | ||||
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): | ||||
Bart Trojanowski
|
r1073 | raise AssertionError("journal already exists - run hg recover") | ||
mpm@selenic.com
|
r162 | |||
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 | |||
def add(self, file, offset): | ||||
mpm@selenic.com
|
r42 | if file in self.map: return | ||
mpm@selenic.com
|
r0 | self.entries.append((file, offset)) | ||
mpm@selenic.com
|
r42 | self.map[file] = 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() | ||||
def close(self): | ||||
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 | ||||
mpm@selenic.com
|
r582 | self.report("transaction abort!\n") | ||
mpm@selenic.com
|
r0 | |||
for f, o in self.entries: | ||||
mpm@selenic.com
|
r108 | try: | ||
self.opener(f, "a").truncate(o) | ||||
except: | ||||
mpm@selenic.com
|
r582 | self.report("failed to truncate %s\n" % f) | ||
mpm@selenic.com
|
r0 | |||
self.entries = [] | ||||
mpm@selenic.com
|
r582 | self.report("rollback completed\n") | ||
mpm@selenic.com
|
r515 | |||
mpm@selenic.com
|
r162 | def rollback(opener, file): | ||
for l in open(file).readlines(): | ||||
f, o = l.split('\0') | ||||
opener(f, "a").truncate(int(o)) | ||||
os.unlink(file) | ||||
mpm@selenic.com
|
r0 | |||