Skip to content

Commit

Permalink
Run formatting, linting and tests in CI
Browse files Browse the repository at this point in the history
To flag any inconsistencies in style or regressions in the test suite,
ensure we run Black, Pylint and pytest for every pull request and push
to main.
  • Loading branch information
mudge committed Aug 14, 2024
1 parent 278bc08 commit 7ef0d58
Show file tree
Hide file tree
Showing 10 changed files with 91 additions and 18 deletions.
22 changes: 22 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
name: Test

on:
workflow_dispatch:
push:
branches:
- main
pull_request:

jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-python@v5
with:
python-version: '3.9'
cache: 'pip'
- run: pip install -r requirements.txt
- run: black --check .
- run: pylint doxygentoasciidoc
- run: pytest
8 changes: 8 additions & 0 deletions .pylintrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
[MESSAGES CONTROL]
disable=missing-module-docstring,missing-function-docstring,missing-class-docstring,too-many-lines

[FORMAT]
good-names=id

[MASTER]
ignore=tests
20 changes: 20 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,23 @@ Allowed args:
`-c`: process a node other than `doxygenindex`

The following attributes from the XML will be preserved in the generated asciidoc: role, tag, type.

## Development

Run the test suite:

```console
$ pytest
```

Ensure code is formatted consistently:

```console
$ black --check .
```

Ensure code passes linting:

```console
$ pylint doxygentoasciidoc
```
17 changes: 13 additions & 4 deletions cli.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
import os
import sys
import argparse
import yaml

from bs4 import BeautifulSoup
from .nodes import Node, DoxygenindexNode
Expand All @@ -10,9 +9,19 @@
def main():
"""Convert the given Doxygen index.xml to AsciiDoc and output the result."""
parser = argparse.ArgumentParser()
parser.add_argument("-f", "--file", help="The path of the file to convert", default=None)
parser.add_argument("-o", "--output", help="The path of the output file", default=None)
parser.add_argument("-c", "--child", help="Is NOT the root index file", default=False, action='store_true')
parser.add_argument(
"-f", "--file", help="The path of the file to convert", default=None
)
parser.add_argument(
"-o", "--output", help="The path of the output file", default=None
)
parser.add_argument(
"-c",
"--child",
help="Is NOT the root index file",
default=False,
action="store_true",
)
args = parser.parse_args()
filename = args.file
output_filename = args.output
Expand Down
12 changes: 6 additions & 6 deletions helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,16 +22,16 @@ def title(text, level, attributes=None):
"""Return text formatted as a title with the given level."""
if level > 5:
if attributes is not None:
if re.search(r'([,\s]role=)', attributes) is not None:
attributes = re.sub(r'([,\s]role=)(.*?[,\s]?$)', "\\1h6 \\2", attributes)
if re.search(r"([,\s]role=)", attributes) is not None:
attributes = re.sub(
r"([,\s]role=)(.*?[,\s]?$)", "\\1h6 \\2", attributes
)
else:
attributes += ",role=h6"
return f"[{attributes}]\n*{escape_text(text)}*"
else:
return f"[.h6]\n*{escape_text(text)}*"
return f"[.h6]\n*{escape_text(text)}*"

marker = "=" * (level + 1)
if attributes is not None:
return f"[{attributes}]\n{marker} {escape_text(text)}"
else:
return f"{marker} {escape_text(text)}"
return f"{marker} {escape_text(text)}"
14 changes: 10 additions & 4 deletions nodes.py
Original file line number Diff line number Diff line change
Expand Up @@ -163,8 +163,10 @@ def children(self, selector=None, **kwargs):
for position, child in enumerate(children)
]

def attributes(self, skip=[], **kwargs):
def attributes(self, skip=None):
"""Return an asciidoc string of any attributes specified, plus the node id."""
if skip is None:
skip = []
preserved_attributes = ["role", "tag", "type"]
atts = []
if self.node.get("id", None) is not None and "id" not in skip:
Expand Down Expand Up @@ -395,6 +397,7 @@ def to_asciidoc_row(self, depth=0):

