##// END OF EJS Templates
inline-changelog: fix a critical bug in write_pending that delete data...
inline-changelog: fix a critical bug in write_pending that delete data Since a93e52f0b6ff we no longer use inline-revlog for the changelog. The goal there was to solve the lack of testing for the two variants (inline vs split) and reduce the complexity of the interaction with "diverted-write" on the changelog level. However many existing repository still have inline-changelog and we automatically move them to normal revlog as soon as we have the chances. Unfortunately This conversion is buggy and can result in the destruction of the changelog.i if hook triggers the "write pending" mechanism. The bugs comes from the "revlog splitting" logic and the "write_pending" logic stepping over each other. Ironically the change in a93e52f0b6ff aims at no longer having this kind of problem. This changesets fix this issue and add associated tests. Fixing this reveal that the transaction hooks end up not seeing the pending transaction content, because the name is not right ("changelog.i.s.a" instead of "changelog.i.s") we fix this in the next changeset.

File last commit:

r49730:6000f5b2 default
r52530:3cf9e52f stable
Show More
sidedata.py
174 lines | 6.1 KiB | text/x-python | PythonLexer
sidedata: add a new module with basic documentation...
r43301 # sidedata.py - Logic around store extra data alongside revlog revisions
#
# Copyright 2019 Pierre-Yves David <pierre-yves.david@octobus.net)
#
# This software may be used and distributed according to the terms of the
# GNU General Public License version 2 or any later version.
"""core code for "sidedata" support
The "sidedata" are stored alongside the revision without actually being part of
its content and not affecting its hash. It's main use cases is to cache
important information related to a changesets.
The current implementation is experimental and subject to changes. Do not rely
on it in production.
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 Sidedata are stored in the revlog itself, thanks to a new version of the
revlog. The following format is currently used::
sidedata: add a new module with basic documentation...
r43301
initial header:
<number of sidedata; 2 bytes>
sidedata (repeated N times):
<sidedata-key; 2 bytes>
<sidedata-entry-length: 4 bytes>
<sidedata-content-sha1-digest: 20 bytes>
<sidedata-content; X bytes>
normal raw text:
<all bytes remaining in the rawtext>
Joerg Sonnenberger
comments: fix typos...
r46811 This is a simple and effective format. It should be enough to experiment with
sidedata: add a new module with basic documentation...
r43301 the concept.
"""
sidedata: add a function to read sidedata from revlog raw text...
r43302
Raphaël Gomès
sidedata: move sidedata-related utils to the dedicated module...
r47848 import collections
sidedata: add a function to read sidedata from revlog raw text...
r43302 import struct
Raphaël Gomès
sidedata: move sidedata-related utils to the dedicated module...
r47848 from .. import error, requirements as requirementsmod
from ..revlogutils import constants, flagutil
Augie Fackler
core: migrate uses of hashlib.sha1 to hashutil.sha1...
r44517 from ..utils import hashutil
sidedata: add a function to read sidedata from revlog raw text...
r43302
sidedata: test we can successfully write sidedata...
r43308 ## sidedata type constant
# reserve a block for testing purposes.
SD_TEST1 = 1
SD_TEST2 = 2
SD_TEST3 = 3
SD_TEST4 = 4
SD_TEST5 = 5
SD_TEST6 = 6
SD_TEST7 = 7
sidedatacopies: write copies information in sidedata when applicable...
r43412 # key to store copies related information
SD_P1COPIES = 8
SD_P2COPIES = 9
SD_FILESADDED = 10
SD_FILESREMOVED = 11
changing-files: rework the way we store changed files in side-data...
r46211 SD_FILES = 12
sidedatacopies: write copies information in sidedata when applicable...
r43412
sidedata: test we can successfully write sidedata...
r43308 # internal format constant
Augie Fackler
cleanup: remove pointless r-prefixes on single-quoted strings...
r43906 SIDEDATA_HEADER = struct.Struct('>H')
SIDEDATA_ENTRY = struct.Struct('>HL20s')
sidedata: add a function to read sidedata from revlog raw text...
r43302
Augie Fackler
formatting: blacken the codebase...
r43346
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 def serialize_sidedata(sidedata):
sidedata: add a function to write sidedata into a raw text...
r43303 sidedata = list(sidedata.items())
sidedata.sort()
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 buf = [SIDEDATA_HEADER.pack(len(sidedata))]
sidedata: add a function to write sidedata into a raw text...
r43303 for key, value in sidedata:
Augie Fackler
core: migrate uses of hashlib.sha1 to hashutil.sha1...
r44517 digest = hashutil.sha1(value).digest()
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 buf.append(SIDEDATA_ENTRY.pack(key, len(value), digest))
sidedata: add a function to write sidedata into a raw text...
r43303 for key, value in sidedata:
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 buf.append(value)
buf = b''.join(buf)
return buf
sidedata: add a function to write sidedata into a raw text...
r43303
Augie Fackler
formatting: blacken the codebase...
r43346
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 def deserialize_sidedata(blob):
sidedata: add a function to read sidedata from revlog raw text...
r43302 sidedata = {}
offset = 0
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 (nbentry,) = SIDEDATA_HEADER.unpack(blob[: SIDEDATA_HEADER.size])
sidedata: add a function to read sidedata from revlog raw text...
r43302 offset += SIDEDATA_HEADER.size
dataoffset = SIDEDATA_HEADER.size + (SIDEDATA_ENTRY.size * nbentry)
for i in range(nbentry):
nextoffset = offset + SIDEDATA_ENTRY.size
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 key, size, storeddigest = SIDEDATA_ENTRY.unpack(blob[offset:nextoffset])
sidedata: add a function to read sidedata from revlog raw text...
r43302 offset = nextoffset
# read the data associated with that entry
nextdataoffset = dataoffset + size
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 entrytext = bytes(blob[dataoffset:nextdataoffset])
Augie Fackler
core: migrate uses of hashlib.sha1 to hashutil.sha1...
r44517 readdigest = hashutil.sha1(entrytext).digest()
sidedata: add a function to read sidedata from revlog raw text...
r43302 if storeddigest != readdigest:
raise error.SidedataHashError(key, storeddigest, readdigest)
sidedata[key] = entrytext
dataoffset = nextdataoffset
Raphaël Gomès
sidedata: move to new sidedata storage in revlogv2...
r47443 return sidedata
Raphaël Gomès
sidedata: move sidedata-related utils to the dedicated module...
r47848
def get_sidedata_helpers(repo, remote_sd_categories, pull=False):
Raphaël Gomès
sidedata: move documentation about sidedata helpers to sidedata module...
r47849 """
Returns a dictionary mapping revlog types to tuples of
`(repo, computers, removers)`:
* `repo` is used as an argument for computers
* `computers` is a list of `(category, (keys, computer, flags)` that
compute the missing sidedata categories that were asked:
* `category` is the sidedata category
* `keys` are the sidedata keys to be affected
* `flags` is a bitmask (an integer) of flags to remove when
removing the category.
* `computer` is the function `(repo, store, rev, sidedata)` that
returns a tuple of
`(new sidedata dict, (flags to add, flags to remove))`.
For example, it will return `({}, (0, 1 << 15))` to return no
sidedata, with no flags to add and one flag to remove.
* `removers` will remove the keys corresponding to the categories
that are present, but not needed.
If both `computers` and `removers` are empty, sidedata will simply not
be transformed.
"""
Raphaël Gomès
sidedata: move sidedata-related utils to the dedicated module...
r47848 # Computers for computing sidedata on-the-fly
sd_computers = collections.defaultdict(list)
# Computers for categories to remove from sidedata
sd_removers = collections.defaultdict(list)
to_generate = remote_sd_categories - repo._wanted_sidedata
to_remove = repo._wanted_sidedata - remote_sd_categories
if pull:
to_generate, to_remove = to_remove, to_generate
for revlog_kind, computers in repo._sidedata_computers.items():
for category, computer in computers.items():
if category in to_generate:
sd_computers[revlog_kind].append(computer)
if category in to_remove:
sd_removers[revlog_kind].append(computer)
sidedata_helpers = (repo, sd_computers, sd_removers)
return sidedata_helpers
def run_sidedata_helpers(store, sidedata_helpers, sidedata, rev):
"""Returns the sidedata for the given revision after running through
the given helpers.
- `store`: the revlog this applies to (changelog, manifest, or filelog
instance)
Raphaël Gomès
sidedata: move documentation about sidedata helpers to sidedata module...
r47849 - `sidedata_helpers`: see `get_sidedata_helpers`
Raphaël Gomès
sidedata: move sidedata-related utils to the dedicated module...
r47848 - `sidedata`: previous sidedata at the given rev, if any
- `rev`: affected rev of `store`
"""
repo, sd_computers, sd_removers = sidedata_helpers
kind = store.revlog_kind
flags_to_add = 0
flags_to_remove = 0
for _keys, sd_computer, _flags in sd_computers.get(kind, []):
sidedata, flags = sd_computer(repo, store, rev, sidedata)
flags_to_add |= flags[0]
flags_to_remove |= flags[1]
for keys, _computer, flags in sd_removers.get(kind, []):
for key in keys:
sidedata.pop(key, None)
flags_to_remove |= flags
return sidedata, (flags_to_add, flags_to_remove)
def set_sidedata_spec_for_repo(repo):
# prevent cycle metadata -> revlogutils.sidedata -> metadata
from .. import metadata
if requirementsmod.COPIESSDC_REQUIREMENT in repo.requirements:
repo.register_wanted_sidedata(SD_FILES)
repo.register_sidedata_computer(
constants.KIND_CHANGELOG,
SD_FILES,
(SD_FILES,),
metadata.copies_sidedata_computer,
flagutil.REVIDX_HASCOPIESINFO,
)