2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-05-25 21:11:33 +03:00
|
|
|
import collections
|
|
|
|
import os
|
|
|
|
import sys
|
2020-02-18 01:22:15 +03:00
|
|
|
import warnings
|
2024-06-12 14:15:55 +03:00
|
|
|
from typing import IO
|
2019-05-25 21:11:33 +03:00
|
|
|
|
|
|
|
import PIL
|
2024-07-08 19:07:43 +03:00
|
|
|
from PIL import _deprecate
|
2019-07-06 23:40:53 +03:00
|
|
|
|
2017-01-17 16:22:18 +03:00
|
|
|
from . import Image
|
2015-05-14 12:43:28 +03:00
|
|
|
|
|
|
|
modules = {
|
2020-06-14 06:35:43 +03:00
|
|
|
"pil": ("PIL._imaging", "PILLOW_VERSION"),
|
2021-01-26 00:01:26 +03:00
|
|
|
"tkinter": ("PIL._tkinter_finder", "tk_version"),
|
2020-06-14 06:35:43 +03:00
|
|
|
"freetype2": ("PIL._imagingft", "freetype2_version"),
|
|
|
|
"littlecms2": ("PIL._imagingcms", "littlecms_version"),
|
|
|
|
"webp": ("PIL._webp", "webpdecoder_version"),
|
2015-05-14 12:43:28 +03:00
|
|
|
}
|
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def check_module(feature: str) -> bool:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
Checks if a module is available.
|
|
|
|
|
|
|
|
:param feature: The module to check for.
|
2020-06-14 01:45:29 +03:00
|
|
|
:returns: ``True`` if available, ``False`` otherwise.
|
2020-06-13 05:01:38 +03:00
|
|
|
:raises ValueError: If the module is not defined in this version of Pillow.
|
|
|
|
"""
|
2023-06-14 01:52:55 +03:00
|
|
|
if feature not in modules:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Unknown module {feature}"
|
|
|
|
raise ValueError(msg)
|
2015-05-14 13:57:01 +03:00
|
|
|
|
2020-06-14 06:35:43 +03:00
|
|
|
module, ver = modules[feature]
|
2015-05-14 12:43:28 +03:00
|
|
|
|
|
|
|
try:
|
2018-03-03 12:28:58 +03:00
|
|
|
__import__(module)
|
2015-05-14 12:43:28 +03:00
|
|
|
return True
|
2023-03-31 03:57:58 +03:00
|
|
|
except ModuleNotFoundError:
|
|
|
|
return False
|
|
|
|
except ImportError as ex:
|
|
|
|
warnings.warn(str(ex))
|
2017-05-12 00:01:58 +03:00
|
|
|
return False
|
2015-05-14 12:43:28 +03:00
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def version_module(feature: str) -> str | None:
|
2020-06-14 06:35:43 +03:00
|
|
|
"""
|
|
|
|
:param feature: The module to check for.
|
|
|
|
:returns:
|
|
|
|
The loaded version number as a string, or ``None`` if unknown or not available.
|
|
|
|
:raises ValueError: If the module is not defined in this version of Pillow.
|
|
|
|
"""
|
2019-10-12 16:29:10 +03:00
|
|
|
if not check_module(feature):
|
|
|
|
return None
|
|
|
|
|
2020-06-14 06:35:43 +03:00
|
|
|
module, ver = modules[feature]
|
2019-10-12 16:29:10 +03:00
|
|
|
|
2020-06-14 06:35:43 +03:00
|
|
|
return getattr(__import__(module, fromlist=[ver]), ver)
|
2019-10-12 16:29:10 +03:00
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def get_supported_modules() -> list[str]:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
:returns: A list of all supported modules.
|
|
|
|
"""
|
2017-05-12 00:01:58 +03:00
|
|
|
return [f for f in modules if check_module(f)]
|
2015-05-14 12:43:28 +03:00
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2019-10-12 16:29:10 +03:00
|
|
|
codecs = {
|
|
|
|
"jpg": ("jpeg", "jpeglib"),
|
|
|
|
"jpg_2000": ("jpeg2k", "jp2klib"),
|
|
|
|
"zlib": ("zip", "zlib"),
|
|
|
|
"libtiff": ("libtiff", "libtiff"),
|
|
|
|
}
|
2015-05-14 12:43:28 +03:00
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def check_codec(feature: str) -> bool:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
Checks if a codec is available.
|
|
|
|
|
|
|
|
:param feature: The codec to check for.
|
2020-06-14 01:45:29 +03:00
|
|
|
:returns: ``True`` if available, ``False`` otherwise.
|
2020-06-13 05:01:38 +03:00
|
|
|
:raises ValueError: If the codec is not defined in this version of Pillow.
|
|
|
|
"""
|
2015-05-14 13:57:01 +03:00
|
|
|
if feature not in codecs:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Unknown codec {feature}"
|
|
|
|
raise ValueError(msg)
|
2015-05-14 13:57:01 +03:00
|
|
|
|
2019-10-12 16:29:10 +03:00
|
|
|
codec, lib = codecs[feature]
|
2015-05-14 13:57:01 +03:00
|
|
|
|
2024-05-04 19:21:49 +03:00
|
|
|
return f"{codec}_encoder" in dir(Image.core)
|
2015-05-14 12:43:28 +03:00
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def version_codec(feature: str) -> str | None:
|
2020-06-14 06:35:43 +03:00
|
|
|
"""
|
|
|
|
:param feature: The codec to check for.
|
|
|
|
:returns:
|
|
|
|
The version number as a string, or ``None`` if not available.
|
|
|
|
Checked at compile time for ``jpg``, run-time otherwise.
|
|
|
|
:raises ValueError: If the codec is not defined in this version of Pillow.
|
|
|
|
"""
|
2019-10-12 16:29:10 +03:00
|
|
|
if not check_codec(feature):
|
|
|
|
return None
|
|
|
|
|
|
|
|
codec, lib = codecs[feature]
|
|
|
|
|
2024-05-04 19:21:49 +03:00
|
|
|
version = getattr(Image.core, f"{lib}_version")
|
2019-10-12 16:29:10 +03:00
|
|
|
|
|
|
|
if feature == "libtiff":
|
|
|
|
return version.split("\n")[0].split("Version ")[1]
|
|
|
|
|
|
|
|
return version
|
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def get_supported_codecs() -> list[str]:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
:returns: A list of all supported codecs.
|
|
|
|
"""
|
2017-05-12 00:01:58 +03:00
|
|
|
return [f for f in codecs if check_codec(f)]
|
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2017-05-12 00:01:58 +03:00
|
|
|
features = {
|
2024-07-08 19:07:43 +03:00
|
|
|
"webp_anim": ("PIL._webp", True, None),
|
|
|
|
"webp_mux": ("PIL._webp", True, None),
|
|
|
|
"transp_webp": ("PIL._webp", True, None),
|
2020-06-14 06:35:43 +03:00
|
|
|
"raqm": ("PIL._imagingft", "HAVE_RAQM", "raqm_version"),
|
2020-11-25 17:01:16 +03:00
|
|
|
"fribidi": ("PIL._imagingft", "HAVE_FRIBIDI", "fribidi_version"),
|
|
|
|
"harfbuzz": ("PIL._imagingft", "HAVE_HARFBUZZ", "harfbuzz_version"),
|
2020-06-14 06:35:43 +03:00
|
|
|
"libjpeg_turbo": ("PIL._imaging", "HAVE_LIBJPEGTURBO", "libjpeg_turbo_version"),
|
|
|
|
"libimagequant": ("PIL._imaging", "HAVE_LIBIMAGEQUANT", "imagequant_version"),
|
|
|
|
"xcb": ("PIL._imaging", "HAVE_XCB", None),
|
2017-05-12 00:01:58 +03:00
|
|
|
}
|
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def check_feature(feature: str) -> bool | None:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
Checks if a feature is available.
|
|
|
|
|
|
|
|
:param feature: The feature to check for.
|
2020-06-14 01:45:29 +03:00
|
|
|
:returns: ``True`` if available, ``False`` if unavailable, ``None`` if unknown.
|
2020-06-13 05:01:38 +03:00
|
|
|
:raises ValueError: If the feature is not defined in this version of Pillow.
|
|
|
|
"""
|
2017-05-12 00:01:58 +03:00
|
|
|
if feature not in features:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Unknown feature {feature}"
|
|
|
|
raise ValueError(msg)
|
2017-05-12 00:01:58 +03:00
|
|
|
|
2020-06-14 06:35:43 +03:00
|
|
|
module, flag, ver = features[feature]
|
2017-09-28 05:04:24 +03:00
|
|
|
|
2017-05-12 00:01:58 +03:00
|
|
|
try:
|
|
|
|
imported_module = __import__(module, fromlist=["PIL"])
|
2024-07-08 19:07:43 +03:00
|
|
|
if isinstance(flag, str):
|
|
|
|
return getattr(imported_module, flag)
|
|
|
|
else:
|
|
|
|
_deprecate.deprecate(f'check_feature("{feature}")', 12)
|
|
|
|
return flag
|
2023-03-31 03:57:58 +03:00
|
|
|
except ModuleNotFoundError:
|
|
|
|
return None
|
|
|
|
except ImportError as ex:
|
|
|
|
warnings.warn(str(ex))
|
2017-05-12 00:01:58 +03:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def version_feature(feature: str) -> str | None:
|
2020-06-14 06:35:43 +03:00
|
|
|
"""
|
|
|
|
:param feature: The feature to check for.
|
|
|
|
:returns: The version number as a string, or ``None`` if not available.
|
|
|
|
:raises ValueError: If the feature is not defined in this version of Pillow.
|
|
|
|
"""
|
|
|
|
if not check_feature(feature):
|
|
|
|
return None
|
|
|
|
|
|
|
|
module, flag, ver = features[feature]
|
|
|
|
|
|
|
|
if ver is None:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return getattr(__import__(module, fromlist=[ver]), ver)
|
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def get_supported_features() -> list[str]:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
:returns: A list of all supported features.
|
|
|
|
"""
|
2017-05-12 00:01:58 +03:00
|
|
|
return [f for f in features if check_feature(f)]
|
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def check(feature: str) -> bool | None:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
2020-06-14 06:35:43 +03:00
|
|
|
:param feature: A module, codec, or feature name.
|
2020-06-13 05:01:38 +03:00
|
|
|
:returns:
|
2020-06-14 06:35:43 +03:00
|
|
|
``True`` if the module, codec, or feature is available,
|
2020-06-14 01:45:29 +03:00
|
|
|
``False`` or ``None`` otherwise.
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
|
2020-02-18 01:22:15 +03:00
|
|
|
if feature in modules:
|
|
|
|
return check_module(feature)
|
|
|
|
if feature in codecs:
|
|
|
|
return check_codec(feature)
|
|
|
|
if feature in features:
|
|
|
|
return check_feature(feature)
|
2020-07-16 12:43:29 +03:00
|
|
|
warnings.warn(f"Unknown feature '{feature}'.", stacklevel=2)
|
2020-02-18 01:22:15 +03:00
|
|
|
return False
|
2017-05-12 00:01:58 +03:00
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def version(feature: str) -> str | None:
|
2020-06-14 06:35:43 +03:00
|
|
|
"""
|
|
|
|
:param feature:
|
|
|
|
The module, codec, or feature to check for.
|
|
|
|
:returns:
|
|
|
|
The version number as a string, or ``None`` if unknown or not available.
|
|
|
|
"""
|
2019-10-12 16:29:10 +03:00
|
|
|
if feature in modules:
|
|
|
|
return version_module(feature)
|
|
|
|
if feature in codecs:
|
|
|
|
return version_codec(feature)
|
2020-06-14 06:35:43 +03:00
|
|
|
if feature in features:
|
|
|
|
return version_feature(feature)
|
2019-10-12 16:29:10 +03:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def get_supported() -> list[str]:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
:returns: A list of all supported modules, features, and codecs.
|
|
|
|
"""
|
|
|
|
|
2017-05-12 00:01:58 +03:00
|
|
|
ret = get_supported_modules()
|
|
|
|
ret.extend(get_supported_features())
|
|
|
|
ret.extend(get_supported_codecs())
|
|
|
|
return ret
|
2019-05-25 21:11:33 +03:00
|
|
|
|
|
|
|
|
2024-06-12 14:15:55 +03:00
|
|
|
def pilinfo(out: IO[str] | None = None, supported_formats: bool = True) -> None:
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
Prints information about this installation of Pillow.
|
2021-05-08 05:37:06 +03:00
|
|
|
This function can be called with ``python3 -m PIL``.
|
2024-03-30 10:13:44 +03:00
|
|
|
It can also be called with ``python3 -m PIL.report`` or ``python3 -m PIL --report``
|
|
|
|
to have "supported_formats" set to ``False``, omitting the list of all supported
|
|
|
|
image file formats.
|
2020-06-13 05:01:38 +03:00
|
|
|
|
|
|
|
:param out:
|
2020-06-14 01:45:29 +03:00
|
|
|
The output stream to print to. Defaults to ``sys.stdout`` if ``None``.
|
2020-06-13 05:01:38 +03:00
|
|
|
:param supported_formats:
|
2020-06-14 01:45:29 +03:00
|
|
|
If ``True``, a list of all supported image file formats will be printed.
|
2020-06-13 05:01:38 +03:00
|
|
|
"""
|
|
|
|
|
2019-05-25 21:11:33 +03:00
|
|
|
if out is None:
|
|
|
|
out = sys.stdout
|
|
|
|
|
|
|
|
Image.init()
|
|
|
|
|
|
|
|
print("-" * 68, file=out)
|
2020-07-16 12:43:29 +03:00
|
|
|
print(f"Pillow {PIL.__version__}", file=out)
|
2024-06-12 14:15:55 +03:00
|
|
|
py_version_lines = sys.version.splitlines()
|
|
|
|
print(f"Python {py_version_lines[0].strip()}", file=out)
|
|
|
|
for py_version in py_version_lines[1:]:
|
2020-07-16 12:43:29 +03:00
|
|
|
print(f" {py_version.strip()}", file=out)
|
2019-05-25 21:11:33 +03:00
|
|
|
print("-" * 68, file=out)
|
2024-02-20 22:37:33 +03:00
|
|
|
print(f"Python executable is {sys.executable or 'unknown'}", file=out)
|
|
|
|
if sys.prefix != sys.base_prefix:
|
|
|
|
print(f"Environment Python files loaded from {sys.prefix}", file=out)
|
|
|
|
print(f"System Python files loaded from {sys.base_prefix}", file=out)
|
|
|
|
print("-" * 68, file=out)
|
2019-05-25 21:11:33 +03:00
|
|
|
print(
|
2024-02-20 22:37:33 +03:00
|
|
|
f"Python Pillow modules loaded from {os.path.dirname(Image.__file__)}",
|
2020-09-01 20:16:46 +03:00
|
|
|
file=out,
|
2019-05-25 21:11:33 +03:00
|
|
|
)
|
|
|
|
print(
|
2024-02-20 22:37:33 +03:00
|
|
|
f"Binary Pillow modules loaded from {os.path.dirname(Image.core.__file__)}",
|
2020-09-01 20:16:46 +03:00
|
|
|
file=out,
|
2019-05-25 21:11:33 +03:00
|
|
|
)
|
|
|
|
print("-" * 68, file=out)
|
|
|
|
|
|
|
|
for name, feature in [
|
|
|
|
("pil", "PIL CORE"),
|
|
|
|
("tkinter", "TKINTER"),
|
|
|
|
("freetype2", "FREETYPE2"),
|
|
|
|
("littlecms2", "LITTLECMS2"),
|
|
|
|
("webp", "WEBP"),
|
|
|
|
("jpg", "JPEG"),
|
|
|
|
("jpg_2000", "OPENJPEG (JPEG2000)"),
|
|
|
|
("zlib", "ZLIB (PNG/ZIP)"),
|
|
|
|
("libtiff", "LIBTIFF"),
|
|
|
|
("raqm", "RAQM (Bidirectional Text)"),
|
2019-10-12 17:01:18 +03:00
|
|
|
("libimagequant", "LIBIMAGEQUANT (Quantization method)"),
|
2019-09-19 19:57:59 +03:00
|
|
|
("xcb", "XCB (X protocol)"),
|
2019-05-25 21:11:33 +03:00
|
|
|
]:
|
|
|
|
if check(name):
|
2024-06-12 14:15:55 +03:00
|
|
|
v: str | None = None
|
|
|
|
if name == "jpg":
|
|
|
|
libjpeg_turbo_version = version_feature("libjpeg_turbo")
|
|
|
|
if libjpeg_turbo_version is not None:
|
|
|
|
v = "libjpeg-turbo " + libjpeg_turbo_version
|
|
|
|
if v is None:
|
2020-06-14 06:35:43 +03:00
|
|
|
v = version(name)
|
2019-10-12 16:29:10 +03:00
|
|
|
if v is not None:
|
2020-10-12 04:58:08 +03:00
|
|
|
version_static = name in ("pil", "jpg")
|
|
|
|
if name == "littlecms2":
|
2020-10-13 19:37:19 +03:00
|
|
|
# this check is also in src/_imagingcms.c:setup_module()
|
|
|
|
version_static = tuple(int(x) for x in v.split(".")) < (2, 7)
|
2020-10-12 04:58:08 +03:00
|
|
|
t = "compiled for" if version_static else "loaded"
|
2020-11-25 17:01:16 +03:00
|
|
|
if name == "raqm":
|
|
|
|
for f in ("fribidi", "harfbuzz"):
|
|
|
|
v2 = version_feature(f)
|
|
|
|
if v2 is not None:
|
|
|
|
v += f", {f} {v2}"
|
2020-06-21 20:07:10 +03:00
|
|
|
print("---", feature, "support ok,", t, v, file=out)
|
2019-10-12 16:29:10 +03:00
|
|
|
else:
|
2020-06-14 06:35:43 +03:00
|
|
|
print("---", feature, "support ok", file=out)
|
2019-05-25 21:11:33 +03:00
|
|
|
else:
|
|
|
|
print("***", feature, "support not installed", file=out)
|
|
|
|
print("-" * 68, file=out)
|
|
|
|
|
2019-10-12 17:01:18 +03:00
|
|
|
if supported_formats:
|
2019-09-27 01:09:04 +03:00
|
|
|
extensions = collections.defaultdict(list)
|
|
|
|
for ext, i in Image.EXTENSION.items():
|
|
|
|
extensions[i].append(ext)
|
|
|
|
|
|
|
|
for i in sorted(Image.ID):
|
2020-07-16 12:43:29 +03:00
|
|
|
line = f"{i}"
|
2019-09-27 01:09:04 +03:00
|
|
|
if i in Image.MIME:
|
2020-07-16 12:43:29 +03:00
|
|
|
line = f"{line} {Image.MIME[i]}"
|
2019-09-27 01:09:04 +03:00
|
|
|
print(line, file=out)
|
|
|
|
|
|
|
|
if i in extensions:
|
|
|
|
print(
|
|
|
|
"Extensions: {}".format(", ".join(sorted(extensions[i]))), file=out
|
|
|
|
)
|
|
|
|
|
|
|
|
features = []
|
|
|
|
if i in Image.OPEN:
|
|
|
|
features.append("open")
|
|
|
|
if i in Image.SAVE:
|
|
|
|
features.append("save")
|
|
|
|
if i in Image.SAVE_ALL:
|
|
|
|
features.append("save_all")
|
|
|
|
if i in Image.DECODERS:
|
|
|
|
features.append("decode")
|
|
|
|
if i in Image.ENCODERS:
|
|
|
|
features.append("encode")
|
|
|
|
|
|
|
|
print("Features: {}".format(", ".join(features)), file=out)
|
|
|
|
print("-" * 68, file=out)
|