class GroupNode(Node):
def to_asciidoc(self, **kwargs):
# pylint: disable=too-many-locals,too-many-branches
output = [self.__output_title(**kwargs)]
briefdescription = self.__output_briefdescription(**kwargs)
if briefdescription:
Expand Down Expand Up @@ -990,7 +993,8 @@ def to_asciidoc(self, **kwargs):
)
else:
output.append(
f"[.memname]`#define {escape_text(name)}{escape_text(argsstring)} {escape_text(initializer).rstrip()}`"
f"[.memname]`#define {escape_text(name)}{escape_text(argsstring)} "
f"{escape_text(initializer).rstrip()}`"
)
else:
output.append(
Expand Down Expand Up @@ -1063,7 +1067,8 @@ def to_asciidoc(self, **kwargs):
for memberdef in self.children("memberdef", kind="typedef"):
type_ = memberdef.child("type").to_asciidoc(**kwargs)
typedef = [
f"`typedef {type_} <<{memberdef.id},{escape_text(memberdef.text('name'))}>>{escape_text(memberdef.text('argsstring'))}`::"
f"`typedef {type_} <<{memberdef.id},{escape_text(memberdef.text('name'))}>>"
f"{escape_text(memberdef.text('argsstring'))}`::"
]
briefdescription = memberdef.child("briefdescription").to_asciidoc(**kwargs)
if briefdescription:
Expand Down Expand Up @@ -1148,7 +1153,8 @@ def to_asciidoc(self, **kwargs):
else:
argsstring = ""
macro = [
f"* `#define <<{memberdef.id},{escape_text(memberdef.text('name'))}>>{escape_text(argsstring)}"
f"* `#define <<{memberdef.id},{escape_text(memberdef.text('name'))}>>"
f"{escape_text(argsstring)}"
]
if memberdef.text("initializer"):
initializer = memberdef.child("initializer").to_asciidoc(
Expand Down
4 changes: 3 additions & 1 deletion tests/test_copyright_node.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
def test_copyright_node(tmp_path):
xml = """<copy/>"""

asciidoc = CopyrightNode(BeautifulSoup(xml, "xml").copy, xmldir=tmp_path).to_asciidoc()
asciidoc = CopyrightNode(
BeautifulSoup(xml, "xml").copy, xmldir=tmp_path
).to_asciidoc()

assert asciidoc == "©"
4 changes: 3 additions & 1 deletion tests/test_heading_node.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
def test_heading_node(tmp_path):
xml = """<sect1><title>Examples Index</title></sect1>"""

asciidoc = HeadingNode(BeautifulSoup(xml, "xml").title, xmldir=tmp_path).to_asciidoc()
asciidoc = HeadingNode(
BeautifulSoup(xml, "xml").title, xmldir=tmp_path
).to_asciidoc()

assert asciidoc == "=== Examples Index"
4 changes: 3 additions & 1 deletion tests/test_none_node.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
def test_none_node(tmp_path):
xml = """<compoundname>examples_page</compoundname>"""

asciidoc = NoneNode(BeautifulSoup(xml, "xml").compoundname, xmldir=tmp_path).to_asciidoc()
asciidoc = NoneNode(
BeautifulSoup(xml, "xml").compoundname, xmldir=tmp_path
).to_asciidoc()

assert asciidoc == ""
4 changes: 3 additions & 1 deletion tests/test_sect_node.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,9 @@ def test_compounddef_kind_page_is_processed_as_sect(tmp_path):
</detaileddescription>
</compounddef>"""

asciidoc = Node(BeautifulSoup(xml, "xml").compounddef, xmldir=tmp_path).to_asciidoc()
asciidoc = Node(
BeautifulSoup(xml, "xml").compounddef, xmldir=tmp_path
).to_asciidoc()

assert asciidoc == dedent(
"""\
Expand Down

0 comments on commit 7ef0d58

Please sign in to comment.