diff --git a/.circleci/config.yml b/.circleci/config.yml index 3b36cc48022..ac8cce9b8fe 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -117,6 +117,35 @@ commands: source .venv/bin/activate python -m pytest -x test_init/test_lazy_imports.py + test_io_kaleido_v0: + steps: + - checkout + - browser-tools/install-chrome + - browser-tools/install-chromedriver + - run: + name: Install dependencies + command: | + curl -LsSf https://astral.sh/uv/install.sh | sh + uv venv + source .venv/bin/activate + uv pip install . + uv pip install -r ./test_requirements/requirements_optional.txt + # Install Kaleido v0 instead of the v1 specified in requirements_optional.txt + uv pip uninstall kaleido + uv pip install kaleido==0.2.1 + - run: + name: List installed packages and python version + command: | + source .venv/bin/activate + uv pip list + python --version + - run: + name: Test plotly.io image output with Kaleido v0 + command: | + source .venv/bin/activate + python -m pytest tests/test_optional/test_kaleido + no_output_timeout: 20m + jobs: check-code-formatting: docker: @@ -166,6 +195,17 @@ jobs: pandas_version: <> numpy_version: <> + test_kaleido_v0: + parameters: + python_version: + default: "3.12" + type: string + executor: + name: docker-container + python_version: <> + steps: + - test_io_kaleido_v0 + # Percy python_311_percy: docker: @@ -448,5 +488,10 @@ workflows: python_version: "3.9" pandas_version: "1.2.4" numpy_version: "1.26.4" + - test_kaleido_v0: + matrix: + parameters: + python_version: + - "3.12" - python_311_percy - build-doc diff --git a/CHANGELOG.md b/CHANGELOG.md index 89da1e55f17..8b00373c3a0 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,10 +2,15 @@ All notable changes to this project will be documented in this file. This project adheres to [Semantic Versioning](http://semver.org/). -## Unreleased +## [6.1.0b0] - 2025-03-31 + +### Updated +- Add support for Kaleido >= v1.0.0 for image generation, and deprecate support for Kaleido<1 and Orca [[#5062](https://github.com/plotly/plotly.py/pull/5062)] ### Fixed -- Fix third-party widget display issues in v6 [[#5102]https://github.com/plotly/plotly.py/pull/5102] +- Fix third-party widget display issues in v6 [[#5102](https://github.com/plotly/plotly.py/pull/5102)] +- Add handling for case where `jupyterlab` or `notebook` is not installed [[#5104](https://github.com/plotly/plotly.py/pull/5104/files)] +- Fix issue causing Plotly.js script to be embedded multiple times in Jupyter notebooks [[#5112](https://github.com/plotly/plotly.py/pull/5112)] ## [6.0.1] - 2025-03-14 diff --git a/plotly/basedatatypes.py b/plotly/basedatatypes.py index a3044f6763a..805ecf97162 100644 --- a/plotly/basedatatypes.py +++ b/plotly/basedatatypes.py @@ -3718,23 +3718,29 @@ def to_image(self, *args, **kwargs): - 'webp' - 'svg' - 'pdf' - - 'eps' (Requires the poppler library to be installed) + - 'eps' (deprecated) (Requires the poppler library to be installed) - If not specified, will default to `plotly.io.config.default_format` + If not specified, will default to: + - `plotly.io.defaults.default_format` if engine is "kaleido" + - `plotly.io.orca.config.default_format` if engine is "orca" (deprecated) width: int or None The width of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the width of the exported image in physical pixels. - If not specified, will default to `plotly.io.config.default_width` + If not specified, will default to: + - `plotly.io.defaults.default_width` if engine is "kaleido" + - `plotly.io.orca.config.default_width` if engine is "orca" (deprecated) height: int or None The height of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the height of the exported image in physical pixels. - If not specified, will default to `plotly.io.config.default_height` + If not specified, will default to: + - `plotly.io.defaults.default_height` if engine is "kaleido" + - `plotly.io.orca.config.default_height` if engine is "orca" (deprecated) scale: int or float or None The scale factor to use when exporting the figure. A scale factor @@ -3742,17 +3748,20 @@ def to_image(self, *args, **kwargs): to the figure's layout pixel dimensions. Whereas as scale factor of less than 1.0 will decrease the image resolution. - If not specified, will default to `plotly.io.config.default_scale` + If not specified, will default to: + - `plotly.io.defaults.default_scale` if engine is "kaliedo" + - `plotly.io.orca.config.default_scale` if engine is "orca" (deprecated) validate: bool True if the figure should be validated before being converted to an image, False otherwise. - engine: str - Image export engine to use: - - "kaleido": Use Kaleido for image export - - "orca": Use Orca for image export - - "auto" (default): Use Kaleido if installed, otherwise use orca + engine (deprecated): str + Image export engine to use. This parameter is deprecated and Orca engine support will be + dropped in the next major Plotly version. Until then, the following values are supported: + - "kaleido": Use Kaleido for image export + - "orca": Use Orca for image export + - "auto" (default): Use Kaleido if installed, otherwise use Orca Returns ------- @@ -3760,6 +3769,26 @@ def to_image(self, *args, **kwargs): The image data """ import plotly.io as pio + from plotly.io.kaleido import ( + kaleido_available, + kaleido_major, + KALEIDO_DEPRECATION_MSG, + ORCA_DEPRECATION_MSG, + ENGINE_PARAM_DEPRECATION_MSG, + ) + + if ( + kwargs.get("engine", None) in {None, "auto", "kaleido"} + and kaleido_available() + and kaleido_major() < 1 + ): + warnings.warn(KALEIDO_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + if kwargs.get("engine", None) == "orca": + warnings.warn(ORCA_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + if kwargs.get("engine", None): + warnings.warn( + ENGINE_PARAM_DEPRECATION_MSG, DeprecationWarning, stacklevel=2 + ) return pio.to_image(self, *args, **kwargs) @@ -3781,25 +3810,31 @@ def write_image(self, *args, **kwargs): - 'webp' - 'svg' - 'pdf' - - 'eps' (Requires the poppler library to be installed) + - 'eps' (deprecated) (Requires the poppler library to be installed) If not specified and `file` is a string then this will default to the file extension. If not specified and `file` is not a string then this - will default to `plotly.io.config.default_format` + will default to: + - `plotly.io.defaults.default_format` if engine is "kaleido" + - `plotly.io.orca.config.default_format` if engine is "orca" (deprecated) width: int or None The width of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the width of the exported image in physical pixels. - If not specified, will default to `plotly.io.config.default_width` + If not specified, will default to: + - `plotly.io.defaults.default_width` if engine is "kaleido" + - `plotly.io.orca.config.default_width` if engine is "orca" (deprecated) height: int or None The height of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the height of the exported image in physical pixels. - If not specified, will default to `plotly.io.config.default_height` + If not specified, will default to: + - `plotly.io.defaults.default_height` if engine is "kaleido" + - `plotly.io.orca.config.default_height` if engine is "orca" (deprecated) scale: int or float or None The scale factor to use when exporting the figure. A scale factor @@ -3807,23 +3842,46 @@ def write_image(self, *args, **kwargs): to the figure's layout pixel dimensions. Whereas as scale factor of less than 1.0 will decrease the image resolution. - If not specified, will default to `plotly.io.config.default_scale` + If not specified, will default to: + - `plotly.io.defaults.default_scale` if engine is "kaleido" + - `plotly.io.orca.config.default_scale` if engine is "orca" (deprecated) validate: bool True if the figure should be validated before being converted to an image, False otherwise. - engine: str - Image export engine to use: - - "kaleido": Use Kaleido for image export - - "orca": Use Orca for image export - - "auto" (default): Use Kaleido if installed, otherwise use orca + engine (deprecated): str + Image export engine to use. This parameter is deprecated and Orca engine support will be + dropped in the next major Plotly version. Until then, the following values are supported: + - "kaleido": Use Kaleido for image export + - "orca": Use Orca for image export + - "auto" (default): Use Kaleido if installed, otherwise use Orca + Returns ------- None """ import plotly.io as pio + from plotly.io.kaleido import ( + kaleido_available, + kaleido_major, + KALEIDO_DEPRECATION_MSG, + ORCA_DEPRECATION_MSG, + ENGINE_PARAM_DEPRECATION_MSG, + ) + if ( + kwargs.get("engine", None) in {None, "auto", "kaleido"} + and kaleido_available() + and kaleido_major() < 1 + ): + warnings.warn(KALEIDO_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + if kwargs.get("engine", None) == "orca": + warnings.warn(ORCA_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + if kwargs.get("engine", None): + warnings.warn( + ENGINE_PARAM_DEPRECATION_MSG, DeprecationWarning, stacklevel=2 + ) return pio.write_image(self, *args, **kwargs) # Static helpers diff --git a/plotly/io/__init__.py b/plotly/io/__init__.py index ef5b5ea05c7..539313ce7cc 100644 --- a/plotly/io/__init__.py +++ b/plotly/io/__init__.py @@ -3,7 +3,13 @@ from typing import TYPE_CHECKING if sys.version_info < (3, 7) or TYPE_CHECKING: - from ._kaleido import to_image, write_image, full_figure_for_development + from ._kaleido import ( + to_image, + write_image, + to_images, + write_images, + full_figure_for_development, + ) from . import orca, kaleido from . import json from ._json import to_json, from_json, read_json, write_json @@ -11,10 +17,13 @@ from ._html import to_html, write_html from ._renderers import renderers, show from . import base_renderers + from ._defaults import defaults __all__ = [ "to_image", "write_image", + "to_images", + "write_images", "orca", "json", "to_json", @@ -29,6 +38,7 @@ "show", "base_renderers", "full_figure_for_development", + "defaults", ] else: __all__, __getattr__, __dir__ = relative_import( @@ -37,6 +47,8 @@ [ "._kaleido.to_image", "._kaleido.write_image", + "._kaleido.to_images", + "._kaleido.write_images", "._kaleido.full_figure_for_development", "._json.to_json", "._json.from_json", @@ -48,6 +60,7 @@ "._html.write_html", "._renderers.renderers", "._renderers.show", + "._defaults.defaults", ], ) diff --git a/plotly/io/_defaults.py b/plotly/io/_defaults.py new file mode 100644 index 00000000000..84426e87d8d --- /dev/null +++ b/plotly/io/_defaults.py @@ -0,0 +1,16 @@ +# Default settings for image generation + + +class _Defaults(object): + """ + Class to store default settings for image generation. + """ + + def __init__(self): + self.default_format = "png" + self.default_width = 700 + self.default_height = 500 + self.default_scale = 1 + + +defaults = _Defaults() diff --git a/plotly/io/_kaleido.py b/plotly/io/_kaleido.py index 029b79f1029..9f7ba604b85 100644 --- a/plotly/io/_kaleido.py +++ b/plotly/io/_kaleido.py @@ -1,29 +1,153 @@ import os import json from pathlib import Path +import importlib.metadata as importlib_metadata +from packaging.version import Version +import warnings + import plotly -from plotly.io._utils import validate_coerce_fig_to_dict +from plotly.io._utils import validate_coerce_fig_to_dict, as_individual_args +from plotly.io import defaults -try: - from kaleido.scopes.plotly import PlotlyScope +ENGINE_SUPPORT_TIMELINE = "September 2025" - scope = PlotlyScope() - # Compute absolute path to the 'plotly/package_data/' directory - root_dir = os.path.dirname(os.path.abspath(plotly.__file__)) - package_dir = os.path.join(root_dir, "package_data") - scope.plotlyjs = os.path.join(package_dir, "plotly.min.js") - if scope.mathjax is None: - scope.mathjax = ( - "https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.5/MathJax.js" - ) -except ImportError: +# TODO: Remove --pre flag once Kaleido v1 full release is available +KALEIDO_DEPRECATION_MSG = f""" +Support for Kaleido versions less than 1.0.0 is deprecated and will be removed after {ENGINE_SUPPORT_TIMELINE}. +Please upgrade Kaleido to version 1.0.0 or greater (`pip install --upgrade --pre kaleido` or `pip install plotly[kaleido]`). +""" +ORCA_DEPRECATION_MSG = f""" +Support for the Orca engine is deprecated and will be removed after {ENGINE_SUPPORT_TIMELINE}. +Please install Kaleido (`pip install --upgrade --pre kaleido` or `pip install plotly[kaleido]`) to use the Kaleido engine. +""" +ENGINE_PARAM_DEPRECATION_MSG = f""" +Support for the 'engine' argument is deprecated and will be removed after {ENGINE_SUPPORT_TIMELINE}. +Kaleido will be the only supported engine at that time. +""" + +_KALEIDO_AVAILABLE = None +_KALEIDO_MAJOR = None + +kaleido_scope_default_warning_func = ( + lambda x: f""" +Use of plotly.io.kaleido.scope.{x} is deprecated and support will be removed after {ENGINE_SUPPORT_TIMELINE}. +Please use plotly.io.defaults.{x} instead. +""" +) +bad_attribute_error_msg_func = ( + lambda x: f""" +Attribute plotly.io.defaults.{x} is not valid. +Also, use of plotly.io.kaleido.scope.* is deprecated and support will be removed after {ENGINE_SUPPORT_TIMELINE}. +Please use plotly.io.defaults.* instead. +""" +) + + +def kaleido_available(): + global _KALEIDO_AVAILABLE + global _KALEIDO_MAJOR + if _KALEIDO_AVAILABLE is not None: + return _KALEIDO_AVAILABLE + try: + import kaleido + + _KALEIDO_AVAILABLE = True + except ImportError as e: + _KALEIDO_AVAILABLE = False + return _KALEIDO_AVAILABLE + + +def kaleido_major(): + global _KALEIDO_MAJOR + if _KALEIDO_MAJOR is not None: + return _KALEIDO_MAJOR + if not kaleido_available(): + raise ValueError("Kaleido is not installed.") + else: + _KALEIDO_MAJOR = Version(importlib_metadata.version("kaleido")).major + return _KALEIDO_MAJOR + + +try: + if kaleido_available() and kaleido_major() < 1: + # Kaleido v0 + import kaleido + from kaleido.scopes.plotly import PlotlyScope + + # Show a deprecation warning if the old method of setting defaults is used + class PlotlyScopeWithDeprecationWarnings(PlotlyScope): + def __setattr__(self, name, value): + if name in defaults.__dict__: + warnings.warn( + kaleido_scope_default_warning_func(name), + DeprecationWarning, + stacklevel=2, + ) + setattr(defaults, name, value) + super(PlotlyScopeWithDeprecationWarnings, self).__setattr__(name, value) + + def __getattr__(self, name): + if name in defaults.__dict__: + warnings.warn( + kaleido_scope_default_warning_func(name), + DeprecationWarning, + stacklevel=2, + ) + return super(PlotlyScopeWithDeprecationWarnings, self).__getattr__(name) + + scope = PlotlyScopeWithDeprecationWarnings() + # Compute absolute path to the 'plotly/package_data/' directory + root_dir = os.path.dirname(os.path.abspath(plotly.__file__)) + package_dir = os.path.join(root_dir, "package_data") + scope.plotlyjs = os.path.join(package_dir, "plotly.min.js") + if scope.mathjax is None: + scope.mathjax = ( + "https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.5/MathJax.js" + ) + else: + # Kaleido v1 + import kaleido + + # Show a deprecation warning if the old method of setting defaults is used + class DefaultsDeprecationWarning: + def __getattr__(self, name): + if name in defaults.__dict__: + warnings.warn( + kaleido_scope_default_warning_func(name), + DeprecationWarning, + stacklevel=2, + ) + return getattr(defaults, name) + else: + raise AttributeError(bad_attribute_error_msg_func(name)) + + def __setattr__(self, name, value): + if name in defaults.__dict__: + warnings.warn( + kaleido_scope_default_warning_func(name), + DeprecationWarning, + stacklevel=2, + ) + setattr(defaults, name, value) + else: + raise AttributeError(bad_attribute_error_msg_func(name)) + + scope = DefaultsDeprecationWarning() + +except ImportError as e: PlotlyScope = None scope = None def to_image( - fig, format=None, width=None, height=None, scale=None, validate=True, engine="auto" + fig, + format=None, + width=None, + height=None, + scale=None, + validate=True, + engine=None, ): """ Convert a figure to a static image bytes string @@ -35,16 +159,16 @@ def to_image( format: str or None The desired image format. One of - - 'png' - - 'jpg' or 'jpeg' - - 'webp' - - 'svg' - - 'pdf' - - 'eps' (Requires the poppler library to be installed and on the PATH) + - 'png' + - 'jpg' or 'jpeg' + - 'webp' + - 'svg' + - 'pdf' + - 'eps' (deprecated) (Requires the poppler library to be installed and on the PATH) If not specified, will default to: - - `plotly.io.kaleido.scope.default_format` if engine is "kaleido" - - `plotly.io.orca.config.default_format` if engine is "orca" + - `plotly.io.defaults.default_format` if engine is "kaleido" + - `plotly.io.orca.config.default_format` if engine is "orca" (deprecated) width: int or None The width of the exported image in layout pixels. If the `scale` @@ -52,8 +176,8 @@ def to_image( in physical pixels. If not specified, will default to: - - `plotly.io.kaleido.scope.default_width` if engine is "kaleido" - - `plotly.io.orca.config.default_width` if engine is "orca" + - `plotly.io.defaults.default_width` if engine is "kaleido" + - `plotly.io.orca.config.default_width` if engine is "orca" (deprecated) height: int or None The height of the exported image in layout pixels. If the `scale` @@ -61,8 +185,8 @@ def to_image( in physical pixels. If not specified, will default to: - - `plotly.io.kaleido.scope.default_height` if engine is "kaleido" - - `plotly.io.orca.config.default_height` if engine is "orca" + - `plotly.io.defaults.default_height` if engine is "kaleido" + - `plotly.io.orca.config.default_height` if engine is "orca" (deprecated) scale: int or float or None The scale factor to use when exporting the figure. A scale factor @@ -71,29 +195,35 @@ def to_image( less than 1.0 will decrease the image resolution. If not specified, will default to: - - `plotly.io.kaleido.scope.default_scale` if engine is "kaleido" - - `plotly.io.orca.config.default_scale` if engine is "orca" - + - `plotly.io.defaults.default_scale` if engine is "kaliedo" + - `plotly.io.orca.config.default_scale` if engine is "orca" (deprecated) validate: bool True if the figure should be validated before being converted to an image, False otherwise. - engine: str - Image export engine to use: - - "kaleido": Use Kaleido for image export - - "orca": Use Orca for image export - - "auto" (default): Use Kaleido if installed, otherwise use orca + engine (deprecated): str + Image export engine to use. This parameter is deprecated and Orca engine support will be + dropped in the next major Plotly version. Until then, the following values are supported: + - "kaleido": Use Kaleido for image export + - "orca": Use Orca for image export + - "auto" (default): Use Kaleido if installed, otherwise use Orca Returns ------- bytes The image data """ + # Handle engine # ------------- + if engine is not None: + warnings.warn(ENGINE_PARAM_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + else: + engine = "auto" + if engine == "auto": - if scope is not None: + if kaleido_available(): # Default to kaleido if available engine = "kaleido" else: @@ -109,6 +239,7 @@ def to_image( engine = "kaleido" if engine == "orca": + warnings.warn(ORCA_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) # Fall back to legacy orca image export path from ._orca import to_image as to_image_orca @@ -121,14 +252,10 @@ def to_image( validate=validate, ) elif engine != "kaleido": - raise ValueError( - "Invalid image export engine specified: {engine}".format( - engine=repr(engine) - ) - ) + raise ValueError(f"Invalid image export engine specified: {repr(engine)}") # Raise informative error message if Kaleido is not installed - if scope is None: + if not kaleido_available(): raise ValueError( """ Image export using the "kaleido" engine requires the kaleido package, @@ -137,12 +264,50 @@ def to_image( """ ) - # Validate figure - # --------------- + # Convert figure to dict (and validate if requested) fig_dict = validate_coerce_fig_to_dict(fig, validate) - img_bytes = scope.transform( - fig_dict, format=format, width=width, height=height, scale=scale - ) + + # Request image bytes + if kaleido_major() > 0: + # Kaleido v1 + # Check if trying to export to EPS format, which is not supported in Kaleido v1 + if format == "eps": + raise ValueError( + f""" +EPS export is not supported by Kaleido v1. Please use SVG or PDF instead. +You can also downgrade to Kaleido v0, but support for Kaleido v0 will be removed after {ENGINE_SUPPORT_TIMELINE}. +To downgrade to Kaleido v0, run: + $ pip install kaleido<1.0.0 +""" + ) + import choreographer + + try: + # TODO: Refactor to make it possible to use a shared Kaleido instance here + img_bytes = kaleido.calc_fig_sync( + fig_dict, + opts=dict( + format=format or defaults.default_format, + width=width or defaults.default_width, + height=height or defaults.default_height, + scale=scale or defaults.default_scale, + ), + ) + except choreographer.errors.ChromeNotFoundError: + raise RuntimeError( + """ + +Kaleido requires Google Chrome to be installed. Install it by running: + $ plotly_get_chrome +""" + ) + + else: + # Kaleido v0 + warnings.warn(KALEIDO_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + img_bytes = scope.transform( + fig_dict, format=format, width=width, height=height, scale=scale + ) return img_bytes @@ -177,13 +342,13 @@ def write_image( - 'webp' - 'svg' - 'pdf' - - 'eps' (Requires the poppler library to be installed and on the PATH) + - 'eps' (deprecated) (Requires the poppler library to be installed and on the PATH) If not specified and `file` is a string then this will default to the file extension. If not specified and `file` is not a string then this will default to: - - `plotly.io.kaleido.scope.default_format` if engine is "kaleido" - - `plotly.io.orca.config.default_format` if engine is "orca" + - `plotly.io.defaults.default_format` if engine is "kaleido" + - `plotly.io.orca.config.default_format` if engine is "orca" (deprecated) width: int or None The width of the exported image in layout pixels. If the `scale` @@ -191,8 +356,8 @@ def write_image( in physical pixels. If not specified, will default to: - - `plotly.io.kaleido.scope.default_width` if engine is "kaleido" - - `plotly.io.orca.config.default_width` if engine is "orca" + - `plotly.io.defaults.default_width` if engine is "kaleido" + - `plotly.io.orca.config.default_width` if engine is "orca" (deprecated) height: int or None The height of the exported image in layout pixels. If the `scale` @@ -200,8 +365,8 @@ def write_image( in physical pixels. If not specified, will default to: - - `plotly.io.kaleido.scope.default_height` if engine is "kaleido" - - `plotly.io.orca.config.default_height` if engine is "orca" + - `plotly.io.defaults.default_height` if engine is "kaleido" + - `plotly.io.orca.config.default_height` if engine is "orca" (deprecated) scale: int or float or None The scale factor to use when exporting the figure. A scale factor @@ -210,23 +375,37 @@ def write_image( less than 1.0 will decrease the image resolution. If not specified, will default to: - - `plotly.io.kaleido.scope.default_scale` if engine is "kaleido" - - `plotly.io.orca.config.default_scale` if engine is "orca" + - `plotly.io.defaults.default_scale` if engine is "kaleido" + - `plotly.io.orca.config.default_scale` if engine is "orca" (deprecated) validate: bool True if the figure should be validated before being converted to an image, False otherwise. - engine: str - Image export engine to use: - - "kaleido": Use Kaleido for image export - - "orca": Use Orca for image export - - "auto" (default): Use Kaleido if installed, otherwise use orca + engine (deprecated): str + Image export engine to use. This parameter is deprecated and Orca engine support will be + dropped in the next major Plotly version. Until then, the following values are supported: + - "kaleido": Use Kaleido for image export + - "orca": Use Orca for image export + - "auto" (default): Use Kaleido if installed, otherwise use Orca Returns ------- None """ + # Show Kaleido deprecation warning if needed + # ------------------------------------------ + if ( + engine in {None, "auto", "kaleido"} + and kaleido_available() + and kaleido_major() < 1 + ): + warnings.warn(KALEIDO_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + if engine == "orca": + warnings.warn(ORCA_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + if engine not in {None, "auto"}: + warnings.warn(ENGINE_PARAM_DEPRECATION_MSG, DeprecationWarning, stacklevel=2) + # Try to cast `file` as a pathlib object `path`. # ---------------------------------------------- if isinstance(file, str): @@ -248,16 +427,14 @@ def write_image( format = ext.lstrip(".") else: raise ValueError( - """ + f""" Cannot infer image type from output path '{file}'. Please add a file extension or specify the type using the format parameter. For example: >>> import plotly.io as pio >>> pio.write_image(fig, file_path, format='png') -""".format( - file=file - ) +""" ) # Request image @@ -284,11 +461,9 @@ def write_image( except AttributeError: pass raise ValueError( - """ + f""" The 'file' argument '{file}' is not a string, pathlib.Path object, or file descriptor. -""".format( - file=file - ) +""" ) else: # We previously succeeded in interpreting `file` as a pathlib object. @@ -296,6 +471,62 @@ def write_image( path.write_bytes(img_data) +def to_images(*args, **kwargs): + """ + Convert multiple figures to static images and return a list of image bytes + + Parameters + ---------- + Accepts the same parameters as pio.to_image(), but any parameter may be either + a single value or a list of values. If more than one parameter is a list, + all must be the same length. + + Returns + ------- + list of bytes + The image data + """ + individual_args, individual_kwargs = as_individual_args(*args, **kwargs) + + if kaleido_available() and kaleido_major() > 0: + # Kaleido v1 + # TODO: Use a single shared kaleido instance for all images + return [to_image(*a, **kw) for a, kw in zip(individual_args, individual_kwargs)] + else: + # Kaleido v0, or orca + return [to_image(*a, **kw) for a, kw in zip(individual_args, individual_kwargs)] + + +def write_images(*args, **kwargs): + """ + Write multiple images to files or writeable objects. This is much faster than + calling write_image() multiple times. + + Parameters + ---------- + Accepts the same parameters as pio.write_image(), but any parameter may be either + a single value or a list of values. If more than one parameter is a list, + all must be the same length. + + Returns + ------- + None + """ + + # Get individual arguments + individual_args, individual_kwargs = as_individual_args(*args, **kwargs) + + if kaleido_available() and kaleido_major() > 0: + # Kaleido v1 + # TODO: Use a single shared kaleido instance for all images + for a, kw in zip(individual_args, individual_kwargs): + write_image(*a, **kw) + else: + # Kaleido v0, or orca + for a, kw in zip(individual_args, individual_kwargs): + write_image(*a, **kw) + + def full_figure_for_development(fig, warn=True, as_dict=False): """ Compute default values for all attributes not specified in the input figure and @@ -323,7 +554,7 @@ def full_figure_for_development(fig, warn=True, as_dict=False): """ # Raise informative error message if Kaleido is not installed - if scope is None: + if not kaleido_available(): raise ValueError( """ Full figure generation requires the kaleido package, @@ -333,15 +564,28 @@ def full_figure_for_development(fig, warn=True, as_dict=False): ) if warn: - import warnings - warnings.warn( "full_figure_for_development is not recommended or necessary for " "production use in most circumstances. \n" "To suppress this warning, set warn=False" ) - fig = json.loads(scope.transform(fig, format="json").decode("utf-8")) + if kaleido_available() and kaleido_major() > 0: + # Kaleido v1 + bytes = kaleido.calc_fig_sync( + fig, + opts=dict(format="json"), + ) + fig = json.loads(bytes.decode("utf-8")) + else: + # Kaleido v0 + warnings.warn( + f"Support for Kaleido versions less than 1.0.0 is deprecated and will be removed after {ENGINE_SUPPORT_TIMELINE}. " + + "Please upgrade Kaleido to version 1.0.0 or greater (`pip install --upgrade kaleido`).", + DeprecationWarning, + ) + fig = json.loads(scope.transform(fig, format="json").decode("utf-8")) + if as_dict: return fig else: @@ -350,4 +594,32 @@ def full_figure_for_development(fig, warn=True, as_dict=False): return go.Figure(fig, skip_invalid=True) +def get_chrome(): + """ + Install Google Chrome for Kaleido + This function can be run from the command line using the command `plotly_get_chrome` + defined in pyproject.toml + """ + if not kaleido_available() or kaleido_major() < 1: + raise ValueError( + "This command requires Kaleido v1.0.0 or greater. Install it using `pip install kaleido`." + ) + import sys + + cli_yes = len(sys.argv) > 1 and sys.argv[1] == "-y" + if not cli_yes: + print( + "\nPlotly will install a copy of Google Chrome to be used for generating static images of plots.\n" + ) + # TODO: Print path where Chrome will be installed + # print(f"Chrome will be installed at {chrome_download_path}\n") + response = input("Do you want to proceed? [y/n] ") + if not response or response[0].lower() != "y": + print("Cancelled") + return + print("Installing Chrome for Plotly...") + kaleido.get_chrome_sync() + print("Chrome installed successfully.") + + __all__ = ["to_image", "write_image", "scope", "full_figure_for_development"] diff --git a/plotly/io/_utils.py b/plotly/io/_utils.py index 658540ca71a..ea0caed4473 100644 --- a/plotly/io/_utils.py +++ b/plotly/io/_utils.py @@ -43,6 +43,57 @@ def validate_coerce_output_type(output_type): return cls +def as_individual_args(*args, **kwargs): + """ + Given one or more positional or keyword arguments which may be either a single value + or a list of values, return a list of lists and a list of dictionaries + by expanding the single values into lists. + If more than one item in the input is a list, all lists must be the same length. + + Parameters + ---------- + *args: list + The positional arguments + **kwargs: dict + The keyword arguments + + Returns + ------- + list of lists + A list of lists + list of dicts + A list of dictionaries + """ + # Check that all list arguments have the same length, + # and find out what that length is + # If there are no list arguments, length is 1 + list_lengths = [ + len(v) for v in args + tuple(kwargs.values()) if isinstance(v, list) + ] + if list_lengths and len(set(list_lengths)) > 1: + raise ValueError("All list arguments must have the same length.") + list_length = list_lengths[0] if list_lengths else 1 + + # Expand all arguments to lists of the same length + expanded_args = [[v] * list_length if not isinstance(v, list) else v for v in args] + expanded_kwargs = { + k: [v] * list_length if not isinstance(v, list) else v + for k, v in kwargs.items() + } + + # Reshape into a list of lists + # Each list represents the positional arguments for a single function call + list_of_args = [[v[i] for v in expanded_args] for i in range(list_length)] + + # Reshape into a list of dictionaries + # Each dictionary represents the keyword arguments for a single function call + list_of_kwargs = [ + {k: v[i] for k, v in expanded_kwargs.items()} for i in range(list_length) + ] + + return list_of_args, list_of_kwargs + + def plotly_cdn_url(cdn_ver=get_plotlyjs_version()): """Return a valid plotly CDN url.""" return "https://cdn.plot.ly/plotly-{cdn_ver}.min.js".format( diff --git a/plotly/io/kaleido.py b/plotly/io/kaleido.py index c14b315047b..e4df9f53c71 100644 --- a/plotly/io/kaleido.py +++ b/plotly/io/kaleido.py @@ -1 +1,10 @@ -from ._kaleido import to_image, write_image, scope +from ._kaleido import ( + to_image, + write_image, + scope, + kaleido_available, + kaleido_major, + KALEIDO_DEPRECATION_MSG, + ORCA_DEPRECATION_MSG, + ENGINE_PARAM_DEPRECATION_MSG, +) diff --git a/pyproject.toml b/pyproject.toml index 579a914d6a4..a87691ea481 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -38,7 +38,7 @@ classifiers = [ ] requires-python = ">=3.8" license = {file="LICENSE.txt"} -version = "6.0.1" +version = "6.1.0b0" dependencies = [ "narwhals>=1.15.1", "packaging" @@ -46,6 +46,10 @@ dependencies = [ [project.optional-dependencies] express = ["numpy"] +kaleido = ["kaleido==1.0.0rc11"] + +[project.scripts] +plotly_get_chrome = "plotly.io._kaleido:get_chrome" [tool.setuptools.packages.find] where = ["."] diff --git a/test_requirements/requirements_optional.txt b/test_requirements/requirements_optional.txt index a48fe001d56..d98b883afcc 100644 --- a/test_requirements/requirements_optional.txt +++ b/test_requirements/requirements_optional.txt @@ -17,9 +17,11 @@ pyshp matplotlib scikit-image psutil -kaleido +# kaleido>=1.0.0 # Uncomment and delete line below once Kaleido v1 is released +kaleido==1.0.0rc11 orjson polars[timezone] pyarrow plotly-geo vaex;python_version<="3.9" +pdfrw diff --git a/tests/test_optional/test_kaleido/test_kaleido.py b/tests/test_optional/test_kaleido/test_kaleido.py index 263fd85483a..a66c12a7c0b 100644 --- a/tests/test_optional/test_kaleido/test_kaleido.py +++ b/tests/test_optional/test_kaleido/test_kaleido.py @@ -1,66 +1,35 @@ -import plotly.io as pio -import plotly.io.kaleido -from contextlib import contextmanager -from io import BytesIO +from io import BytesIO, StringIO from pathlib import Path -from unittest.mock import Mock - -fig = {"layout": {"title": {"text": "figure title"}}} - - -def make_writeable_mocks(): - """Produce some mocks which we will use for testing the `write_image()` function. - - These mocks should be passed as the `file=` argument to `write_image()`. - - The tests should verify that the method specified in the `active_write_function` - attribute is called once, and that scope.transform is called with the `format=` - argument specified by the `.expected_format` attribute. - - In total we provide two mocks: one for a writable file descriptor, and other for a - pathlib.Path object. - """ - - # Part 1: A mock for a file descriptor - # ------------------------------------ - mock_file_descriptor = Mock() - - # A file descriptor has no write_bytes method, unlike a pathlib Path. - del mock_file_descriptor.write_bytes - - # The expected write method for a file descriptor is .write - mock_file_descriptor.active_write_function = mock_file_descriptor.write - - # Since there is no filename, there should be no format detected. - mock_file_descriptor.expected_format = None - - # Part 2: A mock for a pathlib path - # --------------------------------- - mock_pathlib_path = Mock(spec=Path) - - # A pathlib Path object has no write method, unlike a file descriptor. - del mock_pathlib_path.write - - # The expected write method for a pathlib Path is .write_bytes - mock_pathlib_path.active_write_function = mock_pathlib_path.write_bytes - - # Mock a path with PNG suffix - mock_pathlib_path.suffix = ".png" - mock_pathlib_path.expected_format = "png" +import tempfile +from contextlib import redirect_stdout +import base64 +import unittest +from unittest import mock + +from pdfrw import PdfReader +from PIL import Image +import plotly.io as pio - return mock_file_descriptor, mock_pathlib_path +fig = {"data": [], "layout": {"title": {"text": "figure title"}}} -@contextmanager -def mocked_scope(): - # Code to acquire resource, e.g.: - scope_mock = Mock() - original_scope = pio._kaleido.scope - pio._kaleido.scope = scope_mock - try: - yield scope_mock - finally: - pio._kaleido.scope = original_scope +def check_image(path_or_buffer, size=(700, 500), format="PNG"): + if format == "PDF": + img = PdfReader(path_or_buffer) + # TODO: There is a conversion factor needed here + # In Kaleido v0 the conversion factor is 0.75 + factor = 0.75 + expected_size = tuple(int(s * factor) for s in size) + actual_size = tuple(int(s) for s in img.pages[0].MediaBox[2:]) + assert actual_size == expected_size + else: + if isinstance(path_or_buffer, (str, Path)): + with open(path_or_buffer, "rb") as f: + img = Image.open(f) + else: + img = Image.open(path_or_buffer) + assert img.size == size + assert img.format == format def test_kaleido_engine_to_image_returns_bytes(): @@ -75,80 +44,71 @@ def test_kaleido_fulljson(): def test_kaleido_engine_to_image(): - with mocked_scope() as scope: - pio.to_image(fig, engine="kaleido", validate=False) + bytes = pio.to_image(fig, engine="kaleido", validate=False) - scope.transform.assert_called_with( - fig, format=None, width=None, height=None, scale=None - ) + # Check that image dimensions match default dimensions (700x500) + # and format is default format (png) + check_image(BytesIO(bytes)) -def test_kaleido_engine_write_image(): - for writeable_mock in make_writeable_mocks(): - with mocked_scope() as scope: - pio.write_image(fig, writeable_mock, engine="kaleido", validate=False) +def test_kaleido_engine_write_image(tmp_path): + path_str = tempfile.mkstemp(suffix=".png", dir=tmp_path)[1] + path_path = Path(tempfile.mkstemp(suffix=".png", dir=tmp_path)[1]) - scope.transform.assert_called_with( - fig, - format=writeable_mock.expected_format, - width=None, - height=None, - scale=None, - ) - - assert writeable_mock.active_write_function.call_count == 1 + for out_path in [path_str, path_path]: + pio.write_image(fig, out_path, engine="kaleido", validate=False) + check_image(out_path) def test_kaleido_engine_to_image_kwargs(): - with mocked_scope() as scope: - pio.to_image( + bytes = pio.to_image( + fig, + format="pdf", + width=700, + height=600, + scale=2, + engine="kaleido", + validate=False, + ) + check_image(BytesIO(bytes), size=(700 * 2, 600 * 2), format="PDF") + + +def test_kaleido_engine_write_image_kwargs(tmp_path): + path_str = tempfile.mkstemp(suffix=".png", dir=tmp_path)[1] + path_path = Path(tempfile.mkstemp(suffix=".png", dir=tmp_path)[1]) + + for out_path in [path_str, path_path]: + pio.write_image( fig, - format="pdf", + out_path, + format="jpg", width=700, height=600, scale=2, engine="kaleido", validate=False, ) - - scope.transform.assert_called_with( - fig, format="pdf", width=700, height=600, scale=2 - ) - - -def test_kaleido_engine_write_image_kwargs(): - for writeable_mock in make_writeable_mocks(): - with mocked_scope() as scope: - pio.write_image( - fig, - writeable_mock, - format="jpg", - width=700, - height=600, - scale=2, - engine="kaleido", - validate=False, - ) - - scope.transform.assert_called_with( - fig, format="jpg", width=700, height=600, scale=2 - ) - - assert writeable_mock.active_write_function.call_count == 1 + check_image(out_path, size=(700 * 2, 600 * 2), format="JPEG") def test_image_renderer(): - with mocked_scope() as scope: - pio.show(fig, renderer="svg", engine="kaleido", validate=False) - - renderer = pio.renderers["svg"] - scope.transform.assert_called_with( - fig, - format="svg", - width=None, - height=None, - scale=renderer.scale, + """Verify that the image renderer returns the expected mimebundle.""" + with redirect_stdout(StringIO()) as f: + pio.show(fig, renderer="png", engine="kaleido", validate=False) + mimebundle = f.getvalue().strip() + mimebundle_expected = str( + { + "image/png": base64.b64encode( + pio.to_image( + fig, + format="png", + engine="kaleido", + validate=False, + ) + ).decode("utf8") + } ) + assert mimebundle == mimebundle_expected def test_bytesio(): @@ -163,3 +123,144 @@ def test_bytesio(): bio_bytes = bio.read() to_image_bytes = pio.to_image(fig, format="jpg", engine="kaleido", validate=False) assert bio_bytes == to_image_bytes + + +@mock.patch("plotly.io._kaleido.to_image") +def test_to_images_single(mock_to_image): + """Test to_images with a single figure""" + pio.to_images( + fig, + format="png", + width=800, + height=600, + scale=2, + validate=True, + ) + + # Verify that to_image was called once with the correct arguments + expected_calls = [ + mock.call( + fig, + format="png", + width=800, + height=600, + scale=2, + validate=True, + ) + ] + mock_to_image.assert_has_calls(expected_calls, any_order=False) + assert mock_to_image.call_count == 1 + + +@mock.patch("plotly.io._kaleido.to_image") +def test_to_images_multiple(mock_to_image): + """Test to_images with lists""" + fig1 = {"data": [], "layout": {"title": {"text": "figure 1"}}} + fig2 = {"data": [], "layout": {"title": {"text": "figure 2"}}} + pio.to_images( + [fig1, fig2], + "png", + width=[800, 400], + height=600, + scale=[1, 2], + validate=True, + ) + + # Verify that to_image was called with the correct arguments in the correct order + expected_calls = [ + mock.call( + fig1, + "png", + width=800, + height=600, + scale=1, + validate=True, + ), + mock.call( + fig2, + "png", + width=400, + height=600, + scale=2, + validate=True, + ), + ] + mock_to_image.assert_has_calls(expected_calls, any_order=False) + assert mock_to_image.call_count == 2 + + +@mock.patch("plotly.io._kaleido.write_image") +def test_write_images_single(mock_write_image): + """Test write_images with only single arguments""" + pio.write_images( + fig, + "output.png", + format="png", + width=800, + height=600, + scale=2, + ) + + # Verify that write_image was called once with the correct arguments + expected_calls = [ + mock.call( + fig, + "output.png", + format="png", + width=800, + height=600, + scale=2, + ) + ] + mock_write_image.assert_has_calls(expected_calls, any_order=False) + assert mock_write_image.call_count == 1 + + +@mock.patch("plotly.io._kaleido.write_image") +def test_write_images_multiple(mock_write_image): + """Test write_images with list arguments""" + fig1 = {"data": [], "layout": {"title": {"text": "figure 1"}}} + fig2 = {"data": [], "layout": {"title": {"text": "figure 2"}}} + pio.write_images( + [fig1, fig2], + ["output1.png", "output2.jpg"], + format=["png", "jpeg"], + width=800, + height=[600, 400], + scale=2, + ) + + # Verify that write_image was called with the correct arguments in the correct order + expected_calls = [ + mock.call( + fig1, + "output1.png", + format="png", + width=800, + height=600, + scale=2, + ), + mock.call( + fig2, + "output2.jpg", + format="jpeg", + width=800, + height=400, + scale=2, + ), + ] + mock_write_image.assert_has_calls(expected_calls, any_order=False) + assert mock_write_image.call_count == 2 + + +def test_defaults(): + """Test that image output defaults can be set using pio.defaults.*""" + try: + assert pio.defaults.default_format == "png" + pio.defaults.default_format = "svg" + assert pio.defaults.default_format == "svg" + result = pio.to_image(fig, format="svg", validate=False) + assert result.startswith(b"