##// END OF EJS Templates
flagutil: make the error class used by the mixin configurable...
marmoute -
r43142:7907008a default
parent child Browse files
Show More
@@ -1,151 +1,153
1 1 # flagutils.py - code to deal with revlog flags and their processors
2 2 #
3 3 # Copyright 2016 Remi Chaintron <remi@fb.com>
4 4 # Copyright 2016-2019 Pierre-Yves David <pierre-yves.david@ens-lyon.org>
5 5 #
6 6 # This software may be used and distributed according to the terms of the
7 7 # GNU General Public License version 2 or any later version.
8 8
9 9 from __future__ import absolute_import
10 10
11 11 from ..i18n import _
12 12
13 13 from .constants import (
14 14 REVIDX_DEFAULT_FLAGS,
15 15 REVIDX_ELLIPSIS,
16 16 REVIDX_EXTSTORED,
17 17 REVIDX_FLAGS_ORDER,
18 18 REVIDX_ISCENSORED,
19 19 REVIDX_RAWTEXT_CHANGING_FLAGS,
20 20 )
21 21
22 22 from .. import (
23 23 error,
24 24 util
25 25 )
26 26
27 27 # blanked usage of all the name to prevent pyflakes constraints
28 28 # We need these name available in the module for extensions.
29 29 REVIDX_ISCENSORED
30 30 REVIDX_ELLIPSIS
31 31 REVIDX_EXTSTORED
32 32 REVIDX_DEFAULT_FLAGS
33 33 REVIDX_FLAGS_ORDER
34 34 REVIDX_RAWTEXT_CHANGING_FLAGS
35 35
36 36 REVIDX_KNOWN_FLAGS = util.bitsfrom(REVIDX_FLAGS_ORDER)
37 37
38 38 # Store flag processors (cf. 'addflagprocessor()' to register)
39 39 flagprocessors = {
40 40 REVIDX_ISCENSORED: None,
41 41 }
42 42
43 43 def addflagprocessor(flag, processor):
44 44 """Register a flag processor on a revision data flag.
45 45
46 46 Invariant:
47 47 - Flags need to be defined in REVIDX_KNOWN_FLAGS and REVIDX_FLAGS_ORDER,
48 48 and REVIDX_RAWTEXT_CHANGING_FLAGS if they can alter rawtext.
49 49 - Only one flag processor can be registered on a specific flag.
50 50 - flagprocessors must be 3-tuples of functions (read, write, raw) with the
51 51 following signatures:
52 52 - (read) f(self, rawtext) -> text, bool
53 53 - (write) f(self, text) -> rawtext, bool
54 54 - (raw) f(self, rawtext) -> bool
55 55 "text" is presented to the user. "rawtext" is stored in revlog data, not
56 56 directly visible to the user.
57 57 The boolean returned by these transforms is used to determine whether
58 58 the returned text can be used for hash integrity checking. For example,
59 59 if "write" returns False, then "text" is used to generate hash. If
60 60 "write" returns True, that basically means "rawtext" returned by "write"
61 61 should be used to generate hash. Usually, "write" and "read" return
62 62 different booleans. And "raw" returns a same boolean as "write".
63 63
64 64 Note: The 'raw' transform is used for changegroup generation and in some
65 65 debug commands. In this case the transform only indicates whether the
66 66 contents can be used for hash integrity checks.
67 67 """
68 68 insertflagprocessor(flag, processor, flagprocessors)
69 69
70 70 def insertflagprocessor(flag, processor, flagprocessors):
71 71 if not flag & REVIDX_KNOWN_FLAGS:
72 72 msg = _("cannot register processor on unknown flag '%#x'.") % (flag)
73 73 raise error.ProgrammingError(msg)
74 74 if flag not in REVIDX_FLAGS_ORDER:
75 75 msg = _("flag '%#x' undefined in REVIDX_FLAGS_ORDER.") % (flag)
76 76 raise error.ProgrammingError(msg)
77 77 if flag in flagprocessors:
78 78 msg = _("cannot register multiple processors on flag '%#x'.") % (flag)
79 79 raise error.Abort(msg)
80 80 flagprocessors[flag] = processor
81 81
82 82 class flagprocessorsmixin(object):
83 83 """basic mixin to support revlog flag processing
84 84
85 85 Make sure the `_flagprocessors` attribute is set at ``__init__`` time.
86 86
87 87 See the documentation of the ``_processflags`` method for details.
88 88 """
89 89
90 _flagserrorclass = error.RevlogError
91
90 92 def _processflags(self, text, flags, operation, raw=False):
91 93 """Inspect revision data flags and applies transforms defined by
92 94 registered flag processors.
93 95
94 96 ``text`` - the revision data to process
95 97 ``flags`` - the revision flags
96 98 ``operation`` - the operation being performed (read or write)
97 99 ``raw`` - an optional argument describing if the raw transform should be
98 100 applied.
99 101
100 102 This method processes the flags in the order (or reverse order if
101 103 ``operation`` is 'write') defined by REVIDX_FLAGS_ORDER, applying the
102 104 flag processors registered for present flags. The order of flags defined
103 105 in REVIDX_FLAGS_ORDER needs to be stable to allow non-commutativity.
104 106
105 107 Returns a 2-tuple of ``(text, validatehash)`` where ``text`` is the
106 108 processed text and ``validatehash`` is a bool indicating whether the
107 109 returned text should be checked for hash integrity.
108 110
109 111 Note: If the ``raw`` argument is set, it has precedence over the
110 112 operation and will only update the value of ``validatehash``.
111 113 """
112 114 # fast path: no flag processors will run
113 115 if flags == 0:
114 116 return text, True
115 117 if not operation in ('read', 'write'):
116 118 raise error.ProgrammingError(_("invalid '%s' operation") %
117 119 operation)
118 120 # Check all flags are known.
119 121 if flags & ~REVIDX_KNOWN_FLAGS:
120 raise error.RevlogError(_("incompatible revision flag '%#x'") %
122 raise self._flagserrorclass(_("incompatible revision flag '%#x'") %
121 123 (flags & ~REVIDX_KNOWN_FLAGS))
122 124 validatehash = True
123 125 # Depending on the operation (read or write), the order might be
124 126 # reversed due to non-commutative transforms.
125 127 orderedflags = REVIDX_FLAGS_ORDER
126 128 if operation == 'write':
127 129 orderedflags = reversed(orderedflags)
128 130
129 131 for flag in orderedflags:
130 132 # If a flagprocessor has been registered for a known flag, apply the
131 133 # related operation transform and update result tuple.
132 134 if flag & flags:
133 135 vhash = True
134 136
135 137 if flag not in self._flagprocessors:
136 138 message = _("missing processor for flag '%#x'") % (flag)
137 raise error.RevlogError(message)
139 raise self._flagserrorclass(message)
138 140
139 141 processor = self._flagprocessors[flag]
140 142 if processor is not None:
141 143 readtransform, writetransform, rawtransform = processor
142 144
143 145 if raw:
144 146 vhash = rawtransform(self, text)
145 147 elif operation == 'read':
146 148 text, vhash = readtransform(self, text)
147 149 else: # write operation
148 150 text, vhash = writetransform(self, text)
149 151 validatehash = validatehash and vhash
150 152
151 153 return text, validatehash
General Comments 0
You need to be logged in to leave comments. Login now