Skip to content

Commit

Permalink
Merge branch 'main' into diagrams-support
Browse files Browse the repository at this point in the history
  • Loading branch information
JakubAndrysek committed Apr 3, 2024
2 parents 9f7f677 + 62afbe4 commit 2728b1f
Show file tree
Hide file tree
Showing 25 changed files with 382 additions and 58 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/deploy-web.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,4 @@ jobs:
python -m pip install -e ".[dev]"
sudo apt-get install doxygen
- name: Build docs
run: mkdocs build --clean --verbose
run: mkdocs gh-deploy --force
28 changes: 28 additions & 0 deletions .github/workflows/mkdoxy-pytest-repo.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
name: MkDoxy Pytest MkDoxy with all supported Python versions

on: [push, pull_request]

jobs:
build:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-latest] # windows-latest, macos-latest requires doxygen to be installed manually
python-version: ["3.9", "3.10", "3.11", "3.12"]

steps:
- uses: actions/checkout@v3
- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
with:
python-version: ${{ matrix.python-version }}
- name: Display Python version
run: python -c "import sys; print(sys.version)"
- name: Install dependencies
run: |
python -m pip install --upgrade pip
python -m pip install -e ".[dev]"
sudo apt-get install doxygen
- name: Run Pytest
run: |
pytest tests/
2 changes: 1 addition & 1 deletion .github/workflows/mkdoxy-test-demos.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ jobs:
strategy:
matrix:
os: [ubuntu-latest] # windows-latest, macos-latest requires doxygen to be installed manually
python-version: ["3.9", "3.10", "3.11"]
python-version: ["3.9", "3.10", "3.11", "3.12"]

steps:
- uses: actions/checkout@v3
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/mkdoxy-test-repo.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ jobs:
strategy:
matrix:
os: [ubuntu-latest] # windows-latest, macos-latest requires doxygen to be installed manually
python-version: ["3.9", "3.10", "3.11"]
python-version: ["3.9", "3.10", "3.11", "3.12"]

steps:
- uses: actions/checkout@v3
Expand Down
25 changes: 23 additions & 2 deletions .idea/workspace.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

7 changes: 7 additions & 0 deletions demo-projects/animal/Doxyfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
DOXYFILE_ENCODING = UTF-8
GENERATE_XML = YES
RECURSIVE = YES
EXAMPLE_PATH = examples
SHOW_NAMESPACES = YES
GENERATE_HTML = NO
GENERATE_LATEX = YES
18 changes: 15 additions & 3 deletions docs/usage/advanced.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
## Disabling the plugin
You can use the `enabled` option to optionally disable this plugin. A possible use case is local development where you might want faster build times.

```yaml
```yaml hl_lines="3"
plugins:
- mkdoxy:
enabled: !ENV [ENABLE_MKDOXY, True]
Expand All @@ -25,15 +25,27 @@ mkdocs serve
## Configure custom Doxygen binary

By default, the plugin will use the `doxygen` binary from the system path. You can configure a custom binary using the `doxygen-bin-path` option.
If `doxygen-bin-path` is not found, the plugin will raise DoxygenBinPathNotValid exception.

