basestore.py
184 lines
| 6.5 KiB
| text/x-python
|
PythonLexer
various
|
r15168 | # Copyright 2009-2010 Gregory P. Ward | ||
# Copyright 2009-2010 Intelerad Medical Systems Incorporated | ||||
# Copyright 2010-2011 Fog Creek Software | ||||
# Copyright 2010-2011 Unity Technologies | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Greg Ward
|
r15252 | '''base class for store implementations and store-related utility code''' | ||
various
|
r15168 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
various
|
r15168 | from mercurial.i18n import _ | ||
Joerg Sonnenberger
|
r47771 | from mercurial.node import short | ||
from mercurial import util | ||||
r47669 | from mercurial.utils import ( | |||
urlutil, | ||||
) | ||||
liscju
|
r29307 | |||
from . import lfutil | ||||
various
|
r15168 | |||
Augie Fackler
|
r43346 | |||
various
|
r15168 | class StoreError(Exception): | ||
Augie Fackler
|
r46554 | """Raised when there is a problem getting files from or putting | ||
files to a central store.""" | ||||
Augie Fackler
|
r43346 | |||
various
|
r15168 | def __init__(self, filename, hash, url, detail): | ||
self.filename = filename | ||||
self.hash = hash | ||||
self.url = url | ||||
self.detail = detail | ||||
def longmessage(self): | ||||
Augie Fackler
|
r43347 | return _(b"error getting id %s from url %s for file %s: %s\n") % ( | ||
Augie Fackler
|
r43346 | self.hash, | ||
r47669 | urlutil.hidepassword(self.url), | |||
Augie Fackler
|
r43346 | self.filename, | ||
self.detail, | ||||
) | ||||
various
|
r15168 | |||
def __str__(self): | ||||
r47669 | return b"%s: %s" % (urlutil.hidepassword(self.url), self.detail) | |||
various
|
r15168 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class basestore: | ||
various
|
r15168 | def __init__(self, ui, repo, url): | ||
self.ui = ui | ||||
self.repo = repo | ||||
self.url = url | ||||
def put(self, source, hash): | ||||
Mads Kiilerich
|
r19007 | '''Put source file into the store so it can be retrieved by hash.''' | ||
Augie Fackler
|
r43347 | raise NotImplementedError(b'abstract method') | ||
various
|
r15168 | |||
Na'Tosha Bard
|
r17127 | def exists(self, hashes): | ||
Augie Fackler
|
r46554 | """Check to see if the store contains the given hashes. Given an | ||
iterable of hashes it returns a mapping from hash to bool.""" | ||||
Augie Fackler
|
r43347 | raise NotImplementedError(b'abstract method') | ||
various
|
r15168 | |||
def get(self, files): | ||||
Augie Fackler
|
r46554 | """Get the specified largefiles from the store and write to local | ||
various
|
r15168 | files under repo.root. files is a list of (filename, hash) | ||
Mads Kiilerich
|
r17424 | tuples. Return (success, missing), lists of files successfully | ||
various
|
r15168 | downloaded and those not found in the store. success is a list | ||
of (filename, hash) tuples; missing is a list of filenames that | ||||
we could not get. (The detailed error message will already have | ||||
been presented to the user, so missing is just supplied as a | ||||
Augie Fackler
|
r46554 | summary.)""" | ||
various
|
r15168 | success = [] | ||
missing = [] | ||||
ui = self.ui | ||||
at = 0 | ||||
Augie Fackler
|
r44937 | available = self.exists({hash for (_filename, hash) in files}) | ||
Augie Fackler
|
r43346 | with ui.makeprogress( | ||
Augie Fackler
|
r43347 | _(b'getting largefiles'), unit=_(b'files'), total=len(files) | ||
Augie Fackler
|
r43346 | ) as progress: | ||
Matt Harbison
|
r39427 | for filename, hash in files: | ||
progress.update(at) | ||||
at += 1 | ||||
Augie Fackler
|
r43347 | ui.note(_(b'getting %s:%s\n') % (filename, hash)) | ||
various
|
r15168 | |||
Matt Harbison
|
r39427 | if not available.get(hash): | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'%s: largefile %s not available from %s\n') | ||
r47669 | % (filename, hash, urlutil.hidepassword(self.url)) | |||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r39427 | missing.append(filename) | ||
continue | ||||
Mads Kiilerich
|
r19008 | |||
Matt Harbison
|
r39427 | if self._gethash(filename, hash): | ||
success.append((filename, hash)) | ||||
else: | ||||
missing.append(filename) | ||||
various
|
r15168 | |||
return (success, missing) | ||||
Mads Kiilerich
|
r19918 | def _gethash(self, filename, hash): | ||
"""Get file with the provided hash and store it in the local repo's | ||||
store and in the usercache. | ||||
filename is for informational messages only. | ||||
""" | ||||
Augie Fackler
|
r43347 | util.makedirs(lfutil.storepath(self.repo, b'')) | ||
Mads Kiilerich
|
r19918 | storefilename = lfutil.storepath(self.repo, hash) | ||
Augie Fackler
|
r43347 | tmpname = storefilename + b'.tmp' | ||
Augie Fackler
|
r43346 | with util.atomictempfile( | ||
tmpname, createmode=self.repo.store.createmode | ||||
) as tmpfile: | ||||
Mads Kiilerich
|
r30142 | try: | ||
gothash = self._getfile(tmpfile, filename, hash) | ||||
except StoreError as err: | ||||
self.ui.warn(err.longmessage()) | ||||
Augie Fackler
|
r43347 | gothash = b"" | ||
Mads Kiilerich
|
r19918 | |||
if gothash != hash: | ||||
Augie Fackler
|
r43347 | if gothash != b"": | ||
Augie Fackler
|
r43346 | self.ui.warn( | ||
Augie Fackler
|
r43347 | _(b'%s: data corruption (expected %s, got %s)\n') | ||
Augie Fackler
|
r43346 | % (filename, hash, gothash) | ||
) | ||||
Mads Kiilerich
|
r19918 | util.unlink(tmpname) | ||
return False | ||||
util.rename(tmpname, storefilename) | ||||
lfutil.linktousercache(self.repo, hash) | ||||
return True | ||||
various
|
r15168 | def verify(self, revs, contents=False): | ||
Augie Fackler
|
r46554 | """Verify the existence (and, optionally, contents) of every big | ||
various
|
r15168 | file revision referenced by every changeset in revs. | ||
Augie Fackler
|
r46554 | Return 0 if all is well, non-zero on any errors.""" | ||
various
|
r15168 | |||
Augie Fackler
|
r43346 | self.ui.status( | ||
Augie Fackler
|
r43347 | _(b'searching %d changesets for largefiles\n') % len(revs) | ||
Augie Fackler
|
r43346 | ) | ||
verified = set() # set of (filename, filenode) tuples | ||||
filestocheck = [] # list of (cset, filename, expectedhash) | ||||
various
|
r15168 | for rev in revs: | ||
cctx = self.repo[rev] | ||||
Joerg Sonnenberger
|
r47771 | cset = b"%d:%s" % (cctx.rev(), short(cctx.node())) | ||
various
|
r15168 | |||
Mads Kiilerich
|
r18486 | for standin in cctx: | ||
liscju
|
r29067 | filename = lfutil.splitstandin(standin) | ||
if filename: | ||||
fctx = cctx[standin] | ||||
key = (filename, fctx.filenode()) | ||||
if key not in verified: | ||||
verified.add(key) | ||||
FUJIWARA Katsunori
|
r31740 | expectedhash = lfutil.readasstandin(fctx) | ||
liscju
|
r29067 | filestocheck.append((cset, filename, expectedhash)) | ||
failed = self._verifyfiles(contents, filestocheck) | ||||
various
|
r15168 | |||
Na'Tosha Bard
|
r16247 | numrevs = len(verified) | ||
Martin von Zweigbergk
|
r42224 | numlfiles = len({fname for (fname, fnode) in verified}) | ||
various
|
r15168 | if contents: | ||
Mads Kiilerich
|
r18546 | self.ui.status( | ||
Augie Fackler
|
r43347 | _(b'verified contents of %d revisions of %d largefiles\n') | ||
Augie Fackler
|
r43346 | % (numrevs, numlfiles) | ||
) | ||||
various
|
r15168 | else: | ||
Mads Kiilerich
|
r18546 | self.ui.status( | ||
Augie Fackler
|
r43347 | _(b'verified existence of %d revisions of %d largefiles\n') | ||
Augie Fackler
|
r43346 | % (numrevs, numlfiles) | ||
) | ||||
various
|
r15168 | return int(failed) | ||
def _getfile(self, tmpfile, filename, hash): | ||||
Augie Fackler
|
r46554 | """Fetch one revision of one file from the store and write it | ||
various
|
r15168 | to tmpfile. Compute the hash of the file on-the-fly as it | ||
Mads Kiilerich
|
r18999 | downloads and return the hash. Close tmpfile. Raise | ||
various
|
r15168 | StoreError if unable to download the file (e.g. it does not | ||
Augie Fackler
|
r46554 | exist in the store).""" | ||
Augie Fackler
|
r43347 | raise NotImplementedError(b'abstract method') | ||
various
|
r15168 | |||
liscju
|
r29067 | def _verifyfiles(self, contents, filestocheck): | ||
Augie Fackler
|
r46554 | """Perform the actual verification of files in the store. | ||
Mads Kiilerich
|
r18574 | 'contents' controls verification of content hash. | ||
liscju
|
r29067 | 'filestocheck' is list of files to check. | ||
Returns _true_ if any problems are found! | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43347 | raise NotImplementedError(b'abstract method') | ||