diff options
author | Boris Bobrov <breton@cynicmansion.ru> | 2015-03-02 16:44:13 +0300 |
---|---|---|
committer | Boris Bobrov <breton@cynicmansion.ru> | 2015-03-02 16:44:13 +0300 |
commit | 2b4c339de6a5762c59054182034793e3b6002ee4 (patch) | |
tree | b6bac49f255bd3aa13be17b831628dd5d390abd9 /mediagoblin/storage | |
parent | 33055472f9fd0df5944fff232f0458dad9977e08 (diff) | |
download | mediagoblin-2b4c339de6a5762c59054182034793e3b6002ee4.tar.lz mediagoblin-2b4c339de6a5762c59054182034793e3b6002ee4.tar.xz mediagoblin-2b4c339de6a5762c59054182034793e3b6002ee4.zip |
Fix bug 647
Reading a file into memory resulted in depletion of memory. Now files
are read and written lazily, by chunks.
Diffstat (limited to 'mediagoblin/storage')
-rw-r--r-- | mediagoblin/storage/cloudfiles.py | 21 | ||||
-rw-r--r-- | mediagoblin/storage/filestorage.py | 12 |
2 files changed, 11 insertions, 22 deletions
diff --git a/mediagoblin/storage/cloudfiles.py b/mediagoblin/storage/cloudfiles.py index 532e5bac..61665ea0 100644 --- a/mediagoblin/storage/cloudfiles.py +++ b/mediagoblin/storage/cloudfiles.py @@ -193,27 +193,6 @@ class CloudFilesStorageObjectWrapper(): return self.storage_object.read(*args, **kwargs) def write(self, data, *args, **kwargs): - """ - write data to the cloudfiles storage object - - The original motivation for this wrapper is to ensure - that buffered writing to a cloudfiles storage object does not overwrite - any preexisting data. - - Currently this method does not support any write modes except "append". - However if we should need it it would be easy implement. - """ - _log.warn( - '{0}.write() has bad performance! Use .send instead for now'\ - .format(self.__class__.__name__)) - - if self.storage_object.size and type(data) == str: - _log.debug('{0} is > 0 in size, appending data'.format( - self.storage_object.name)) - data = self.read() + data - - _log.debug('Writing {0}'.format( - self.storage_object.name)) self.storage_object.write(data, *args, **kwargs) def send(self, *args, **kw): diff --git a/mediagoblin/storage/filestorage.py b/mediagoblin/storage/filestorage.py index f989539c..89f43276 100644 --- a/mediagoblin/storage/filestorage.py +++ b/mediagoblin/storage/filestorage.py @@ -14,6 +14,7 @@ # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. +import io import os import shutil @@ -24,6 +25,15 @@ from mediagoblin.storage import ( clean_listy_filepath, NoWebServing) +class FileObjectAwareFile(io.FileIO): + def write(self, data): + if hasattr(data, 'read'): + # We can call data.read(). It means that the data is a file-like + # object, which should be saved RAM-friendly way + shutil.copyfileobj(data, self) + else: + super(FileObjectAwareFile, self).write(data) + class BasicFileStorage(StorageInterface): """ @@ -60,7 +70,7 @@ class BasicFileStorage(StorageInterface): os.makedirs(directory) # Grab and return the file in the mode specified - return open(self._resolve_filepath(filepath), mode) + return FileObjectAwareFile(self._resolve_filepath(filepath), mode) def delete_file(self, filepath): """Delete file at filepath |