sapling/eden/scm/edenscm/hgext/infinitepush/fileindex.py
Durham Goode 5827145ef2 py3: fix some infinitepush py3 issues
Reviewed By: quark-zju

Differential Revision: D19772618

fbshipit-source-id: c5e520a3cec6496e24cc5b3825ef7287ab82f304
2020-02-17 14:52:36 -08:00

172 lines
5.5 KiB
Python

# Copyright (c) Facebook, Inc. and its affiliates.
#
# This software may be used and distributed according to the terms of the
# GNU General Public License version 2.
"""
[infinitepush]
# Server-side option. Used only if indextype=disk.
# Filesystem path to the index store
indexpath = PATH
"""
import os
import posixpath
from edenscm.mercurial import error, pycompat, util
from edenscm.mercurial.i18n import _
if pycompat.iswindows:
def _normalizepath(path):
# Remove known characters that is disallowed by Windows.
# ":" can appear in some tests where the path is joined like:
# "C:\\repo1\\.hg\\scratchbranches\\index\\bookmarkmap\\infinitepush/backups/test/HOSTNAME/C:\\repo2/heads"
return path.replace(":", "")
else:
def _normalizepath(path):
return path
class fileindex(object):
"""File-based backend for infinitepush index.
This is a context manager. All write operations should use:
with index:
index.addbookmark(...)
...
"""
def __init__(self, repo):
self._repo = repo
root = repo.ui.config("infinitepush", "indexpath")
if not root:
root = os.path.join("scratchbranches", "index")
self._nodemap = os.path.join(root, "nodemap")
self._bookmarkmap = os.path.join(root, "bookmarkmap")
self._metadatamap = os.path.join(root, "nodemetadatamap")
self._lock = None
def __enter__(self):
self._lock = self._repo.wlock()
return self
def __exit__(self, exc_type, exc_val, exc_tb):
if self._lock:
self._lock.__exit__(exc_type, exc_val, exc_tb)
def addbundle(self, bundleid, nodesctx):
"""Record a bundleid containing the given nodes."""
for node in nodesctx:
nodepath = os.path.join(self._nodemap, node.hex())
self._write(nodepath, bundleid)
def addbookmark(self, bookmark, node, _isbackup):
"""Record a bookmark pointing to a particular node."""
bookmarkpath = os.path.join(self._bookmarkmap, bookmark)
self._write(bookmarkpath, node)
def addmanybookmarks(self, bookmarks, isbackup):
"""Record the contents of the ``bookmarks`` dict as bookmarks."""
for bookmark, node in bookmarks.items():
self.addbookmark(bookmark, node, isbackup)
def deletebookmarks(self, patterns):
"""Delete all bookmarks that match any of the patterns in ``patterns``."""
for pattern in patterns:
for bookmark, _node in self._listbookmarks(pattern):
bookmarkpath = os.path.join(self._bookmarkmap, bookmark)
self._delete(bookmarkpath)
def getbundle(self, node):
"""Get the bundleid for a bundle that contains the given node."""
nodepath = os.path.join(self._nodemap, node)
bundleid = self._read(nodepath)
return pycompat.decodeutf8(bundleid) if bundleid is not None else None
def getnodebyprefix(self, hashprefix):
"""Get the node that matches the given hash prefix.
If there is no match, returns None.
If there are multiple matches, raises an exception."""
vfs = self._repo.localvfs
if not vfs.exists(self._nodemap):
return None
files = vfs.listdir(self._nodemap)
nodefiles = list(filter(lambda n: n.startswith(hashprefix), files))
if not nodefiles:
return None
if len(nodefiles) > 1:
raise error.Abort(
_(
"ambiguous identifier '%s'\n"
"suggestion: provide longer commithash prefix"
)
% hashprefix
)
return nodefiles[0]
def getnode(self, bookmark):
"""Get the node for the given bookmark."""
bookmarkpath = os.path.join(self._bookmarkmap, bookmark)
value = self._read(bookmarkpath)
return pycompat.decodeutf8(value) if value is not None else None
def getbookmarks(self, pattern):
"""Get all bookmarks that match the pattern."""
return sorted(self._listbookmarks(pattern))
def saveoptionaljsonmetadata(self, node, jsonmetadata):
"""Save optional metadata for the given node."""
vfs = self._repo.localvfs
vfs.write(os.path.join(self._metadatamap, node), jsonmetadata)
def _listbookmarks(self, pattern):
if pattern.endswith("*"):
pattern = "re:^" + pattern[:-1] + ".*"
kind, pat, matcher = util.stringmatcher(pattern)
prefixlen = len(self._bookmarkmap) + 1
for dirpath, _dirs, books in self._repo.localvfs.walk(self._bookmarkmap):
for book in books:
bookmark = posixpath.join(dirpath, book)[prefixlen:]
if not matcher(bookmark):
continue
yield bookmark, pycompat.decodeutf8(
self._read(os.path.join(dirpath, book))
)
def _write(self, path, value):
vfs = self._repo.localvfs
path = _normalizepath(path)
dirname = vfs.dirname(path)
if not vfs.exists(dirname):
vfs.makedirs(dirname)
vfs.writeutf8(path, value)
def _read(self, path):
vfs = self._repo.localvfs
path = _normalizepath(path)
if not vfs.exists(path):
return None
return vfs.read(path)
def _delete(self, path):
vfs = self._repo.localvfs
path = _normalizepath(path)
if not vfs.exists(path):
return
return vfs.unlink(path)