##// END OF EJS Templates
phases: fix performance regression with Python 2...
r46127:29a259be default
Show More
store.py
197 lines | 6.1 KiB | text/x-python | PythonLexer
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 # This software may be used and distributed according to the terms of the
# GNU General Public License version 2 or any later version.
# based on bundleheads extension by Gregory Szorc <gps@mozilla.com>
from __future__ import absolute_import
import abc
import os
import subprocess
Gregory Szorc
py3: manually import pycompat.open into files that need it...
r43355 from mercurial.pycompat import open
Matt Harbison
py3: convert arguments, cwd and env to native strings when spawning subprocess...
r39851 from mercurial import (
Mark Thomas
py3: fix infinitepush extension tests...
r40288 node,
Matt Harbison
py3: convert arguments, cwd and env to native strings when spawning subprocess...
r39851 pycompat,
)
Augie Fackler
hgext: replace references to hashlib.sha1 with hashutil.sha1...
r44519 from mercurial.utils import (
hashutil,
procutil,
)
Matt Harbison
py3: convert arguments, cwd and env to native strings when spawning subprocess...
r39851
Augie Fackler
formatting: blacken the codebase...
r43346
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 class BundleWriteException(Exception):
pass
Augie Fackler
formatting: blacken the codebase...
r43346
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 class BundleReadException(Exception):
pass
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
pytype: don't warn us about ignored-on-py3 metaclasses...
r43775 class abstractbundlestore(object): # pytype: disable=ignored-metaclass
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 """Defines the interface for bundle stores.
A bundle store is an entity that stores raw bundle data. It is a simple
key-value store. However, the keys are chosen by the store. The keys can
be any Python object understood by the corresponding bundle index (see
``abstractbundleindex`` below).
"""
Augie Fackler
formatting: blacken the codebase...
r43346
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 __metaclass__ = abc.ABCMeta
@abc.abstractmethod
def write(self, data):
"""Write bundle data to the store.
This function receives the raw data to be written as a str.
Throws BundleWriteException
The key of the written data MUST be returned.
"""
@abc.abstractmethod
def read(self, key):
"""Obtain bundle data for a key.
Returns None if the bundle isn't known.
Throws BundleReadException
The returned object should be a file object supporting read()
and close().
"""
Augie Fackler
formatting: blacken the codebase...
r43346
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 class filebundlestore(object):
"""bundle store in filesystem
meant for storing bundles somewhere on disk and on network filesystems
"""
Augie Fackler
formatting: blacken the codebase...
r43346
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 def __init__(self, ui, repo):
self.ui = ui
self.repo = repo
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 self.storepath = ui.configpath(b'scratchbranch', b'storepath')
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 if not self.storepath:
Augie Fackler
formatting: blacken the codebase...
r43346 self.storepath = self.repo.vfs.join(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"scratchbranches", b"filebundlestore"
Augie Fackler
formatting: blacken the codebase...
r43346 )
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 if not os.path.exists(self.storepath):
os.makedirs(self.storepath)
def _dirpath(self, hashvalue):
"""First two bytes of the hash are the name of the upper
level directory, next two bytes are the name of the
next level directory"""
return os.path.join(self.storepath, hashvalue[0:2], hashvalue[2:4])
def _filepath(self, filename):
return os.path.join(self._dirpath(filename), filename)
def write(self, data):
Augie Fackler
hgext: replace references to hashlib.sha1 with hashutil.sha1...
r44519 filename = node.hex(hashutil.sha1(data).digest())
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 dirpath = self._dirpath(filename)
if not os.path.exists(dirpath):
os.makedirs(dirpath)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 with open(self._filepath(filename), b'wb') as f:
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 f.write(data)
return filename
def read(self, key):
try:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 with open(self._filepath(key), b'rb') as f:
Yuya Nishihara
inifinitepush: fix filebundlestore to close file
r37814 return f.read()
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 except IOError:
return None
Augie Fackler
formatting: blacken the codebase...
r43346
Connor Sheehan
infinitepush: fix `{get,put}_args` formatting on Python 3...
r45753 def format_placeholders_args(args, filename=None, handle=None):
"""Formats `args` with Infinitepush replacements.
Hack to get `str.format()`-ed strings working in a BC way with
bytes.
"""
formatted_args = []
for arg in args:
if filename and arg == b'{filename}':
formatted_args.append(filename)
elif handle and arg == b'{handle}':
formatted_args.append(handle)
else:
formatted_args.append(arg)
return formatted_args
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 class externalbundlestore(abstractbundlestore):
def __init__(self, put_binary, put_args, get_binary, get_args):
"""
`put_binary` - path to binary file which uploads bundle to external
storage and prints key to stdout
`put_args` - format string with additional args to `put_binary`
{filename} replacement field can be used.
`get_binary` - path to binary file which accepts filename and key
(in that order), downloads bundle from store and saves it to file
`get_args` - format string with additional args to `get_binary`.
{filename} and {handle} replacement field can be used.
"""
self.put_args = put_args
self.get_args = get_args
self.put_binary = put_binary
self.get_binary = get_binary
def _call_binary(self, args):
p = subprocess.Popen(
Matt Harbison
py3: convert arguments, cwd and env to native strings when spawning subprocess...
r39851 pycompat.rapply(procutil.tonativestr, args),
Augie Fackler
formatting: blacken the codebase...
r43346 stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
close_fds=True,
)
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 stdout, stderr = p.communicate()
returncode = p.returncode
return returncode, stdout, stderr
def write(self, data):
# Won't work on windows because you can't open file second time without
# closing it
Yuya Nishihara
py3: wrap tempfile.NamedTemporaryFile() to return bytes fp.name...
r38184 # TODO: rewrite without str.format() and replace NamedTemporaryFile()
# with pycompat.namedtempfile()
Connor Sheehan
infinitepush: replace `NamedTemporaryFile` with `pycompat.namedtempfile`...
r45752 with pycompat.namedtempfile() as temp:
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 temp.write(data)
temp.flush()
temp.seek(0)
Connor Sheehan
infinitepush: fix `{get,put}_args` formatting on Python 3...
r45753 formatted_args = format_placeholders_args(
self.put_args, filename=temp.name
)
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 returncode, stdout, stderr = self._call_binary(
Augie Fackler
formatting: blacken the codebase...
r43346 [self.put_binary] + formatted_args
)
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204
if returncode != 0:
raise BundleWriteException(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'Failed to upload to external store: %s' % stderr
Augie Fackler
formatting: blacken the codebase...
r43346 )
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 stdout_lines = stdout.splitlines()
if len(stdout_lines) == 1:
return stdout_lines[0]
else:
raise BundleWriteException(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'Bad output from %s: %s' % (self.put_binary, stdout)
Augie Fackler
formatting: blacken the codebase...
r43346 )
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204
def read(self, handle):
# Won't work on windows because you can't open file second time without
# closing it
Connor Sheehan
infinitepush: replace `NamedTemporaryFile` with `pycompat.namedtempfile`...
r45752 with pycompat.namedtempfile() as temp:
Connor Sheehan
infinitepush: fix `{get,put}_args` formatting on Python 3...
r45753 formatted_args = format_placeholders_args(
self.get_args, filename=temp.name, handle=handle
)
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 returncode, stdout, stderr = self._call_binary(
Augie Fackler
formatting: blacken the codebase...
r43346 [self.get_binary] + formatted_args
)
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204
if returncode != 0:
raise BundleReadException(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'Failed to download from external store: %s' % stderr
Augie Fackler
formatting: blacken the codebase...
r43346 )
Pulkit Goyal
infinitepush: move the extension to core from fb-hgext...
r37204 return temp.read()