penpot-mcp/python-scripts/prepare_api_docs.py
2026-01-21 17:12:29 +01:00

265 lines
10 KiB
Python

import collections
import dataclasses
import os
from dataclasses import dataclass
from io import StringIO
from pathlib import Path
import requests
from bs4 import BeautifulSoup, Tag
from markdownify import MarkdownConverter
from ruamel.yaml import YAML
from ruamel.yaml.scalarstring import LiteralScalarString
from sensai.util import logging
import sys
log = logging.getLogger(__name__)
DEFAULT_API_DOCS_URL="https://penpot-plugins-api-doc.pages.dev"
class PenpotAPIContentMarkdownConverter(MarkdownConverter):
"""
Markdown converter for Penpot API docs, specifically for the .col-content element
(and sub-elements thereof)
"""
def process_tag(self, node, parent_tags=None):
soup = BeautifulSoup(str(node), "html.parser")
# skip breadcrumbs
if "class" in node.attrs and "tsd-breadcrumb" in node.attrs["class"]:
return ""
# convert h5 and h4 to plain text
if node.name in ["h5", "h4"]:
return soup.get_text()
text = soup.get_text()
# convert tsd-tag code elements (containing e.g. "Readonly" and "Optional" designations)
# If we encounter them at this level, we just remove them, as they are redundant.
# The significant such tags are handled in the tsd-signature processing below.
if node.name == "code" and "class" in node.attrs and "tsd-tag" in node.attrs["class"]:
return ""
# skip buttons (e.g. "Copy")
if node.name == "button":
return ""
# skip links to definitions in <li> elements
if node.name == "li" and text.startswith("Defined in"):
return ""
# for links, just return the text
if node.name == "a":
return text
# skip inheritance information
if node.name == "p" and text.startswith("Inherited from"):
return ""
# remove index with links
if "class" in node.attrs and "tsd-index-content" in node.attrs["class"]:
return ""
# convert <pre> blocks to markdown code blocks
if node.name == "pre":
for button in soup.find_all("button"):
button.decompose()
return f"\n```\n{soup.get_text().strip()}\n```\n\n"
# convert tsd-signature elements to code blocks, converting <br> to newlines
if "class" in node.attrs and "tsd-signature" in node.attrs["class"]:
# convert <br> to newlines
for br in soup.find_all("br"):
br.replace_with("\n")
# process tsd-tags (keeping only "readonly"; optional is redundant, as it is indicated via "?")
for tag in soup.find_all(attrs={"class": "tsd-tag"}):
tag_lower = tag.get_text().strip().lower()
if tag_lower in ["readonly"]:
tag.replace_with(f"{tag_lower} ")
else:
tag.decompose()
# return as code block
return f"\n```\n{soup.get_text()}\n```\n\n"
# other cases: use the default processing
return super().process_tag(node, parent_tags=parent_tags)
@dataclass
class TypeInfo:
overview: str
"""
the main type information, which contains all the declarations/signatures but no descriptions
"""
members: dict[str, dict[str, str]]
"""
mapping from member type (e.g. "Properties", "Methods") to a mapping of member name to markdown description
"""
def add_referencing_types(self, referencing_types: set[str]):
if referencing_types:
self.overview += "\n\nReferenced by: " + ", ".join(sorted(referencing_types))
class YamlConverter:
"""Converts dictionaries to YAML with all strings in block literal style"""
def __init__(self):
self.yaml = YAML()
self.yaml.preserve_quotes = True
self.yaml.width = 4096 # Prevent line wrapping
def _convert_strings_to_block(self, obj):
if isinstance(obj, dict):
return {k: self._convert_strings_to_block(v) for k, v in obj.items()}
elif isinstance(obj, list):
return [self._convert_strings_to_block(item) for item in obj]
elif isinstance(obj, str):
return LiteralScalarString(obj)
else:
return obj
def to_yaml(self, data):
processed_data = self._convert_strings_to_block(data)
stream = StringIO()
self.yaml.dump(processed_data, stream)
return stream.getvalue()
def to_file(self, data, filepath):
processed_data = self._convert_strings_to_block(data)
with open(filepath, 'w', encoding='utf-8') as f:
self.yaml.dump(processed_data, f)
class PenpotAPIDocsProcessor:
def __init__(self, url=None):
self.md_converter = PenpotAPIContentMarkdownConverter()
self.base_url = DEFAULT_API_DOCS_URL
self.types: dict[str, TypeInfo] = {}
self.type_referenced_by: dict[str, set[str]] = collections.defaultdict(set)
def run(self, target_dir: str):
os.makedirs(target_dir, exist_ok=True)
# find links to all interfaces and types
modules_page = self._fetch("modules")
soup = BeautifulSoup(modules_page, "html.parser")
content = soup.find(attrs={"class": "col-content"})
links = content.find_all("a", href=True)
# process each link, converting interface and type pages to markdown
for link in links:
href = link['href']
if href.startswith("interfaces/") or href.startswith("types/"):
type_name = href.split("/")[-1].replace(".html", "")
log.info("Processing page: %s", type_name)
type_info = self.process_page(href, type_name)
self.types[type_name] = type_info
# add type reference information
for type_name, type_info in self.types.items():
referencing_types = self.type_referenced_by.get(type_name, set())
type_info.add_referencing_types(referencing_types)
# save to yaml
yaml_path = os.path.join(target_dir, "api_types.yml")
log.info("Writing API type information to %s", yaml_path)
data_dict = {k: dataclasses.asdict(v) for k, v in self.types.items()}
YamlConverter().to_file(data_dict, yaml_path)
def _fetch(self, rel_url: str) -> str:
response = requests.get(f"{self.base_url}/{rel_url}")
if response.status_code != 200:
raise Exception(f"Failed to retrieve page: {response.status_code}")
html_content = response.text
return html_content
def _html_to_markdown(self, html_content: str) -> str:
md = self.md_converter.convert(html_content)
md = md.replace("\xa0", " ") # replace non-breaking spaces
return md.strip()
def process_page(self, rel_url: str, type_name: str) -> TypeInfo:
html_content = self._fetch(rel_url)
soup = BeautifulSoup(html_content, "html.parser")
content = soup.find(attrs={"class": "col-content"})
# full_text = self._html_to_markdown(str(content))
# extract individual members
members = {}
member_group_tags = []
for el in content.children:
if isinstance(el, Tag):
if "class" in el.attrs and "tsd-member-group" in el.attrs["class"]:
member_group_tags.append(el)
members_type = el.find("h2").get_text().strip()
members_in_group = {}
members[members_type] = members_in_group
for member_tag in el.find_all(attrs={"class": "tsd-member"}):
member_anchor = member_tag.find("a", attrs={"class": "tsd-anchor"}, recursive=False)
member_name = member_anchor.attrs["id"]
member_heading = member_tag.find("h3")
# extract tsd-tag info (e.g., "Readonly") from the heading and reinsert it into the signature,
# where we want to see it. The heading is removed, as it is redundant.
if member_heading:
tags_in_heading = member_heading.find_all(attrs={"class": "tsd-tag"})
if tags_in_heading:
signature_tag = member_tag.find(attrs={"class": "tsd-signature"})
if signature_tag:
for tag in reversed(tags_in_heading):
signature_tag.insert(0, tag)
member_heading.decompose()
# convert to markdown
tag_text = str(member_tag)
members_in_group[member_name] = self._html_to_markdown(tag_text)
# record references to other types in signature
signature = content.find("div", attrs={"class": "tsd-signature"})
for link_to_type in signature.find_all("a", attrs={"class": "tsd-signature-type"}):
referenced_type_name = link_to_type.get_text().strip()
self.type_referenced_by[referenced_type_name].add(type_name)
# remove the member groups from the soup
for tag in member_group_tags:
tag.decompose()
# overview is what remains in content after removing member groups
overview = self._html_to_markdown(str(content))
return TypeInfo(
overview=overview,
members=members
)
def main():
target_dir = Path(__file__).parent.parent / "mcp-server" / "data"
url = sys.argv[1] if len(sys.argv) > 1 else DEFAULT_API_DOCS_URL
print("Fetching plugin data from: {}".format(url))
PenpotAPIDocsProcessor(url).run(target_dir=str(target_dir))
def debug_type_conversion(rel_url: str):
"""
This function is for debugging purposes only.
It processes a single type page and prints the converted markdown to the console.
:param rel_url: relative URL of the type page (e.g., "interfaces/ShapeBase")
"""
type_name = rel_url.split("/")[-1]
processor = PenpotAPIDocsProcessor()
type_info = processor.process_page(rel_url, type_name)
print(f"--- overview ---\n{type_info.overview}\n")
for member_type, members in type_info.members.items():
print(f"\n{member_type}:")
for member_name, member_md in members.items():
print(f"--- {member_name} ---\n{member_md}\n")
if __name__ == '__main__':
# debug_type_conversion("interfaces/ShapeBase")
logging.run_main(main)