1
0
mirror of synced 2024-11-11 11:58:51 +01:00
mat2/libmat2/images.py

203 lines
8.0 KiB
Python
Raw Normal View History

import imghdr
2018-04-01 00:43:36 +02:00
import os
2019-09-01 18:28:46 +02:00
import re
2022-08-28 22:29:06 +02:00
from typing import Union, Any
2018-03-25 15:09:12 +02:00
import cairo
2018-03-25 15:09:12 +02:00
import gi
gi.require_version('GdkPixbuf', '2.0')
2019-07-13 21:26:05 +02:00
gi.require_version('Rsvg', '2.0')
from gi.repository import GdkPixbuf, GLib, Rsvg
2018-03-25 15:09:12 +02:00
2019-09-01 18:28:46 +02:00
from . import exiftool, abstract
2018-03-25 15:09:12 +02:00
# Make pyflakes happy
2019-09-01 18:28:46 +02:00
assert Any
2019-07-13 21:26:05 +02:00
class SVGParser(exiftool.ExiftoolParser):
mimetypes = {'image/svg+xml', }
meta_allowlist = {'Directory', 'ExifToolVersion', 'FileAccessDate',
'FileInodeChangeDate', 'FileModifyDate', 'FileName',
'FilePermissions', 'FileSize', 'FileType',
'FileTypeExtension', 'ImageHeight', 'ImageWidth',
'MIMEType', 'SVGVersion', 'SourceFile', 'ViewBox'
}
def remove_all(self) -> bool:
try:
svg = Rsvg.Handle.new_from_file(self.filename)
except GLib.GError:
raise ValueError
try:
_, _, _, _, has_viewbox, viewbox = svg.get_intrinsic_dimensions()
if has_viewbox is False:
raise ValueError
_, width, height = svg.get_intrinsic_size_in_pixels()
except AttributeError:
dimensions = svg.get_dimensions()
height, width = dimensions.height, dimensions.width
surface = cairo.SVGSurface(self.output_filename, height, width)
2019-07-13 21:26:05 +02:00
context = cairo.Context(surface)
try:
svg.render_document(context, viewbox)
except AttributeError:
svg.render_cairo(context)
2019-07-13 21:26:05 +02:00
surface.finish()
return True
2022-08-28 22:29:06 +02:00
def get_meta(self) -> dict[str, Union[str, dict]]:
2019-07-13 21:26:05 +02:00
meta = super().get_meta()
2019-07-22 23:20:37 +02:00
# The namespace is mandatory, but only the …/2000/svg is valid.
2019-07-13 21:26:05 +02:00
ns = 'http://www.w3.org/2000/svg'
if meta.get('Xmlns') == ns:
2019-07-13 21:26:05 +02:00
meta.pop('Xmlns')
return meta
class PNGParser(exiftool.ExiftoolParser):
mimetypes = {'image/png', }
meta_allowlist = {'SourceFile', 'ExifToolVersion', 'FileName',
2018-05-16 22:36:59 +02:00
'Directory', 'FileSize', 'FileModifyDate',
'FileAccessDate', 'FileInodeChangeDate',
'FilePermissions', 'FileType', 'FileTypeExtension',
'MIMEType', 'ImageWidth', 'BitDepth', 'ColorType',
'Compression', 'Filter', 'Interlace', 'BackgroundColor',
'ImageSize', 'Megapixels', 'ImageHeight'}
def __init__(self, filename):
super().__init__(filename)
if imghdr.what(filename) != 'png':
raise ValueError
try: # better fail here than later
cairo.ImageSurface.create_from_png(self.filename)
except: # pragma: no cover
2019-12-15 15:57:32 +01:00
# Cairo is returning some weird exceptions :/
raise ValueError
2018-10-12 11:58:01 +02:00
def remove_all(self) -> bool:
if self.lightweight_cleaning:
return self._lightweight_cleanup()
surface = cairo.ImageSurface.create_from_png(self.filename)
surface.write_to_png(self.output_filename)
return True
2019-02-03 21:01:58 +01:00
class GIFParser(exiftool.ExiftoolParser):
mimetypes = {'image/gif'}
meta_allowlist = {'AnimationIterations', 'BackgroundColor', 'BitsPerPixel',
2019-02-03 21:01:58 +01:00
'ColorResolutionDepth', 'Directory', 'Duration',
'ExifToolVersion', 'FileAccessDate',
'FileInodeChangeDate', 'FileModifyDate', 'FileName',
'FilePermissions', 'FileSize', 'FileType',
'FileTypeExtension', 'FrameCount', 'GIFVersion',
'HasColorMap', 'ImageHeight', 'ImageSize', 'ImageWidth',
'MIMEType', 'Megapixels', 'SourceFile',}
def remove_all(self) -> bool:
return self._lightweight_cleanup()
class GdkPixbufAbstractParser(exiftool.ExiftoolParser):
2018-04-02 23:40:08 +02:00
""" GdkPixbuf can handle a lot of surfaces, so we're rending images on it,
2018-07-19 23:10:27 +02:00
this has the side-effect of completely removing metadata.
2018-04-02 23:40:08 +02:00
"""
_type = ''
2018-10-12 11:58:01 +02:00
def __init__(self, filename):
super().__init__(filename)
2018-10-24 19:35:07 +02:00
# we can't use imghdr here because of https://bugs.python.org/issue28591
try:
GdkPixbuf.Pixbuf.new_from_file(self.filename)
except GLib.GError:
2018-10-12 11:58:01 +02:00
raise ValueError
def remove_all(self) -> bool:
2018-10-24 19:35:07 +02:00
if self.lightweight_cleaning:
return self._lightweight_cleanup()
2018-04-01 00:43:36 +02:00
_, extension = os.path.splitext(self.filename)
2018-03-25 15:09:12 +02:00
pixbuf = GdkPixbuf.Pixbuf.new_from_file(self.filename)
if extension.lower() == '.jpg':
extension = '.jpeg' # gdk is picky
elif extension.lower() == '.tif':
extension = '.tiff' # gdk is picky
try:
pixbuf.savev(self.output_filename, type=extension[1:],
option_keys=[], option_values=[])
except GLib.GError: # pragma: no cover
return False
2018-03-25 15:09:12 +02:00
return True
2018-04-01 00:43:36 +02:00
class JPGParser(GdkPixbufAbstractParser):
_type = 'jpeg'
mimetypes = {'image/jpeg'}
meta_allowlist = {'SourceFile', 'ExifToolVersion', 'FileName',
2018-05-16 22:36:59 +02:00
'Directory', 'FileSize', 'FileModifyDate',
'FileAccessDate', "FileInodeChangeDate",
'FilePermissions', 'FileType', 'FileTypeExtension',
'MIMEType', 'ImageWidth', 'ImageSize', 'BitsPerSample',
'ColorComponents', 'EncodingProcess', 'JFIFVersion',
'ResolutionUnit', 'XResolution', 'YCbCrSubSampling',
'YResolution', 'Megapixels', 'ImageHeight'}
2018-04-01 00:43:36 +02:00
class TiffParser(GdkPixbufAbstractParser):
_type = 'tiff'
2018-04-01 00:43:36 +02:00
mimetypes = {'image/tiff'}
meta_allowlist = {'Compression', 'ExifByteOrder', 'ExtraSamples',
2018-05-16 22:36:59 +02:00
'FillOrder', 'PhotometricInterpretation',
'PlanarConfiguration', 'RowsPerStrip', 'SamplesPerPixel',
'StripByteCounts', 'StripOffsets', 'BitsPerSample',
'Directory', 'ExifToolVersion', 'FileAccessDate',
'FileInodeChangeDate', 'FileModifyDate', 'FileName',
'FilePermissions', 'FileSize', 'FileType',
'FileTypeExtension', 'ImageHeight', 'ImageSize',
'ImageWidth', 'MIMEType', 'Megapixels', 'SourceFile'}
2019-09-01 18:28:46 +02:00
class PPMParser(abstract.AbstractParser):
mimetypes = {'image/x-portable-pixmap'}
2022-08-28 22:29:06 +02:00
def get_meta(self) -> dict[str, Union[str, dict]]:
meta = {} # type: dict[str, Union[str, dict[Any, Any]]]
2019-09-01 18:28:46 +02:00
with open(self.filename) as f:
for idx, line in enumerate(f):
if line.lstrip().startswith('#'):
meta[str(idx)] = line.lstrip().rstrip()
return meta
def remove_all(self) -> bool:
with open(self.filename) as fin:
with open(self.output_filename, 'w') as fout:
for line in fin:
if not line.lstrip().startswith('#'):
line = re.sub(r"\s+", "", line, flags=re.UNICODE)
fout.write(line)
return True
class HEICParser(exiftool.ExiftoolParser):
mimetypes = {'image/heic'}
meta_allowlist = {'SourceFile', 'ExifToolVersion', 'FileName','Directory',
'FileSize', 'FileModifyDate', 'FileAccessDate',
'FileInodeChangeDate', 'FilePermissions', 'FileType',
'FileTypeExtension', 'MIMEType', 'MajorBrand', 'MinorVersion',
'CompatibleBrands','HandlerType', 'PrimaryItemReference',
'HEVCConfigurationVersion', 'GeneralProfileSpace',
'GeneralTierFlag', 'GeneralProfileIDC',
'GenProfileCompatibilityFlags', 'ConstraintIndicatorFlags',
'GeneralLevelIDC', 'MinSpatialSegmentationIDC',
'ParallelismType','ChromaFormat', 'BitDepthLuma', 'BitDepthChroma',
'NumTemporalLayers', 'TemporalIDNested', 'ImageWidth',
'ImageHeight', 'ImageSpatialExtent', 'ImagePixelDepth',
'AverageFrameRate', 'ConstantFrameRate', 'MediaDataSize',
'MediaDataOffset','ImageSize', 'Megapixels'}
def remove_all(self) -> bool:
return self._lightweight_cleanup()