1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288
|
Contributing
============
This page provides a guide for developers wishing to contribute to **Sphinx-Needs**.
Bugs, Features and PRs
-----------------------
For **bug reports** and well-described **technical feature requests**, please use our issue tracker:
|br| https://github.com/useblocks/sphinx-needs/issues
For **feature ideas** and **questions**, please use our discussion board:
|br| https://github.com/useblocks/sphinx-needs/discussions
If you have already created a **PR**, you can send it in. Our CI workflow will check (test and code styles) and
a maintainer will perform a review, before we can merge it.
Your PR should conform with the following rules:
* A meaningful description or link, which describes the change
* The changed code (for sure :) )
* Test cases for the change (important!)
* Updated documentation, if behavior gets changed or new options/directives are introduced.
* Update of ``docs/changelog.rst``.
* If this is your first PR, feel free to add your name in the ``AUTHORS`` file.
Installing Dependencies
-----------------------
To develop **Sphinx-Needs** it can be installed, with development extras, into an existing Python environment using ``pip``:
.. code-block:: bash
pip install sphinx-needs[test,benchmark,docs]
or using `uv <https://docs.astral.sh/uv/>`__ to install the dependencies into an isolated environment:
.. code-block:: bash
uv sync
To run the formatting and linting suite, `pre-commit <https://pre-commit.com/>`__ is used:
.. code-block:: bash
pre-commit install # to auto-run on every commit
pre-commit run --all-files # to run manually
To run testing and documentation building, `tox <https://tox.readthedocs.io/>`__ is used:
.. code-block:: bash
tox -av # to see all environments
Note, it is recommended to also install the `tox-uv <https://github.com/tox-dev/tox-uv>`__ plugin, which will use `uv` to create isolated environments faster, and to use `pyenv <https://github.com/pyenv/pyenv>`__ to manage multiple Python versions.
Build docs
----------
To build the **Sphinx-Needs** documentation stored under ``/docs``, run:
.. code-block:: bash
# Build HTML pages with the furo theme,
# and first remove all old build files
CLEAN=true tox -e docs-furo
or to build with a different builder:
.. code-block:: bash
# Check links in the documentation
CLEAN=true BUILDER=linkcheck tox -e docs-furo
Running Tests
-------------
You can either run the tests directly using ``pytest``, in an existing environment:
.. code-block:: bash
pytest tests/
Or use tox (recommended):
.. code-block:: bash
tox -e py38
Note some tests use `syrupy <https://github.com/tophat/syrupy>`__ to perform snapshot testing.
These snapshots can be updated by running:
.. code-block:: bash
pytest tests/ --snapshot-update
.. hint::
Please be sure to have the dependencies of the official documentation also installed:
.. code-block:: bash
pip install -r docs/requirements.txt
Running JS Testcases with PyTest
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
**Setup Cypress Locally**
* Install Node JS on your computer and ensure it can be accessed through the CMD.
* Install Cypress using the npm package manager by running ``npm install cypress``. Visit this link for more information on `how to install Cypress <https://docs.cypress.io/guides/getting-started/installing-cypress>`_.
* Verify if Cypress is installed correctly and is executable by running: ``npx cypress verify``. Get out this page for more information about `Cypress commandline <https://docs.cypress.io/guides/guides/command-line>`_.
* If everything is successful then we can use Cypress.
**Enable Cypress Test in Python Test Files**
Under the ``js_test`` folder, you can save your Cypress JS test files (files should end with: ``*.cy.js``). For each Cypress JS test file, you will need to write the Cypress JS test cases in the file. You can read more from the `Cypress Docs <https://docs.cypress.io/>`_. You can also check the ``tests/js_test/sn-collapse-button.cy.js`` file as reference.
In your Python test files, you must mark every JS related test case with the marker - ``jstest`` and you must include the ``spec_pattern`` key-value pair as part of the ``test_app`` fixture parameter.
Also, you need to pass the ``test_server`` fixture to your test function for it to use the automated HTTP test server. For example, your test case could look like this:
.. code-block:: python
# tests/test_sn_collapse_button.py
import pytest
@pytest.mark.jstest
@pytest.mark.parametrize(
"test_app",
[
{
"buildername": "html",
"srcdir": "doc_test/variant_doc",
"tags": ["tag_a"],
"spec_pattern": "js_test/js-test-sn-collapse-button.cy.js"
}
],
indirect=True,
)
def test_collapse_button_in_docs(test_app, test_server):
...
.. note::
The ``spec_pattern`` key is required to ensure Cypress locates your test files or folder. Visit this link for more info on how to set the `spec_pattern <https://docs.cypress.io/guides/guides/command-line#cypress-run-spec-lt-spec-gt>`_.
After you set the ``spec_pattern`` key-value pair as part of the ``test_app`` fixture parameter, you can call ``app.test_js()`` in your Python test case to run a JS test for the ``spec_pattern`` you provided. For example, you can use ``app.test_js()`` like below:
.. code-block:: python
# tests/test_sn_collapse_button.py
import pytest
@pytest.mark.jstest
@pytest.mark.parametrize(
"test_app",
[
{
"buildername": "html",
"srcdir": "doc_test/variant_doc",
"tags": ["tag_a"],
"spec_pattern": "js_test/js-test-sn-collapse-button.cy.js"
}
],
indirect=True,
)
def test_collapse_button_in_docs(test_app, test_server):
"""Check if the Sphinx-Needs collapse button works in the provided documentation source."""
app = test_app
app.build()
# Call `app.test_js()` to run the JS test for a particular specPattern
js_test_result = app.test_js()
# Check the return code and stdout
assert js_test_result["returncode"] == 0
assert "All specs passed!" in js_test_result["stdout"].decode("utf-8")
.. note::
``app.test_js()`` will return a dictionary object containing the ``returncode``, ``stdout``, and ``stderr``. Example:
.. code-block:: python
return {
"returncode": 0,
"stdout": "Test passed string",
"stderr": "Errors encountered,
}
You can run the ``make test-js`` command to check all JS testcases.
.. note::
The ``http_server`` process invoked by the ``make test-js`` command may not terminate properly in some instances.
Kindly check your system's monitoring app to end the process if not terminated automatically.
Benchmarks
----------
**Sphinx-Needs** own documentation is used for creating a benchmark for each PR.
If the runtime takes 10% longer as the previous ones, the benchmark test will fail.
Benchmark test cases are available under ``tests/benchmarks``.
The results for each PR/commit get added to a chart, which is available under
http://useblocks.com/sphinx-needs/bench/index.html.
The benchmark data is stored on the `benchmarks` branch, which is also used by github-pages as
source.
Publishing a new release
------------------------
There is a release pipeline installed for the CI.
This gets triggered automatically, if a tag is created and pushed.
The tag must follow the format: ``[0-9].[0-9]+.[0-9]``. Otherwise the release jobs won't trigger.
The release jobs will build the source and wheel distribution and try to upload them.
Structure of the extension's logic
----------------------------------
The following is an outline of the build events which this extension adds to the :ref:`Sphinx build process <events>`:
#. After configuration has been initialised (``config-inited`` event):
- Register additional directives, directive options and warnings (``load_config``)
- Check configuration consistency (``check_configuration``)
#. Before reading changed documents (``env-before-read-docs`` event):
- Initialise ``BuildEnvironment`` variables (``prepare_env``)
- Register services (``prepare_env``)
- Register functions (``prepare_env``)
- Initialise default extra options (``prepare_env``)
- Initialise extra link types (``prepare_env``)
- Ensure default configurations are set (``prepare_env``)
- Start process timing, if enabled (``prepare_env``)
- Load external needs (``load_external_needs``)
#. For all removed and changed documents (``env-purge-doc`` event):
- Remove all cached need items that originate from the document (``purge_needs``)
#. For changed documents (``doctree-read`` event, priority 880 of transforms)
- Determine and add data on parent sections and needs(``analyse_need_locations``)
- Remove ``Need`` nodes marked as ``hidden`` (``analyse_need_locations``)
#. When building in parallel mode (``env-merge-info`` event), merge ``BuildEnvironment`` data (``merge_data``)
#. After all documents have been read and transformed (``env-updated`` event) (NOTE these are skipped for ``needs`` builder)
- Copy vendored JS libraries (with CSS) to build folder (``install_lib_static_files``)
- Generate permalink file (``install_permalink_file``)
- Copy vendored CSS files to build folder (``install_styles_static_files``)
#. Note, the ``BuildEnvironment`` is cached at this point, only if any documents were updated.
#. For all changed documents, or their dependants (``doctree-resolved``)
- Replace all ``Needextract`` nodes with a list of the collected ``Need`` (``process_creator``)
- Remove all ``Need`` nodes, if ``needs_include_needs`` is ``True`` (``process_need_nodes``)
- Call dynamic functions, set as values on the need data items and replace them with their return values (``process_need_nodes -> resolve_dynamic_values``)
- Replace needs data variant values (``process_need_nodes -> resolve_variants_options``)
- Check for dead links (``process_need_nodes -> check_links``)
- Generate back links (``process_need_nodes -> create_back_links``)
- Process constraints, for each ``Need`` node (``process_need_nodes -> process_constraints``)
- Perform all modifications on need data items, due to ``Needextend`` nodes (``process_need_nodes -> extend_needs_data``)
- Format each ``Need`` node to give the desired visual output (``process_need_nodes -> print_need_nodes``)
- Process all other need specific nodes, replacing them with the desired visual output (``process_creator``)
#. At the end of the build (``build-finished`` event)
- Call all user defined need data checks, a.k.a `needs_warnings` (``process_warnings``)
- Write the ``needs.json`` to the output folder, if `needs_build_json = True` (``build_needs_json``)
- Write the ``needs.json`` per ID to the output folder, if `needs_build_json_per_id = True` (``build_needs_id_json``)
- Write all UML files to the output folder, if `needs_build_needumls = True` (``build_needumls_pumls``)
- Print process timing, if `needs_debug_measurement = True` (``process_timing``)
.. Include our contributors and maintainers.
.. include:: ../AUTHORS
|