Fix potential thumbnail memory leaks. (#12932)

This commit is contained in:
Erik Johnston 2022-06-01 11:57:49 +01:00 committed by GitHub
parent 2e8763ec96
commit 5949ab86f8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 204 additions and 137 deletions

1
changelog.d/12932.bugfix Normal file
View File

@ -0,0 +1 @@
Fix potential memory leak when generating thumbnails.

View File

@ -587,6 +587,7 @@ class MediaRepository:
) )
return None return None
with thumbnailer:
t_byte_source = await defer_to_thread( t_byte_source = await defer_to_thread(
self.hs.get_reactor(), self.hs.get_reactor(),
self._generate_thumbnail, self._generate_thumbnail,
@ -657,6 +658,7 @@ class MediaRepository:
) )
return None return None
with thumbnailer:
t_byte_source = await defer_to_thread( t_byte_source = await defer_to_thread(
self.hs.get_reactor(), self.hs.get_reactor(),
self._generate_thumbnail, self._generate_thumbnail,
@ -749,6 +751,7 @@ class MediaRepository:
) )
return None return None
with thumbnailer:
m_width = thumbnailer.width m_width = thumbnailer.width
m_height = thumbnailer.height m_height = thumbnailer.height
@ -790,11 +793,19 @@ class MediaRepository:
# Generate the thumbnail # Generate the thumbnail
if t_method == "crop": if t_method == "crop":
t_byte_source = await defer_to_thread( t_byte_source = await defer_to_thread(
self.hs.get_reactor(), thumbnailer.crop, t_width, t_height, t_type self.hs.get_reactor(),
thumbnailer.crop,
t_width,
t_height,
t_type,
) )
elif t_method == "scale": elif t_method == "scale":
t_byte_source = await defer_to_thread( t_byte_source = await defer_to_thread(
self.hs.get_reactor(), thumbnailer.scale, t_width, t_height, t_type self.hs.get_reactor(),
thumbnailer.scale,
t_width,
t_height,
t_type,
) )
else: else:
logger.error("Unrecognized method: %r", t_method) logger.error("Unrecognized method: %r", t_method)
@ -815,7 +826,11 @@ class MediaRepository:
), ),
) )
with self.media_storage.store_into_file(file_info) as (f, fname, finish): with self.media_storage.store_into_file(file_info) as (
f,
fname,
finish,
):
try: try:
await self.media_storage.write_to_file(t_byte_source, f) await self.media_storage.write_to_file(t_byte_source, f)
await finish() await finish()
@ -849,13 +864,15 @@ class MediaRepository:
t_len, t_len,
) )
except Exception as e: except Exception as e:
thumbnail_exists = await self.store.get_remote_media_thumbnail( thumbnail_exists = (
await self.store.get_remote_media_thumbnail(
server_name, server_name,
media_id, media_id,
t_width, t_width,
t_height, t_height,
t_type, t_type,
) )
)
if not thumbnail_exists: if not thumbnail_exists:
raise e raise e
else: else:

View File