- addad by [thb-sb](https://github.com/thb-sb)

```yaml
```yaml hl_lines="3"
plugins:
- mkdoxy:
doxygen-bin-path: /path/to/doxygen
...
```

## Configure custom Doxygen configuration file
If you want to use a standard Doxygen configuration file, you can specify the path to the file using the `doxygen-config` option in the plugin configuration.

Hi, I have released new version, please try it and let me know if it has been fixed. Thanks
```yaml hl_lines="6"
plugins:
- mkdoxy:
projects:
myProjectCpp:
src-dirs: ...
doxy-cfg-file: path/to/Doxyfile # relative path to the Doxygen configuration file (relative to the mkdocs.yml file)
doxy-cfg: # standard doxygen configuration (key: value)
FILE_PATTERNS: ... # other configuration options - merge (this will override the configuration from the Doxyfile)
```
3 changes: 3 additions & 0 deletions mkdocs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ plugins:
src-dirs: mkdoxy
full-doc: True
# template-dir: templates-custom
doxy-cfg-file: demo-projects/animal/Doxyfile
doxy-cfg:
FILE_PATTERNS: "*.py"
EXAMPLE_PATH: ""
Expand Down Expand Up @@ -109,6 +110,8 @@ markdown_extensions:
- pymdownx.highlight
- pymdownx.superfences
- def_list
- toc:
permalink: True
- pymdownx.superfences
- admonition
- pymdownx.details
Expand Down
147 changes: 126 additions & 21 deletions mkdoxy/doxyrun.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import hashlib
import logging
import os
import shutil

from pathlib import Path, PurePath
from subprocess import PIPE, Popen
Expand All @@ -19,6 +21,7 @@ def __init__(
doxygenSource: str,
tempDoxyFolder: str,
doxyCfgNew,
doxyConfigFile: Optional[str] = None,
runPath: Optional[str] = None,
):
"""! Constructor.
Expand All @@ -38,29 +41,92 @@ def __init__(
@param doxygenBinPath: (str) Path to the Doxygen binary.
@param doxygenSource: (str) Source files for Doxygen.
@param tempDoxyFolder: (str) Temporary folder for Doxygen.
@param doxyConfigFile: (str) Path to a Doxygen config file.
@param doxyCfgNew: (dict) New Doxygen config options that will be added to the default config (new options will overwrite default options)
""" # noqa: E501

if not self.is_doxygen_valid_path(doxygenBinPath):
raise DoxygenBinPathNotValid(
f"Invalid Doxygen binary path: {doxygenBinPath}\n"
f"Make sure Doxygen is installed and the path is correct.\n"
f"Look at https://mkdoxy.kubaandrysek.cz/usage/advanced/#configure-custom-doxygen-binary."
)

self.doxygenBinPath: str = doxygenBinPath
self.doxygenSource: str = doxygenSource
self.tempDoxyFolder: str = tempDoxyFolder
self.doxyCfgNew: dict = doxyCfgNew
self.doxyConfigFile: Optional[str] = doxyConfigFile
self.hashFileName: str = "hashChanges.yaml"
self.hashFilePath: PurePath = PurePath.joinpath(Path(self.tempDoxyFolder), Path(self.hashFileName))
self.runPath: Optional[str] = runPath
self.doxyCfg: dict = self.setDoxyCfg(doxyCfgNew)

def setDoxyCfg(self, doxyCfgNew: dict) -> dict:
"""! Set the Doxygen configuration.
@details If a custom Doxygen config file is provided, it will be used. Otherwise, default options will be used.
@details Order of application of parameters:
@details 1. Custom Doxygen config file
@details 2. If not provided, default options - in documentation
@details 3. New Doxygen config options from mkdocs.yml
@details 3. Overwrite INPUT and OUTPUT_DIRECTORY with the provided values for correct plugin operation.
@details Overwrite options description:
@details - INPUT: <doxygenSource>
@details - OUTPUT_DIRECTORY: <tempDoxyFolder>
@details Default Doxygen config options:
@details - DOXYFILE_ENCODING: UTF-8
@details - GENERATE_XML: YES
@details - RECURSIVE: YES
@details - EXAMPLE_PATH: examples
@details - SHOW_NAMESPACES: YES
@details - GENERATE_HTML: NO
@details - GENERATE_LATEX: NO
@param doxyCfgNew: (dict) New Doxygen config options that will be
added to the default config (new options will overwrite default options)
@return: (dict) Doxygen configuration.
"""
doxyCfg = {}

if self.doxyConfigFile is not None and self.doxyConfigFile != "":
try:
with open(self.doxyConfigFile, "r") as file:
doxyCfg.update(self.str2dox_dict(file.read()))
except FileNotFoundError as e:
raise DoxygenCustomConfigNotFound(
f"Custom Doxygen config file not found: {self.doxyConfigFile}\n"
f"Make sure the path is correct."
f"Look at https://mkdoxy.kubaandrysek.cz/usage/advanced/#configure-custom-doxygen-configuration-file."
) from e
else:
doxyCfg = {
"DOXYFILE_ENCODING": "UTF-8",
"GENERATE_XML": "YES",
"RECURSIVE": "YES",
"EXAMPLE_PATH": "examples",
"SHOW_NAMESPACES": "YES",
"GENERATE_HTML": "NO",
"GENERATE_LATEX": "NO",
}

self.doxyCfg: dict = {
"INPUT": self.doxygenSource,
"OUTPUT_DIRECTORY": self.tempDoxyFolder,
"DOXYFILE_ENCODING": "UTF-8",
"GENERATE_XML": "YES",
"RECURSIVE": "YES",
"SHOW_NAMESPACES": "YES",
"GENERATE_HTML": "NO",
"GENERATE_LATEX": "NO",
}
doxyCfg.update(doxyCfgNew)
doxyCfg["INPUT"] = self.doxygenSource
doxyCfg["OUTPUT_DIRECTORY"] = self.tempDoxyFolder
return doxyCfg

def is_doxygen_valid_path(self, doxygen_bin_path: str) -> bool:
"""! Check if the Doxygen binary path is valid.
@details Accepts a full path or just 'doxygen' if it exists in the system's PATH.
@param doxygen_bin_path: (str) The path to the Doxygen binary or just 'doxygen'.
@return: (bool) True if the Doxygen binary path is valid, False otherwise.
"""
# If the path is just 'doxygen', search for it in the system's PATH
if doxygen_bin_path.lower() == "doxygen":
return shutil.which("doxygen") is not None

self.doxyCfg.update(self.doxyCfgNew)
self.doxyCfgStr: str = self.dox_dict2str(self.doxyCfg)
# Use pathlib to check if the provided full path is a file and executable
path = Path(doxygen_bin_path)
return path.is_file() and os.access(path, os.X_OK)

# Source of dox_dict2str: https://xdress-fabio.readthedocs.io/en/latest/_modules/xdress/doxygen.html#XDressPlugin
def dox_dict2str(self, dox_dict: dict) -> str:
Expand All @@ -84,17 +150,43 @@ def dox_dict2str(self, dox_dict: dict) -> str:
# Don't need an empty line at the end
return s.strip()

def hasChanged(self):
def str2dox_dict(self, dox_str: str) -> dict:
"""! Convert a string from a doxygen config file to a dictionary.
@details
@param dox_str: (str) String from a doxygen config file.
@return: (dict) Dictionary.
"""
dox_dict = {}
try:
for line in dox_str.split("\n"):
if line.strip() == "":
continue
key, value = line.split(" = ")
if value == "YES":
dox_dict[key] = True
elif value == "NO":
dox_dict[key] = False
else:
dox_dict[key] = value
except ValueError as e:
raise DoxygenCustomConfigNotValid(
f"Invalid custom Doxygen config file: {self.doxyConfigFile}\n"
f"Make sure the file is in standard Doxygen format."
f"Look at https://mkdoxy.kubaandrysek.cz/usage/advanced/."
) from e
return dox_dict

def hasChanged(self) -> bool:
"""! Check if the source files have changed since the last run.
@details
@return: (bool) True if the source files have changed since the last run.
"""

def heshWrite(filename: str, hash: str):
def hashWrite(filename: PurePath, hash: str):
with open(filename, "w") as file:
file.write(hash)

def hashRead(filename: str) -> str:
def hashRead(filename: PurePath) -> str:
with open(filename, "r") as file:
return str(file.read())

Expand All @@ -104,7 +196,7 @@ def hashRead(filename: str) -> str:
for path in Path(src).rglob("*.*"):
# # Code from https://stackoverflow.com/a/22058673/15411117
# # BUF_SIZE is totally arbitrary, change for your app!
BUF_SIZE = 65536 # lets read stuff in 64kb chunks!
BUF_SIZE = 65536 # let's read stuff in 64kb chunks!
if path.is_file():
with open(path, "rb") as f:
while True:
Expand All @@ -114,13 +206,13 @@ def hashRead(filename: str) -> str:
sha1.update(data)
# print(f"{path}: {sha1.hexdigest()}")

hahsNew = sha1.hexdigest()
hashNew = sha1.hexdigest()
if Path(self.hashFilePath).is_file():
hashOld = hashRead(self.hashFilePath)
if hahsNew == hashOld:
if hashNew == hashOld:
return False

heshWrite(self.hashFilePath, hahsNew)
hashWrite(self.hashFilePath, hashNew)
return True

def run(self):
Expand All @@ -134,7 +226,7 @@ def run(self):
stderr=PIPE,
cwd=self.runPath,
)
(doxyBuilder.communicate(self.doxyCfgStr.encode("utf-8"))[0].decode().strip())
(doxyBuilder.communicate(self.dox_dict2str(self.doxyCfg).encode("utf-8"))[0].decode().strip())
# log.info(self.destinationDir)
# log.info(stdout_data)

Expand All @@ -155,3 +247,16 @@ def getOutputFolder(self) -> PurePath:
@return: (PurePath) Path to the XML output folder.
"""
return Path.joinpath(Path(self.tempDoxyFolder), Path("xml"))


# not valid path exception
class DoxygenBinPathNotValid(Exception):
pass


class DoxygenCustomConfigNotFound(Exception):
pass


class DoxygenCustomConfigNotValid(Exception):
pass
Loading

0 comments on commit 2728b1f

Please sign in to comment.