From 3c06e95e36648e5f0fa213bb62b07b06fe7f4129 Mon Sep 17 00:00:00 2001
From: ouhammou rachid <93659459+ouhammmourachid@users.noreply.github.com>
Date: Mon, 28 Oct 2024 18:14:56 +0100
Subject: [PATCH] Fix import error message while import mermaid in env without
Ipython (#206)
* Fix error while import mermaid in env without Ipython
Related to #201
Change the error handling code in `mermaid/__main__.py` to display a warning instead of an error.
* Update the error message to "Warning: IPython is not installed. Mermaidjs magic function is not available."
---
For more details, open the [Copilot Workspace session](https://copilot-workspace.githubnext.com/ouhammmourachid/mermaid-py/issues/201?shareId=XXXX-XXXX-XXXX-XXXX).
* [pre-commit.ci] auto fixes from pre-commit.com hooks
for more information, see https://pre-commit.ci
---------
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
---
mermaid/__main__.py | 355 ++++++++++++++++++++++----------------------
1 file changed, 178 insertions(+), 177 deletions(-)
diff --git a/mermaid/__main__.py b/mermaid/__main__.py
index cceed5b..35d480a 100644
--- a/mermaid/__main__.py
+++ b/mermaid/__main__.py
@@ -1,177 +1,178 @@
-import base64
-from enum import Enum
-from pathlib import Path
-from typing import Optional, Union
-from urllib.parse import urlencode
-
-import requests
-from requests import Response
-
-from .graph import Graph
-
-
-class Position(Enum):
- """
- This class represents the position of the node in a Mermaid diagram.
- """
-
- LEFT = "left"
- RIGHT = "right"
- CENTER = "center"
- NONE = "none"
-
-
-class Mermaid:
- """
- This class represents a Mermaid diagram.
-
- Attributes:
- _diagram (str): The base64 encoded string of the Mermaid diagram script.
- svg_response (Response): The response from the GET request to the Mermaid SVG API.
- img_response (Response): The response from the GET request to the Mermaid IMG API.
- """
-
- def __init__(
- self,
- graph: Graph,
- width: Optional[int] = None,
- height: Optional[int] = None,
- scale: Optional[float] = None,
- position: Union[Position, str] = Position.NONE,
- ):
- """
- The constructor for the Mermaid class.
-
- Parameters:
- graph (Graph): The Graph object containing the Mermaid diagram script.
- width (Optional[int]): The width of the SVG image.
- height (Optional[int]): The height of the SVG image.
- scale (Optional[float]): The scale of the SVG image.
- Must be an float between 1 and 3, and one of height or width must be provided.
- position (Union[Position, str]): The position of the node in the Mermaid diagram.
- """
- if scale:
- assert 1 <= scale <= 3, "Scale must be between 1 and 3"
- assert any(
- [width, height]
- ), "One or both of width and height must be provided"
-
- self.__position: str = position if isinstance(position, str) else position.value
- self.__height = height if height else None
- self.__width = width if width else None
- self.__scale = scale if scale else None
-
- self._diagram = self._process_diagram(graph.script)
-
- if any([self.__width, self.__height, self.__scale]):
- self._diagram += "?" + self._build_query_params()
- self._make_request_to_mermaid()
-
- def _build_query_params(self) -> str:
- """Build the query parameters for the Mermaid API request."""
- params = {
- param: value
- for param, value in [
- ("width", self.__width),
- ("height", self.__height),
- ("scale", self.__scale),
- ]
- if value
- }
-
- return urlencode(params, doseq=True)
-
- def set_position(self, position: Union[Position, str]) -> None:
- """
- Set the position of the node in the Mermaid diagram.
-
- Parameters:
- position (Union[Position, str]): The position of the node.
- """
- self.__position = position if isinstance(position, str) else position.value
-
- @staticmethod
- def _process_diagram(diagram: str) -> str:
- """
- Process the Mermaid diagram script into a base64 encoded string.
-
- Parameters:
- diagram (str): The Mermaid diagram script.
-
- Returns:
- str: The base64 encoded string of the Mermaid diagram script.
- """
- graphbytes = diagram.encode("utf8")
- base64_bytes = base64.b64encode(graphbytes)
- diagram = base64_bytes.decode("ascii")
- return diagram
-
- def _repr_html_(self) -> str:
- """
- Return the text of the SVG response.
-
- Returns:
- str: The text of the SVG response.
- """
- if self.__position == Position.NONE.value:
- return self.svg_response.text
- return (
- f'
{self.svg_response.text}
'
- )
-
- def _make_request_to_mermaid(self) -> None:
- """
- Make GET requests to the Mermaid SVG and IMG APIs using
- the base64 encoded string of the Mermaid diagram script.
- """
-
- self.svg_response: Response = requests.get(
- "https://mermaid.ink/svg/" + self._diagram
- )
- self.img_response: Response = requests.get(
- "https://mermaid.ink/img/" + self._diagram
- )
-
- def to_svg(self, path: Union[str, Path]) -> None:
- """
- Write the SVG response text to a file.
-
- Parameters:
- path (Union[str, Path]): The path of the file to write to.
- """
- with open(path, "w", encoding="utf-8") as file:
- file.write(self.svg_response.text)
-
- def to_png(self, path: Union[str, Path]) -> None:
- """
- Write the IMG response content to a file.
-
- Parameters:
- path (Union[str, Path]): The path of the file to write to.
- """
- with open(path, "wb") as file:
- file.write(self.img_response.content)
-
-
-try:
- from IPython import get_ipython
-
- if get_ipython() is not None:
- from IPython.core.magic import register_cell_magic
- from IPython.display import Image, display
-
- @register_cell_magic
- def mermaidjs(line, cell):
- options = line.strip().split()
- script: str = cell.strip()
- graph: Graph = Graph("mermaid diagram", script)
- mermaid: Mermaid = Mermaid(graph)
- if "--img" in options:
- display(Image(mermaid.img_response.content))
- else:
- display(Mermaid(graph))
-
- get_ipython().register_magic_function(mermaidjs, magic_kind="cell")
-except ImportError:
- # TODO: add a suitible handler for exception.
- print("Error acured while importing mermaidjs .")
+import base64
+from enum import Enum
+from pathlib import Path
+from typing import Optional, Union
+from urllib.parse import urlencode
+
+import requests
+from requests import Response
+
+from .graph import Graph
+
+
+class Position(Enum):
+ """
+ This class represents the position of the node in a Mermaid diagram.
+ """
+
+ LEFT = "left"
+ RIGHT = "right"
+ CENTER = "center"
+ NONE = "none"
+
+
+class Mermaid:
+ """
+ This class represents a Mermaid diagram.
+
+ Attributes:
+ _diagram (str): The base64 encoded string of the Mermaid diagram script.
+ svg_response (Response): The response from the GET request to the Mermaid SVG API.
+ img_response (Response): The response from the GET request to the Mermaid IMG API.
+ """
+
+ def __init__(
+ self,
+ graph: Graph,
+ width: Optional[int] = None,
+ height: Optional[int] = None,
+ scale: Optional[float] = None,
+ position: Union[Position, str] = Position.NONE,
+ ):
+ """
+ The constructor for the Mermaid class.
+
+ Parameters:
+ graph (Graph): The Graph object containing the Mermaid diagram script.
+ width (Optional[int]): The width of the SVG image.
+ height (Optional[int]): The height of the SVG image.
+ scale (Optional[float]): The scale of the SVG image.
+ Must be an float between 1 and 3, and one of height or width must be provided.
+ position (Union[Position, str]): The position of the node in the Mermaid diagram.
+ """
+ if scale:
+ assert 1 <= scale <= 3, "Scale must be between 1 and 3"
+ assert any(
+ [width, height]
+ ), "One or both of width and height must be provided"
+
+ self.__position: str = position if isinstance(position, str) else position.value
+ self.__height = height if height else None
+ self.__width = width if width else None
+ self.__scale = scale if scale else None
+
+ self._diagram = self._process_diagram(graph.script)
+
+ if any([self.__width, self.__height, self.__scale]):
+ self._diagram += "?" + self._build_query_params()
+ self._make_request_to_mermaid()
+
+ def _build_query_params(self) -> str:
+ """Build the query parameters for the Mermaid API request."""
+ params = {
+ param: value
+ for param, value in [
+ ("width", self.__width),
+ ("height", self.__height),
+ ("scale", self.__scale),
+ ]
+ if value
+ }
+
+ return urlencode(params, doseq=True)
+
+ def set_position(self, position: Union[Position, str]) -> None:
+ """
+ Set the position of the node in the Mermaid diagram.
+
+ Parameters:
+ position (Union[Position, str]): The position of the node.
+ """
+ self.__position = position if isinstance(position, str) else position.value
+
+ @staticmethod
+ def _process_diagram(diagram: str) -> str:
+ """
+ Process the Mermaid diagram script into a base64 encoded string.
+
+ Parameters:
+ diagram (str): The Mermaid diagram script.
+
+ Returns:
+ str: The base64 encoded string of the Mermaid diagram script.
+ """
+ graphbytes = diagram.encode("utf8")
+ base64_bytes = base64.b64encode(graphbytes)
+ diagram = base64_bytes.decode("ascii")
+ return diagram
+
+ def _repr_html_(self) -> str:
+ """
+ Return the text of the SVG response.
+
+ Returns:
+ str: The text of the SVG response.
+ """
+ if self.__position == Position.NONE.value:
+ return self.svg_response.text
+ return (
+ f'{self.svg_response.text}
'
+ )
+
+ def _make_request_to_mermaid(self) -> None:
+ """
+ Make GET requests to the Mermaid SVG and IMG APIs using
+ the base64 encoded string of the Mermaid diagram script.
+ """
+
+ self.svg_response: Response = requests.get(
+ "https://mermaid.ink/svg/" + self._diagram
+ )
+ self.img_response: Response = requests.get(
+ "https://mermaid.ink/img/" + self._diagram
+ )
+
+ def to_svg(self, path: Union[str, Path]) -> None:
+ """
+ Write the SVG response text to a file.
+
+ Parameters:
+ path (Union[str, Path]): The path of the file to write to.
+ """
+ with open(path, "w", encoding="utf-8") as file:
+ file.write(self.svg_response.text)
+
+ def to_png(self, path: Union[str, Path]) -> None:
+ """
+ Write the IMG response content to a file.
+
+ Parameters:
+ path (Union[str, Path]): The path of the file to write to.
+ """
+ with open(path, "wb") as file:
+ file.write(self.img_response.content)
+
+
+try:
+ from IPython import get_ipython
+
+ if get_ipython() is not None:
+ from IPython.core.magic import register_cell_magic
+ from IPython.display import Image, display
+
+ @register_cell_magic
+ def mermaidjs(line, cell):
+ options = line.strip().split()
+ script: str = cell.strip()
+ graph: Graph = Graph("mermaid diagram", script)
+ mermaid: Mermaid = Mermaid(graph)
+ if "--img" in options:
+ display(Image(mermaid.img_response.content))
+ else:
+ display(Mermaid(graph))
+
+ get_ipython().register_magic_function(mermaidjs, magic_kind="cell")
+except ImportError:
+ print(
+ "Warning: IPython is not installed. Mermaidjs magic function is not available."
+ )