generate_pyi.py 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. # Copyright 2021-2024 Avaiga Private Limited
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
  4. # the License. You may obtain a copy of the License at
  5. #
  6. # http://www.apache.org/licenses/LICENSE-2.0
  7. #
  8. # Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
  9. # an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
  10. # specific language governing permissions and limitations under the License.
  11. import json
  12. import os
  13. import re
  14. import sys
  15. from typing import Any, Dict, List
  16. from markdownify import markdownify
  17. __RE_INDEXED_PROPERTY = re.compile(r"^([\w_]+)\[(<\w+>)?([\w]+)(</\w+>)?\]$")
  18. # Make sure we can import the mandatory packages
  19. script_dir = os.path.dirname(os.path.realpath(__file__))
  20. if not os.path.isdir(os.path.abspath(os.path.join(script_dir, "taipy"))):
  21. sys.path.append(os.path.abspath(os.path.join(script_dir, os.pardir, os.pardir)))
  22. # ##################################################################################################
  23. # Generate gui pyi file (gui/gui.pyi)
  24. # ##################################################################################################
  25. gui_py_file = "./taipy/gui/gui.py"
  26. gui_pyi_file = f"{gui_py_file}i"
  27. from taipy.config import Config # noqa: E402
  28. # Generate Python interface definition files
  29. os.system(f"pipenv run stubgen {gui_py_file} --no-import --parse-only --export-less -o ./")
  30. gui_config = "".join(
  31. (
  32. f", {k}: {v.__name__} = ..."
  33. if "<class" in str(v)
  34. else f", {k}: {str(v).replace('typing', 't').replace('taipy.gui.config.', '')} = ..."
  35. )
  36. for k, v in Config.__annotations__.items()
  37. )
  38. replaced_content = ""
  39. with open(gui_pyi_file, "r") as file:
  40. for line in file:
  41. if "def run(" in line:
  42. replace_str = line[line.index(", run_server") : (line.index("**kwargs") + len("**kwargs"))]
  43. # ", run_server: bool = ..., run_in_thread: bool = ..., async_mode: str = ..., **kwargs"
  44. line = line.replace(replace_str, gui_config)
  45. replaced_content += line
  46. with open(gui_pyi_file, "w") as write_file:
  47. write_file.write(replaced_content)
  48. # ##################################################################################################
  49. # Generate Page Builder pyi file (gui/builder/__init__.pyi)
  50. # ##################################################################################################
  51. # Read the version
  52. current_version = "latest"
  53. with open("./taipy/gui/version.json", "r") as vfile:
  54. version = json.load(vfile)
  55. if "dev" in version.get("ext", ""):
  56. current_version = "develop"
  57. else:
  58. current_version = f'release-{version.get("major", 0)}.{version.get("minor", 0)}'
  59. taipy_doc_url = f"https://docs.taipy.io/en/{current_version}/manuals/userman/gui/viselements/generic/"
  60. builder_py_file = "./taipy/gui/builder/__init__.py"
  61. builder_pyi_file = f"{builder_py_file}i"
  62. with open("./taipy/gui/viselements.json", "r") as file:
  63. viselements = json.load(file)
  64. os.system(f"pipenv run stubgen {builder_py_file} --no-import --parse-only --export-less -o ./")
  65. with open(builder_pyi_file, "a") as file:
  66. file.write("from datetime import datetime\n")
  67. file.write("from typing import Any, Callable, Union\n")
  68. file.write("\n")
  69. file.write("from .. import Icon\n")
  70. file.write("from ._element import _Block, _Control\n")
  71. def resolve_inherit(name: str, properties, inherits, viselements) -> List[Dict[str, Any]]:
  72. if not inherits:
  73. return properties
  74. for inherit_name in inherits:
  75. inherited_desc = next((e for e in viselements["undocumented"] if e[0] == inherit_name), None)
  76. if inherited_desc is None:
  77. inherited_desc = next((e for e in viselements["blocks"] if e[0] == inherit_name), None)
  78. if inherited_desc is None:
  79. inherited_desc = next((e for e in viselements["controls"] if e[0] == inherit_name), None)
  80. if inherited_desc is None:
  81. raise RuntimeError(f"Element type '{name}' inherits from unknown element type '{inherit_name}'")
  82. inherited_desc = inherited_desc[1]
  83. for inherit_prop in inherited_desc["properties"]:
  84. prop_desc = next((p for p in properties if p["name"] == inherit_prop["name"]), None)
  85. if prop_desc: # Property exists
  86. def override(current, inherits, p: str):
  87. if p not in current and (inherited := inherits.get(p, None)):
  88. current[p] = inherited
  89. override(prop_desc, inherit_prop, "type")
  90. override(prop_desc, inherit_prop, "default_value")
  91. override(prop_desc, inherit_prop, "doc")
  92. override(prop_desc, inherit_prop, "signature")
  93. else:
  94. properties.append(inherit_prop)
  95. properties = resolve_inherit(inherit_name, properties, inherited_desc.get("inherits", None), viselements)
  96. return properties
  97. def format_as_parameter(property):
  98. type = property["type"]
  99. if m := re.match(r"indexed\((.*)\)", type):
  100. type = m[1]
  101. property["indexed"] = " (indexed)"
  102. else:
  103. property["indexed"] = ""
  104. if m := re.match(r"dynamic\((.*)\)", type):
  105. type = m[1]
  106. property["dynamic"] = " (dynamic)"
  107. else:
  108. property["dynamic"] = ""
  109. name = property["name"]
  110. if match := __RE_INDEXED_PROPERTY.match(name):
  111. name = f"{match.group(1)}__{match.group(3)}"
  112. if type == "Callback" or type == "Function":
  113. type = ""
  114. else:
  115. type = f": {type}"
  116. default_value = property.get("default_value", None)
  117. if default_value is not None:
  118. try:
  119. eval(default_value)
  120. default_value = f" = {default_value}"
  121. except Exception:
  122. default_value = ""
  123. else:
  124. default_value = ""
  125. return f"{name}{type}{default_value}"
  126. def build_doc(name: str, desc: Dict[str, Any]):
  127. if "doc" not in desc:
  128. return ""
  129. doc = str(desc["doc"])
  130. # Hack to replace the actual element name in the class_name property doc
  131. if desc["name"] == "class_name":
  132. doc = doc.replace("[element_type]", name)
  133. # This won't work for Scenario Management and Block elements
  134. doc = re.sub(r"(href=\")\.\.((?:.*?)\")", r"\1" + taipy_doc_url + name + r"/../..\2", doc)
  135. doc = re.sub(r"<tt>([\w_]+)</tt>", r"`\1`", doc) # <tt> not processed properly by markdownify()
  136. doc = "\n ".join(markdownify(doc).split("\n"))
  137. # <, >, `, [, -, _ and * prefixed with a \
  138. doc = doc.replace(" \n", " \\n").replace("\\<", "<").replace("\\>", ">").replace("\\`", "`")
  139. doc = doc.replace("\\[", "[").replace("\\-", "-").replace("\\_", "_").replace("\\*", "*")
  140. # Final dots are prefixed with a \
  141. doc = re.sub(r"\\.$", ".", doc)
  142. # Link anchors # signs are prefixed with a \
  143. doc = re.sub(r"\\(#[a-z_]+\))", r"\1", doc)
  144. doc = re.sub(r"(?:\s+\\n)?\s+See below(?:[^\.]*)?\.", "", doc).replace("\n", "\\n")
  145. return f"{desc['name']}{desc['dynamic']}{desc['indexed']}\\n {doc}\\n\\n"
  146. element_template = """
  147. class {{name}}(_{{base_class}}):
  148. _ELEMENT_NAME: str
  149. def __init__(self, {{properties_decl}}) -> None:
  150. \"\"\"Creates a{{n}} {{name}} element.\\n\\nParameters\\n----------\\n\\n{{properties_doc}}\"\"\" # noqa: E501
  151. ...
  152. """
  153. def generate_elements(category: str, base_class: str):
  154. for element in viselements[category]:
  155. name = element[0]
  156. desc = element[1]
  157. properties_doc = ""
  158. property_list: List[Dict[str, Any]] = []
  159. property_names: List[str] = []
  160. properties = resolve_inherit(name, desc["properties"], desc.get("inherits", None), viselements)
  161. # Remove hidden properties
  162. properties = [p for p in properties if not p.get("hide", False)]
  163. # Generate function parameters
  164. properties_decl = [format_as_parameter(p) for p in properties]
  165. # Generate properties doc
  166. for property in properties:
  167. if "default_property" in property and property["default_property"] is True:
  168. property_list.insert(0, property)
  169. property_names.insert(0, property["name"])
  170. continue
  171. property_list.append(property)
  172. property_names.append(property["name"])
  173. # Append properties doc to element doc (once ordered)
  174. for property in property_list:
  175. property_doc = build_doc(name, property)
  176. properties_doc += property_doc
  177. if len(properties_decl) > 1:
  178. properties_decl.insert(1, "*")
  179. # Append element to __init__.pyi
  180. with open(builder_pyi_file, "a") as file:
  181. n = "n" if name[0] in ["a", "e", "i", "o"] else ""
  182. file.write(
  183. element_template.replace("{{name}}", name)
  184. .replace("{{n}}", n)
  185. .replace("{{base_class}}", base_class)
  186. .replace("{{properties_decl}}", ", ".join(properties_decl))
  187. .replace("{{properties_doc}}", properties_doc)
  188. )
  189. generate_elements("controls", "Control")
  190. generate_elements("blocks", "Block")
  191. os.system(f"pipenv run isort {gui_pyi_file}")
  192. os.system(f"pipenv run black {gui_pyi_file}")
  193. os.system(f"pipenv run isort {builder_pyi_file}")
  194. os.system(f"pipenv run black {builder_pyi_file}")