@ -14,7 +14,8 @@
# limitations under the License. # limitations under the License.
import logging import logging
from io import BytesIO from io import BytesIO
from typing import Tuple from types import TracebackType
from typing import Optional, Tuple, Type
from PIL import Image from PIL import Image
@ -45,6 +46,9 @@ class Thumbnailer:
Image.MAX_IMAGE_PIXELS = max_image_pixels Image.MAX_IMAGE_PIXELS = max_image_pixels
def __init__(self, input_path: str): def __init__(self, input_path: str):
# Have we closed the image?
self._closed = False
try: try:
self.image = Image.open(input_path) self.image = Image.open(input_path)
except OSError as e: except OSError as e:
@ -89,6 +93,7 @@ class Thumbnailer:
# Safety: `transpose` takes an int rather than e.g. an IntEnum. # Safety: `transpose` takes an int rather than e.g. an IntEnum.
# self.transpose_method is set above to be a value in # self.transpose_method is set above to be a value in
# EXIF_TRANSPOSE_MAPPINGS, and that only contains correct values. # EXIF_TRANSPOSE_MAPPINGS, and that only contains correct values.
with self.image:
self.image = self.image.transpose(self.transpose_method) # type: ignore[arg-type] self.image = self.image.transpose(self.transpose_method) # type: ignore[arg-type]
self.width, self.height = self.image.size self.width, self.height = self.image.size
self.transpose_method = None self.transpose_method = None
@ -122,8 +127,10 @@ class Thumbnailer:
# If the image has transparency, use RGBA instead. # If the image has transparency, use RGBA instead.
if self.image.mode in ["1", "L", "P"]: if self.image.mode in ["1", "L", "P"]:
if self.image.info.get("transparency", None) is not None: if self.image.info.get("transparency", None) is not None:
with self.image:
self.image = self.image.convert("RGBA") self.image = self.image.convert("RGBA")
else: else:
with self.image:
self.image = self.image.convert("RGB") self.image = self.image.convert("RGB")
return self.image.resize((width, height), Image.ANTIALIAS) return self.image.resize((width, height), Image.ANTIALIAS)
@ -133,7 +140,7 @@ class Thumbnailer:
Returns: Returns:
BytesIO: the bytes of the encoded image ready to be written to disk BytesIO: the bytes of the encoded image ready to be written to disk
""" """
scaled = self._resize(width, height) with self._resize(width, height) as scaled:
return self._encode_image(scaled, output_type) return self._encode_image(scaled, output_type)
def crop(self, width: int, height: int, output_type: str) -> BytesIO: def crop(self, width: int, height: int, output_type: str) -> BytesIO:
@ -151,17 +158,20 @@ class Thumbnailer:
BytesIO: the bytes of the encoded image ready to be written to disk BytesIO: the bytes of the encoded image ready to be written to disk
""" """
if width * self.height > height * self.width: if width * self.height > height * self.width:
scaled_width = width
scaled_height = (width * self.height) // self.width scaled_height = (width * self.height) // self.width
scaled_image = self._resize(width, scaled_height)
crop_top = (scaled_height - height) // 2 crop_top = (scaled_height - height) // 2
crop_bottom = height + crop_top crop_bottom = height + crop_top
cropped = scaled_image.crop((0, crop_top, width, crop_bottom)) crop = (0, crop_top, width, crop_bottom)
else: else:
scaled_width = (height * self.width) // self.height scaled_width = (height * self.width) // self.height
scaled_image = self._resize(scaled_width, height) scaled_height = height
crop_left = (scaled_width - width) // 2 crop_left = (scaled_width - width) // 2
crop_right = width + crop_left crop_right = width + crop_left
cropped = scaled_image.crop((crop_left, 0, crop_right, height)) crop = (crop_left, 0, crop_right, height)
with self._resize(scaled_width, scaled_height) as scaled_image:
with scaled_image.crop(crop) as cropped:
return self._encode_image(cropped, output_type) return self._encode_image(cropped, output_type)
def _encode_image(self, output_image: Image.Image, output_type: str) -> BytesIO: def _encode_image(self, output_image: Image.Image, output_type: str) -> BytesIO:
@ -171,3 +181,42 @@ class Thumbnailer:
output_image = output_image.convert("RGB") output_image = output_image.convert("RGB")
output_image.save(output_bytes_io, fmt, quality=80) output_image.save(output_bytes_io, fmt, quality=80)
return output_bytes_io return output_bytes_io
def close(self) -> None:
"""Closes the underlying image file.
Once closed no other functions can be called.
Can be called multiple times.
"""
if self._closed:
return
self._closed = True
# Since we run this on the finalizer then we need to handle `__init__`
# raising an exception before it can define `self.image`.
image = getattr(self, "image", None)
if image is None:
return
image.close()
def __enter__(self) -> "Thumbnailer":
"""Make `Thumbnailer` a context manager that calls `close` on
`__exit__`.
"""
return self
def __exit__(
self,
type: Optional[Type[BaseException]],
value: Optional[BaseException],
traceback: Optional[TracebackType],
) -> None:
self.close()
def __del__(self) -> None:
# Make sure we actually do close the image, rather than leak data.
self.close()