/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar

« back to all changes in this revision

Viewing changes to breezy/store/text.py

  • Committer: Martin
  • Date: 2017-06-05 20:48:31 UTC
  • mto: This revision was merged to the branch mainline in revision 6658.
  • Revision ID: gzlist@googlemail.com-20170605204831-20accykspjcrx0a8
Apply 2to3 dict fixer and clean up resulting mess using view helpers

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2005, 2006, 2008, 2009, 2010 Canonical Ltd
 
2
#
 
3
# This program is free software; you can redistribute it and/or modify
 
4
# it under the terms of the GNU General Public License as published by
 
5
# the Free Software Foundation; either version 2 of the License, or
 
6
# (at your option) any later version.
 
7
#
 
8
# This program is distributed in the hope that it will be useful,
 
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
11
# GNU General Public License for more details.
 
12
#
 
13
# You should have received a copy of the GNU General Public License
 
14
# along with this program; if not, write to the Free Software
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
16
 
 
17
"""A store that keeps the full text of every version.
 
18
 
 
19
This store keeps uncompressed versions of the full text. It does not
 
20
do any sort of delta compression.
 
21
"""
 
22
 
 
23
from __future__ import absolute_import
 
24
 
 
25
import gzip
 
26
import os
 
27
 
 
28
from .. import osutils
 
29
from ..errors import BzrError, NoSuchFile, FileExists
 
30
from ..sixish import (
 
31
    BytesIO,
 
32
    )
 
33
from . import TransportStore
 
34
from ..trace import mutter
 
35
 
 
36
 
 
37
 
 
38
class TextStore(TransportStore):
 
39
    """Store that holds files indexed by unique names.
 
40
 
 
41
    Files can be added, but not modified once they are in.  Typically
 
42
    the hash is used as the name, or something else known to be unique,
 
43
    such as a UUID.
 
44
 
 
45
    Files are stored uncompressed, with no delta compression.
 
46
    """
 
47
 
 
48
    def _add_compressed(self, fn, f):
 
49
        from ..osutils import pumpfile
 
50
 
 
51
        if isinstance(f, bytes):
 
52
            f = BytesIO(f)
 
53
 
 
54
        sio = BytesIO()
 
55
        gf = gzip.GzipFile(mode='wb', fileobj=sio)
 
56
        # if pumpfile handles files that don't fit in ram,
 
57
        # so will this function
 
58
        pumpfile(f, gf)
 
59
        gf.close()
 
60
        sio.seek(0)
 
61
        self._try_put(fn, sio)
 
62
 
 
63
    def _add(self, fn, f):
 
64
        if self._compressed:
 
65
            self._add_compressed(fn, f)
 
66
        else:
 
67
            self._try_put(fn, f)
 
68
 
 
69
    def _try_put(self, fn, f):
 
70
        try:
 
71
            self._transport.put_file(fn, f, mode=self._file_mode)
 
72
        except NoSuchFile:
 
73
            if not self._prefixed:
 
74
                raise
 
75
            try:
 
76
                self._transport.mkdir(os.path.dirname(fn), mode=self._dir_mode)
 
77
            except FileExists:
 
78
                pass
 
79
            self._transport.put_file(fn, f, mode=self._file_mode)
 
80
 
 
81
    def _get(self, fn):
 
82
        if fn.endswith('.gz'):
 
83
            return self._get_compressed(fn)
 
84
        else:
 
85
            return self._transport.get(fn)
 
86
 
 
87
    def _copy_one(self, fileid, suffix, other, pb):
 
88
        # TODO: Once the copy_to interface is improved to allow a source
 
89
        #       and destination targets, then we can always do the copy
 
90
        #       as long as other is a TextStore
 
91
        if not (isinstance(other, TextStore)
 
92
            and other._prefixed == self._prefixed):
 
93
            return super(TextStore, self)._copy_one(fileid, suffix, other, pb)
 
94
 
 
95
        mutter('_copy_one: %r, %r', fileid, suffix)
 
96
        path = other._get_name(fileid, suffix)
 
97
        if path is None:
 
98
            raise KeyError(fileid + '-' + str(suffix))
 
99
 
 
100
        try:
 
101
            result = other._transport.copy_to([path], self._transport,
 
102
                                              mode=self._file_mode)
 
103
        except NoSuchFile:
 
104
            if not self._prefixed:
 
105
                raise
 
106
            try:
 
107
                self._transport.mkdir(osutils.dirname(path), mode=self._dir_mode)
 
108
            except FileExists:
 
109
                pass
 
110
            result = other._transport.copy_to([path], self._transport,
 
111
                                              mode=self._file_mode)
 
112
 
 
113
        if result != 1:
 
114
            raise BzrError('Unable to copy file: %r' % (path,))
 
115
 
 
116
    def _get_compressed(self, filename):
 
117
        """Returns a file reading from a particular entry."""
 
118
        f = self._transport.get(filename)
 
119
        # gzip.GzipFile.read() requires a tell() function
 
120
        # but some transports return objects that cannot seek
 
121
        # so buffer them in a BytesIO instead
 
122
        if getattr(f, 'tell', None) is not None:
 
123
            return gzip.GzipFile(mode='rb', fileobj=f)
 
124
        try:
 
125
            sio = BytesIO(f.read())
 
126
            return gzip.GzipFile(mode='rb', fileobj=sio)
 
127
        finally:
 
128
            f.close()