diff --git a/.readthedocs.yaml b/.readthedocs.yaml
new file mode 100644
index 0000000..9a0a64c
--- /dev/null
+++ b/.readthedocs.yaml
@@ -0,0 +1,22 @@
+---
+# .readthedocs.yaml
+# Read the Docs configuration file
+# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details
+
+# Required
+version: 2
+
+# Set the version of Python in the build environment.
+build:
+ os: "ubuntu-22.04"
+ tools:
+ python: "3.10"
+
+mkdocs:
+ configuration: "mkdocs.yml"
+ fail_on_warning: true
+
+# Use our docs/requirements.txt during installation.
+python:
+ install:
+ - requirements: "docs/requirements.txt"
diff --git a/README.md b/README.md
index 3424d30..a1dc3dd 100644
--- a/README.md
+++ b/README.md
@@ -17,173 +17,6 @@ The following templates are available:
- [`nautobot-app`](./nautobot-app) - A template for creating a new Nautobot App.
- [`nautobot-app-ssot`](./nautobot-app-ssot) - A template for creating a new Nautobot App that extends the capabilities of the Nautobot Single Source of Truth (SSoT) App. The [Nautobot SSoT App](https://github.com/nautobot/nautobot-plugin-ssot) facilitates integration and data synchronization between various "source of truth" (SoT) systems, with Nautobot acting as a central clearinghouse for data.
-## Local Quick-start
-
-For local usage or development.
-
-Pre-requisites:
-
-- Docker and Docker Compose
-- git
-- Python 3.8+
-- Invoke
-
-Clone the repository:
-
-```shell
-git clone git@github.com:nautobot/cookiecutter-nautobot-app.git
-cd cookiecutter-nautobot-app
-```
-
-Run all tests:
-
-```shell
-invoke tests
-```
-
-List all available tasks:
-
-```shell
-invoke help
-```
-
-Bake a cookie:
-
-```shell
-invoke bake --template=nautobot-app
-```
-
-## Usage with Docker and Invoke
-
-To use templates locally with Docker and Invoke, you need first set up the repository as explained in [local quick-start](#local-quick-start).
-
-Then, you can use the following command:
-
-```shell
-invoke bake
-```
-
-Output:
-
-```shell
-docker compose run --rm -- dev cookiecutter --output-dir=./outputs ./nautobot-app
- [1/18] codeowner_github_usernames (@smith-ntc):
- [2/18] full_name (Network to Code, LLC):
- [3/18] email (info@networktocode.com):
- [4/18] github_org (nautobot):
- [5/18] plugin_name (my_plugin):
- [6/18] verbose_name (My Plugin):
- [7/18] plugin_slug (my-plugin):
- [8/18] project_slug (nautobot-plugin-my-plugin):
- [9/18] repo_url (https://github.com/nautobot/nautobot-plugin-my-plugin):
- [10/18] base_url (my-plugin):
- [11/18] min_nautobot_version (1.6.0):
- [12/18] max_nautobot_version (1.9999):
- [13/18] camel_name (MyPlugin):
- [14/18] project_short_description (My Plugin):
- [15/18] model_class_name (None):
- [16/18] Select open_source_license
- 1 - Apache-2.0
- 2 - Not open source
- Choose from [1/2] (1):
- [17/18] docs_base_url (https://docs.nautobot.com):
- [18/18] docs_app_url (https://docs.nautobot.com/projects/my-plugin/en/latest):
-
-Congratulations! Your cookie has now been baked. It is located at /opt/ntc/nautobot/cookiecutter-nautobot-app/outputs/nautobot-plugin-my-plugin.
-
-⚠️⚠️ Before you start using your cookie you must run the following commands inside your cookie:
-
-* poetry lock
-* cp development/creds.example.env development/creds.env
-* poetry shell
-* invoke makemigrations
-
-The file `creds.env` will be ignored by git and can be used to override default environment variables.
-```
-
-This command, bakes a new cookie using the `nautobot-app` template into `outputs` directory.
-
-### Help
-
-To see all arguments run:
-
-```shell
-invoke bake --help
-```
-
-Output:
-
-```shell
-Usage: inv[oke] [--core-opts] bake [--options] [other tasks here ...]
-
-Docstring:
- Bake a new cookie from the template.
-
-Options:
- -d, --debug Whether to run in debug mode (defaults to False)
- -i, --[no-]input Whether to require user input, ignored with `--json-file` (defaults to True)
- -j STRING, --json-file=STRING Path to a JSON file containing answers to prompts (defaults to empty)
- -o STRING, --output-dir=STRING Path to the output directory (defaults to ./outputs)
- -t STRING, --template=STRING Path to the cookiecutter template to bake (defaults to ./nautobot-app)
-```
-
-!!! IMPORTANT !!! The `--output-dir` argument should be located in the current directory, as the Cookiecutter creates the cookie inside the Docker container with the current directory bind mounted. When baking somewhere else, the resulting cookie would not be available from calling host system.
-
-### JSON file
-
-You can reuse existing JSON file with pre-filled template prompts. To do so, you need to pass the path to the JSON file using the `--json-file` argument.
-
-When running the command with `--json-file` argument, the command will not prompt for any input.
-
-First create the `my-app.json` file:
-
-```json
-{
- "cookiecutter": {
- "codeowner_github_usernames": "@smith-ntc",
- "full_name": "Network to Code, LLC",
- "email": "info@networktocode.com",
- "github_org": "nautobot",
- "plugin_name": "my_app",
- "verbose_name": "My App",
- "plugin_slug": "my-app",
- "project_slug": "nautobot-plugin-my-app",
- "repo_url": "https://github.com/nautobot/nautobot-plugin-my-app",
- "base_url": "my-app",
- "min_nautobot_version": "2.0.0",
- "max_nautobot_version": "2.9999",
- "camel_name": "MyApp",
- "project_short_description": "My App",
- "model_class_name": "MyModel",
- "open_source_license": "Apache-2.0",
- "docs_base_url": "https://docs.nautobot.com",
- "docs_app_url": "https://docs.nautobot.com/projects/my-app/en/latest"
- }
-}
-```
-
-Then run the following command:
-
-```shell
-invoke bake --json-file=my-app.json
-```
-
-Output:
-
-```shell
-docker compose run --rm -- dev cookiecutter --output-dir=./outputs --replay-file=my-app.json ./nautobot-app
-
-Congratulations! Your cookie has now been baked. It is located at /opt/ntc/nautobot/cookiecutter-nautobot-app/outputs/nautobot-plugin-my-app.
-
-⚠️⚠️ Before you start using your cookie you must run the following commands inside your cookie:
-
-* poetry lock
-* cp development/creds.example.env development/creds.env
-* poetry shell
-* invoke makemigrations
-
-The file `creds.env` will be ignored by git and can be used to override default environment variables.
-```
## Usage with Cookiecutter
@@ -195,7 +28,7 @@ Pre-requisites:
```shell
pip install cookiecutter
-mkdir outpus
+mkdir outputs
```
Then run the following command:
diff --git a/docs/assets/extra.css b/docs/assets/extra.css
new file mode 100644
index 0000000..dfe2e4b
--- /dev/null
+++ b/docs/assets/extra.css
@@ -0,0 +1,161 @@
+:root>* {
+ --md-accent-fg-color: #ff8504;
+ --md-primary-fg-color: #ff8504;
+ --md-typeset-a-color: #0097ff;
+}
+
+[data-md-color-scheme="slate"] {
+ --md-default-bg-color: hsla(var(--md-hue), 0%, 15%, 1);
+ --md-typeset-a-color: #0097ff;
+}
+
+/* Accessibility: Increase fonts for dark theme */
+[data-md-color-scheme="slate"] .md-typeset {
+ font-size: 0.9rem;
+}
+
+[data-md-color-scheme="slate"] .md-typeset table:not([class]) {
+ font-size: 0.7rem;
+}
+
+/*
+* The default max-width is 61rem which does not provide nearly enough space to present code examples or larger tables
+*/
+.md-grid {
+ margin-left: auto;
+ margin-right: auto;
+ max-width: 95%;
+}
+
+.md-tabs__link {
+ font-size: 0.8rem;
+}
+
+.md-tabs__link--active {
+ color: var(--md-primary-fg-color);
+}
+
+.md-header__button.md-logo :is(img, svg) {
+ height: 2rem;
+}
+
+.md-header__button.md-logo :-webkit-any(img, svg) {
+ height: 2rem;
+}
+
+.md-header__title {
+ font-size: 1.2rem;
+}
+
+img.logo {
+ height: 200px;
+}
+
+img.copyright-logo {
+ height: 24px;
+ vertical-align: middle;
+}
+
+[data-md-color-primary=black] .md-header {
+ background-color: #212121;
+}
+
+@media screen and (min-width: 76.25em) {
+ [data-md-color-primary=black] .md-tabs {
+ background-color: #212121;
+ }
+}
+
+/* Customization for mkdocstrings */
+/* Indentation. */
+div.doc-contents:not(.first) {
+ padding-left: 25px;
+ border-left: .2rem solid var(--md-typeset-table-color);
+}
+
+/* Mark external links as such. */
+a.autorefs-external::after {
+ /* https://primer.style/octicons/arrow-up-right-24 */
+ background-image: url('data:image/svg+xml,');
+ content: ' ';
+
+ display: inline-block;
+ position: relative;
+ top: 0.1em;
+ margin-left: 0.2em;
+ margin-right: 0.1em;
+
+ height: 1em;
+ width: 1em;
+ border-radius: 100%;
+ background-color: var(--md-typeset-a-color);
+}
+
+a.autorefs-external:hover::after {
+ background-color: var(--md-accent-fg-color);
+}
+
+
+/* Customization for mkdocs-version-annotations */
+:root {
+ /* Icon for "version-added" admonition: Material Design Icons "plus-box-outline" */
+ --md-admonition-icon--version-added: url('data:image/svg+xml;charset=utf-8,');
+ /* Icon for "version-changed" admonition: Material Design Icons "delta" */
+ --md-admonition-icon--version-changed: url('data:image/svg+xml;charset=utf-8,');
+ /* Icon for "version-removed" admonition: Material Design Icons "minus-circle-outline" */
+ --md-admonition-icon--version-removed: url('data:image/svg+xml;charset=utf-8,');
+}
+
+/* "version-added" admonition in green */
+.md-typeset .admonition.version-added,
+.md-typeset details.version-added {
+ border-color: rgb(0, 200, 83);
+}
+
+.md-typeset .version-added>.admonition-title,
+.md-typeset .version-added>summary {
+ background-color: rgba(0, 200, 83, .1);
+}
+
+.md-typeset .version-added>.admonition-title::before,
+.md-typeset .version-added>summary::before {
+ background-color: rgb(0, 200, 83);
+ -webkit-mask-image: var(--md-admonition-icon--version-added);
+ mask-image: var(--md-admonition-icon--version-added);
+}
+
+/* "version-changed" admonition in orange */
+.md-typeset .admonition.version-changed,
+.md-typeset details.version-changed {
+ border-color: rgb(255, 145, 0);
+}
+
+.md-typeset .version-changed>.admonition-title,
+.md-typeset .version-changed>summary {
+ background-color: rgba(255, 145, 0, .1);
+}
+
+.md-typeset .version-changed>.admonition-title::before,
+.md-typeset .version-changed>summary::before {
+ background-color: rgb(255, 145, 0);
+ -webkit-mask-image: var(--md-admonition-icon--version-changed);
+ mask-image: var(--md-admonition-icon--version-changed);
+}
+
+/* "version-removed" admonition in red */
+.md-typeset .admonition.version-removed,
+.md-typeset details.version-removed {
+ border-color: rgb(255, 82, 82);
+}
+
+.md-typeset .version-removed>.admonition-title,
+.md-typeset .version-removed>summary {
+ background-color: rgba(255, 82, 82, .1);
+}
+
+.md-typeset .version-removed>.admonition-title::before,
+.md-typeset .version-removed>summary::before {
+ background-color: rgb(255, 82, 82);
+ -webkit-mask-image: var(--md-admonition-icon--version-removed);
+ mask-image: var(--md-admonition-icon--version-removed);
+}
diff --git a/docs/assets/favicon.ico b/docs/assets/favicon.ico
new file mode 100644
index 0000000..fcaa3d6
Binary files /dev/null and b/docs/assets/favicon.ico differ
diff --git a/docs/assets/nautobot_logo.png b/docs/assets/nautobot_logo.png
new file mode 100644
index 0000000..3743fea
Binary files /dev/null and b/docs/assets/nautobot_logo.png differ
diff --git a/docs/assets/nautobot_logo.svg b/docs/assets/nautobot_logo.svg
new file mode 100644
index 0000000..c9bba8e
--- /dev/null
+++ b/docs/assets/nautobot_logo.svg
@@ -0,0 +1,131 @@
+
+
diff --git a/docs/assets/networktocode_bw.png b/docs/assets/networktocode_bw.png
new file mode 100644
index 0000000..075c492
Binary files /dev/null and b/docs/assets/networktocode_bw.png differ
diff --git a/docs/assets/overrides/partials/copyright.html b/docs/assets/overrides/partials/copyright.html
new file mode 100644
index 0000000..bfcbab9
--- /dev/null
+++ b/docs/assets/overrides/partials/copyright.html
@@ -0,0 +1,21 @@
+
+ {% if config.copyright %}
+
+ {{ config.copyright }}
+
+
Apache-2.0 LICENSE
+ {% endif %}
+
+
+
+
+
+
diff --git a/docs/index.md b/docs/index.md
new file mode 100644
index 0000000..f32fd72
--- /dev/null
+++ b/docs/index.md
@@ -0,0 +1,6 @@
+---
+hide:
+ - navigation
+---
+
+--8<-- "README.md"
diff --git a/docs/quick-start.md b/docs/quick-start.md
new file mode 100644
index 0000000..84e7638
--- /dev/null
+++ b/docs/quick-start.md
@@ -0,0 +1,169 @@
+# Repository Tooling
+
+## Local Quick-start
+
+For local usage or development.
+
+Pre-requisites:
+
+- Docker and Docker Compose
+- git
+- Python 3.8+
+- Invoke
+
+Clone the repository:
+
+```shell
+git clone git@github.com:nautobot/cookiecutter-nautobot-app.git
+cd cookiecutter-nautobot-app
+```
+
+Run all tests:
+
+```shell
+invoke tests
+```
+
+List all available tasks:
+
+```shell
+invoke help
+```
+
+Bake a cookie:
+
+```shell
+invoke bake --template=nautobot-app
+```
+
+## Usage with Docker and Invoke
+
+To use templates locally with Docker and Invoke, you need first set up the repository as explained in [local quick-start](#local-quick-start).
+
+Then, you can use the following command:
+
+```shell
+invoke bake
+```
+
+Output:
+
+```shell
+docker compose run --rm -- dev cookiecutter --output-dir=./outputs ./nautobot-app
+ [1/18] codeowner_github_usernames (@smith-ntc):
+ [2/18] full_name (Network to Code, LLC):
+ [3/18] email (info@networktocode.com):
+ [4/18] github_org (nautobot):
+ [5/18] plugin_name (my_plugin):
+ [6/18] verbose_name (My Plugin):
+ [7/18] plugin_slug (my-plugin):
+ [8/18] project_slug (nautobot-plugin-my-plugin):
+ [9/18] repo_url (https://github.com/nautobot/nautobot-plugin-my-plugin):
+ [10/18] base_url (my-plugin):
+ [11/18] min_nautobot_version (1.6.0):
+ [12/18] max_nautobot_version (1.9999):
+ [13/18] camel_name (MyPlugin):
+ [14/18] project_short_description (My Plugin):
+ [15/18] model_class_name (None):
+ [16/18] Select open_source_license
+ 1 - Apache-2.0
+ 2 - Not open source
+ Choose from [1/2] (1):
+ [17/18] docs_base_url (https://docs.nautobot.com):
+ [18/18] docs_app_url (https://docs.nautobot.com/projects/my-plugin/en/latest):
+
+Congratulations! Your cookie has now been baked. It is located at /opt/ntc/nautobot/cookiecutter-nautobot-app/outputs/nautobot-plugin-my-plugin.
+
+⚠️⚠️ Before you start using your cookie you must run the following commands inside your cookie:
+
+* poetry lock
+* cp development/creds.example.env development/creds.env
+* poetry shell
+* invoke makemigrations
+
+The file `creds.env` will be ignored by git and can be used to override default environment variables.
+```
+
+This command, bakes a new cookie using the `nautobot-app` template into `outputs` directory.
+
+### Help
+
+To see all arguments run:
+
+```shell
+invoke bake --help
+```
+
+Output:
+
+```shell
+Usage: inv[oke] [--core-opts] bake [--options] [other tasks here ...]
+
+Docstring:
+ Bake a new cookie from the template.
+
+Options:
+ -d, --debug Whether to run in debug mode (defaults to False)
+ -i, --[no-]input Whether to require user input, ignored with `--json-file` (defaults to True)
+ -j STRING, --json-file=STRING Path to a JSON file containing answers to prompts (defaults to empty)
+ -o STRING, --output-dir=STRING Path to the output directory (defaults to ./outputs)
+ -t STRING, --template=STRING Path to the cookiecutter template to bake (defaults to ./nautobot-app)
+```
+
+!!! IMPORTANT !!! The `--output-dir` argument should be located in the current directory, as the Cookiecutter creates the cookie inside the Docker container with the current directory bind mounted. When baking somewhere else, the resulting cookie would not be available from calling host system.
+
+### JSON file
+
+You can reuse existing JSON file with pre-filled template prompts. To do so, you need to pass the path to the JSON file using the `--json-file` argument.
+
+When running the command with `--json-file` argument, the command will not prompt for any input.
+
+First create the `my-app.json` file:
+
+```json
+{
+ "cookiecutter": {
+ "codeowner_github_usernames": "@smith-ntc",
+ "full_name": "Network to Code, LLC",
+ "email": "info@networktocode.com",
+ "github_org": "nautobot",
+ "plugin_name": "my_app",
+ "verbose_name": "My App",
+ "plugin_slug": "my-app",
+ "project_slug": "nautobot-plugin-my-app",
+ "repo_url": "https://github.com/nautobot/nautobot-plugin-my-app",
+ "base_url": "my-app",
+ "min_nautobot_version": "2.0.0",
+ "max_nautobot_version": "2.9999",
+ "camel_name": "MyApp",
+ "project_short_description": "My App",
+ "model_class_name": "MyModel",
+ "open_source_license": "Apache-2.0",
+ "docs_base_url": "https://docs.nautobot.com",
+ "docs_app_url": "https://docs.nautobot.com/projects/my-app/en/latest"
+ }
+}
+```
+
+Then run the following command:
+
+```shell
+invoke bake --json-file=my-app.json
+```
+
+Output:
+
+```shell
+docker compose run --rm -- dev cookiecutter --output-dir=./outputs --replay-file=my-app.json ./nautobot-app
+
+Congratulations! Your cookie has now been baked. It is located at /opt/ntc/nautobot/cookiecutter-nautobot-app/outputs/nautobot-plugin-my-app.
+
+⚠️⚠️ Before you start using your cookie you must run the following commands inside your cookie:
+
+* poetry lock
+* cp development/creds.example.env development/creds.env
+* poetry shell
+* invoke makemigrations
+
+The file `creds.env` will be ignored by git and can be used to override default environment variables.
+```
diff --git a/docs/requirements.txt b/docs/requirements.txt
new file mode 100644
index 0000000..d0d83d1
--- /dev/null
+++ b/docs/requirements.txt
@@ -0,0 +1,3 @@
+mkdocs==1.5.2
+mkdocs-material==9.1.15
+mkdocs-version-annotations==1.0.0
diff --git a/mkdocs.yml b/mkdocs.yml
new file mode 100644
index 0000000..9c2b97b
--- /dev/null
+++ b/mkdocs.yml
@@ -0,0 +1,87 @@
+---
+dev_addr: "127.0.0.1:8001"
+edit_uri: "edit/main/cookiecutter-nautobot-app/docs"
+site_name: "Nautobot Cookiecutter Templates Documentation"
+site_url: "https://docs.nautobot.com/projects/cookiecutter-nautobot-app/en/latest/"
+repo_url: "https://github.com/nautobot/cookiecutter-nautobot-app"
+copyright: "Copyright © The Authors"
+theme:
+ name: "material"
+ navigation_depth: 4
+ custom_dir: "docs/assets/overrides"
+ hljs_languages:
+ - "django"
+ - "yaml"
+ features:
+ - "content.action.edit"
+ - "content.action.view"
+ - "content.code.copy"
+ - "navigation.footer"
+ - "navigation.indexes"
+ - "navigation.tabs"
+ - "navigation.tabs.sticky"
+ - "navigation.tracking"
+ - "search.highlight"
+ - "search.share"
+ - "search.suggest"
+ favicon: "assets/favicon.ico"
+ logo: "assets/nautobot_logo.svg"
+ palette:
+ # Palette toggle for light mode
+ - media: "(prefers-color-scheme: light)"
+ scheme: "default"
+ primary: "black"
+ toggle:
+ icon: "material/weather-sunny"
+ name: "Switch to dark mode"
+
+ # Palette toggle for dark mode
+ - media: "(prefers-color-scheme: dark)"
+ scheme: "slate"
+ primary: "black"
+ toggle:
+ icon: "material/weather-night"
+ name: "Switch to light mode"
+extra_css:
+ - "assets/extra.css"
+
+extra:
+ generator: false
+ ntc_sponsor: true
+ social:
+ - icon: "fontawesome/solid/rss"
+ link: "https://blog.networktocode.com/blog/tags/nautobot"
+ name: "Network to Code Blog"
+ - icon: "fontawesome/brands/youtube"
+ link: "https://www.youtube.com/playlist?list=PLjA0bhxgryJ2Ts4GJMDA-tPzVWEncv4pb"
+ name: "Nautobot Videos"
+ - icon: "fontawesome/brands/slack"
+ link: "https://www.networktocode.com/community/"
+ name: "Network to Code Community"
+ - icon: "fontawesome/brands/github"
+ link: "https://github.com/nautobot/nautobot"
+ name: "GitHub Repo"
+ - icon: "fontawesome/brands/twitter"
+ link: "https://twitter.com/networktocode"
+ name: "Network to Code Twitter"
+markdown_extensions:
+ - "admonition"
+ - "toc":
+ permalink: true
+ - "attr_list"
+ - "md_in_html"
+ - "pymdownx.highlight":
+ anchor_linenums: true
+ - "pymdownx.inlinehilite"
+ - "pymdownx.snippets"
+ - "pymdownx.superfences"
+ - "footnotes"
+plugins:
+ - "search"
+ - "mkdocs-version-annotations"
+watch:
+ - "README.md"
+
+nav:
+ - Overview: "index.md"
+ - Quick Start: "quick-start.md"
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/.github/PULL_REQUEST_TEMPLATE/pull_request_template.md b/nautobot-app/{{ cookiecutter.project_slug }}/.github/PULL_REQUEST_TEMPLATE/pull_request_template.md
index 3950227..df69fc6 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/.github/PULL_REQUEST_TEMPLATE/pull_request_template.md
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/.github/PULL_REQUEST_TEMPLATE/pull_request_template.md
@@ -1,5 +1,5 @@
- [ ] Explanation of Change(s)
-- [ ] Added change log fragment(s) (for more information see [the documentation](https://docs.nautobot.com/projects/core/en/stable/development/#creating-changelog-fragments))
- [ ] Attached Screenshots, Payload Example
- [ ] Unit, Integration Tests
- [ ] Documentation Updates (when adding/changing features)
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/development/Dockerfile b/nautobot-app/{{ cookiecutter.project_slug }}/development/Dockerfile
index 1c4400c..e4851f9 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/development/Dockerfile
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/development/Dockerfile
@@ -63,8 +63,8 @@ RUN pip show nautobot | grep "^Version: " | sed -e 's/Version: /nautobot==/' > c
#
# We can't use the entire freeze as it takes forever to resolve with rigidly fixed non-direct dependencies,
# especially those that are only direct to Nautobot but the container included versions slightly mismatch
-RUN poetry export -f requirements.txt --without-hashes --output poetry_freeze_base.txt
-RUN poetry export -f requirements.txt --with dev --without-hashes --output poetry_freeze_all.txt
+RUN poetry export -f requirements.txt --without-hashes --extras all --output poetry_freeze_base.txt
+RUN poetry export -f requirements.txt --without-hashes --extras all --with dev --output poetry_freeze_all.txt
RUN sort poetry_freeze_base.txt poetry_freeze_all.txt | uniq -u > poetry_freeze_dev.txt
# Install all local project as editable, constrained on Nautobot version, to get any additional
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/docs/admin/uninstall.md b/nautobot-app/{{ cookiecutter.project_slug }}/docs/admin/uninstall.md
index 86eb421..862efdd 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/docs/admin/uninstall.md
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/docs/admin/uninstall.md
@@ -7,7 +7,7 @@ Here you will find any steps necessary to cleanly remove the App from your Nauto
Prior to removing the plugin from the `nautobot_config.py`, run the following command to roll back any migration specific to this plugin.
```shell
-nautobot-server migrate {{ cookiecutter.project_slug.replace('-', '_') }} zero
+nautobot-server migrate {{ cookiecutter.plugin_name }} zero
```
!!! warning "Developer Note - Remove Me!"
@@ -16,3 +16,9 @@ nautobot-server migrate {{ cookiecutter.project_slug.replace('-', '_') }} zero
## Remove App configuration
Remove the configuration you added in `nautobot_config.py` from `PLUGINS` & `PLUGINS_CONFIG`.
+
+## Uninstall the package
+
+```bash
+$ pip3 uninstall {{ cookiecutter.plugin_slug }}
+```
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/docs/assets/extra.css b/nautobot-app/{{ cookiecutter.project_slug }}/docs/assets/extra.css
index dfe2e4b..1eff119 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/docs/assets/extra.css
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/docs/assets/extra.css
@@ -159,3 +159,8 @@ a.autorefs-external:hover::after {
-webkit-mask-image: var(--md-admonition-icon--version-removed);
mask-image: var(--md-admonition-icon--version-removed);
}
+
+/* Do not wrap code blocks in markdown tables. */
+div.md-typeset__table>table>tbody>tr>td>code {
+ white-space: nowrap;
+}
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/pyproject.toml b/nautobot-app/{{ cookiecutter.project_slug }}/pyproject.toml
index d2006d1..7804ada 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/pyproject.toml
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/pyproject.toml
@@ -59,6 +59,10 @@ mkdocs-version-annotations = "1.0.0"
mkdocstrings = "0.22.0"
mkdocstrings-python = "1.5.2"
+[tool.poetry.extras]
+all = [
+]
+
[tool.black]
line-length = 120
target-version = ['py38', 'py39', 'py310', 'py311']
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/tasks.py b/nautobot-app/{{ cookiecutter.project_slug }}/tasks.py
index f903d7b..a0a7d9f 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/tasks.py
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/tasks.py
@@ -13,6 +13,8 @@
"""
import os
+from pathlib import Path
+from time import sleep
from invoke.collection import Collection
from invoke.tasks import task as invoke_task
@@ -67,6 +69,25 @@ def _is_compose_included(context, name):
return f"docker-compose.{name}.yml" in context.{{ cookiecutter.plugin_name }}.compose_files
+def _await_healthy_service(context, service):
+ container_id = docker_compose(context, f"ps -q -- {service}", pty=False, echo=False, hide=True).stdout.strip()
+ _await_healthy_container(context, container_id)
+
+
+def _await_healthy_container(context, container_id):
+ while True:
+ result = context.run(
+ "docker inspect --format='{% raw %}{{.State.Health.Status}}{% endraw %}' " + container_id,
+ pty=False,
+ echo=False,
+ hide=True,
+ )
+ if result.stdout.strip() == "healthy":
+ break
+ print(f"Waiting for `{container_id}` container to become healthy ...")
+ sleep(1)
+
+
def task(function=None, *args, **kwargs):
"""Task decorator to override the default Invoke task decorator and add each task to the invoke namespace."""
@@ -216,11 +237,46 @@ def stop(context, service=""):
docker_compose(context, "stop" if service else "down --remove-orphans", service=service)
-@task
-def destroy(context):
+@task(
+ aliases=("down",),
+ help={
+ "volumes": "Remove Docker compose volumes (default: True)",
+ "import-db-file": "Import database from `import-db-file` file into the fresh environment (default: empty)",
+ },
+)
+def destroy(context, volumes=True, import_db_file=""):
"""Destroy all containers and volumes."""
print("Destroying Nautobot...")
- docker_compose(context, "down --remove-orphans --volumes")
+ docker_compose(context, f"down --remove-orphans {'--volumes' if volumes else ''}")
+
+ if not import_db_file:
+ return
+
+ if not volumes:
+ raise ValueError("Cannot specify `--no-volumes` and `--import-db-file` arguments at the same time.")
+
+ print(f"Importing database file: {import_db_file}...")
+
+ input_path = Path(import_db_file).absolute()
+ if not input_path.is_file():
+ raise ValueError(f"File not found: {input_path}")
+
+ command = [
+ "run",
+ "--rm",
+ "--detach",
+ f"--volume='{input_path}:/docker-entrypoint-initdb.d/dump.sql'",
+ "--",
+ "db",
+ ]
+
+ container_id = docker_compose(context, " ".join(command), pty=False, echo=False, hide=True).stdout.strip()
+ _await_healthy_container(context, container_id)
+ print("Stopping database container...")
+ context.run(f"docker stop {container_id}", pty=False, echo=False, hide=True)
+
+ print("Database import complete, you can start Nautobot with the following command:")
+ print("invoke start")
@task
@@ -424,27 +480,43 @@ def dbshell(context, db_name="", input_file="", output_file="", query=""):
@task(
help={
+ "db-name": "Database name to create (default: Nautobot database)",
"input-file": "SQL dump file to replace the existing database with. This can be generated using `invoke backup-db` (default: `dump.sql`).",
}
)
-def import_db(context, input_file="dump.sql"):
- """Stop Nautobot containers and replace the current database with the dump into the running `db` container."""
- docker_compose(context, "stop -- nautobot worker")
+def import_db(context, db_name="", input_file="dump.sql"):
+ """Stop Nautobot containers and replace the current database with the dump into `db` container."""
+ docker_compose(context, "stop -- nautobot worker beat")
+ start(context, "db")
+ _await_healthy_service(context, "db")
command = ["exec -- db sh -c '"]
if _is_compose_included(context, "mysql"):
+ if not db_name:
+ db_name = "$MYSQL_DATABASE"
command += [
+ "mysql --user root --password=$MYSQL_ROOT_PASSWORD",
+ '--execute="',
+ f"DROP DATABASE IF EXISTS {db_name};",
+ f"CREATE DATABASE {db_name};",
+ ""
+ if db_name == "$MYSQL_DATABASE"
+ else f"GRANT ALL PRIVILEGES ON {db_name}.* TO $MYSQL_USER; FLUSH PRIVILEGES;",
+ '"',
+ "&&",
"mysql",
- "--database=$MYSQL_DATABASE",
+ f"--database={db_name}",
"--user=$MYSQL_USER",
"--password=$MYSQL_PASSWORD",
]
elif _is_compose_included(context, "postgres"):
+ if not db_name:
+ db_name = "$POSTGRES_DB"
command += [
- "psql",
- "--username=$POSTGRES_USER",
- "postgres",
+ f"dropdb --if-exists --user=$POSTGRES_USER {db_name} &&",
+ f"createdb --user=$POSTGRES_USER {db_name} &&",
+ f"psql --user=$POSTGRES_USER --dbname={db_name}",
]
else:
raise ValueError("Unsupported database backend.")
@@ -467,7 +539,10 @@ def import_db(context, input_file="dump.sql"):
}
)
def backup_db(context, db_name="", output_file="dump.sql", readable=True):
- """Dump database into `output_file` file from running `db` container."""
+ """Dump database into `output_file` file from `db` container."""
+ start(context, "db")
+ _await_healthy_service(context, "db")
+
command = ["exec -- db sh -c '"]
if _is_compose_included(context, "mysql"):
@@ -475,17 +550,12 @@ def backup_db(context, db_name="", output_file="dump.sql", readable=True):
"mysqldump",
"--user=root",
"--password=$MYSQL_ROOT_PASSWORD",
- "--add-drop-database",
"--skip-extended-insert" if readable else "",
- "--databases",
db_name if db_name else "$MYSQL_DATABASE",
]
elif _is_compose_included(context, "postgres"):
command += [
"pg_dump",
- "--clean",
- "--create",
- "--if-exists",
"--username=$POSTGRES_USER",
f"--dbname={db_name or '$POSTGRES_DB'}",
"--inserts" if readable else "",
diff --git a/nautobot-app/{{ cookiecutter.project_slug }}/{{ cookiecutter.plugin_name }}/tests/test_basic.py b/nautobot-app/{{ cookiecutter.project_slug }}/{{ cookiecutter.plugin_name }}/tests/test_basic.py
index 95e15ce..bcf7d12 100644
--- a/nautobot-app/{{ cookiecutter.project_slug }}/{{ cookiecutter.plugin_name }}/tests/test_basic.py
+++ b/nautobot-app/{{ cookiecutter.project_slug }}/{{ cookiecutter.plugin_name }}/tests/test_basic.py
@@ -3,18 +3,6 @@
import os
import toml
-from {{ cookiecutter.plugin_name }} import __version__ as project_version
-
-
-class TestVersion(unittest.TestCase):
- """Test Version is the same."""
-
- def test_version(self):
- """Verify that pyproject.toml version is same as version specified in the package."""
- parent_path = os.path.dirname(os.path.dirname(os.path.dirname(os.path.realpath(__file__))))
- poetry_version = toml.load(os.path.join(parent_path, "pyproject.toml"))["tool"]["poetry"]["version"]
- self.assertEqual(project_version, poetry_version)
-
class TestDocsPackaging(unittest.TestCase):
"""Test Version in doc requirements is the same pyproject."""
diff --git a/poetry.lock b/poetry.lock
index c92890c..5965018 100644
--- a/poetry.lock
+++ b/poetry.lock
@@ -296,6 +296,23 @@ files = [
[package.extras]
test = ["pytest (>=6)"]
+[[package]]
+name = "ghp-import"
+version = "2.1.0"
+description = "Copy your docs directly to the gh-pages branch."
+optional = false
+python-versions = "*"
+files = [
+ {file = "ghp-import-2.1.0.tar.gz", hash = "sha256:9c535c4c61193c2df8871222567d7fd7e5014d835f97dc7b7439069e2413d343"},
+ {file = "ghp_import-2.1.0-py3-none-any.whl", hash = "sha256:8337dd7b50877f163d4c0289bc1f1c7f127550241988d568c1db512c4324a619"},
+]
+
+[package.dependencies]
+python-dateutil = ">=2.8.1"
+
+[package.extras]
+dev = ["flake8", "markdown", "twine", "wheel"]
+
[[package]]
name = "gitdb"
version = "4.0.10"
@@ -338,6 +355,25 @@ files = [
{file = "idna-3.4.tar.gz", hash = "sha256:814f528e8dead7d329833b91c5faa87d60bf71824cd12a7530b5526063d02cb4"},
]
+[[package]]
+name = "importlib-metadata"
+version = "6.8.0"
+description = "Read metadata from Python packages"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "importlib_metadata-6.8.0-py3-none-any.whl", hash = "sha256:3ebb78df84a805d7698245025b975d9d67053cd94c79245ba4b3eb694abe68bb"},
+ {file = "importlib_metadata-6.8.0.tar.gz", hash = "sha256:dbace7892d8c0c4ac1ad096662232f831d4e64f4c4545bd53016a3e9d4654743"},
+]
+
+[package.dependencies]
+zipp = ">=0.5"
+
+[package.extras]
+docs = ["furo", "jaraco.packaging (>=9)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"]
+perf = ["ipython"]
+testing = ["flufl.flake8", "importlib-resources (>=1.3)", "packaging", "pyfakefs", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf (>=0.9.2)", "pytest-ruff"]
+
[[package]]
name = "iniconfig"
version = "2.0.0"
@@ -439,6 +475,24 @@ files = [
{file = "lazy_object_proxy-1.9.0-cp39-cp39-win_amd64.whl", hash = "sha256:db1c1722726f47e10e0b5fdbf15ac3b8adb58c091d12b3ab713965795036985f"},
]
+[[package]]
+name = "markdown"
+version = "3.5.1"
+description = "Python implementation of John Gruber's Markdown."
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "Markdown-3.5.1-py3-none-any.whl", hash = "sha256:5874b47d4ee3f0b14d764324d2c94c03ea66bee56f2d929da9f2508d65e722dc"},
+ {file = "Markdown-3.5.1.tar.gz", hash = "sha256:b65d7beb248dc22f2e8a31fb706d93798093c308dc1aba295aedeb9d41a813bd"},
+]
+
+[package.dependencies]
+importlib-metadata = {version = ">=4.4", markers = "python_version < \"3.10\""}
+
+[package.extras]
+docs = ["mdx-gh-links (>=0.2)", "mkdocs (>=1.5)", "mkdocs-gen-files", "mkdocs-literate-nav", "mkdocs-nature (>=0.6)", "mkdocs-section-index", "mkdocstrings[python]"]
+testing = ["coverage", "pyyaml"]
+
[[package]]
name = "markdown-it-py"
version = "3.0.0"
@@ -490,6 +544,16 @@ files = [
{file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5bbe06f8eeafd38e5d0a4894ffec89378b6c6a625ff57e3028921f8ff59318ac"},
{file = "MarkupSafe-2.1.3-cp311-cp311-win32.whl", hash = "sha256:dd15ff04ffd7e05ffcb7fe79f1b98041b8ea30ae9234aed2a9168b5797c3effb"},
{file = "MarkupSafe-2.1.3-cp311-cp311-win_amd64.whl", hash = "sha256:134da1eca9ec0ae528110ccc9e48041e0828d79f24121a1a146161103c76e686"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f698de3fd0c4e6972b92290a45bd9b1536bffe8c6759c62471efaa8acb4c37bc"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aa57bd9cf8ae831a362185ee444e15a93ecb2e344c8e52e4d721ea3ab6ef1823"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffcc3f7c66b5f5b7931a5aa68fc9cecc51e685ef90282f4a82f0f5e9b704ad11"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47d4f1c5f80fc62fdd7777d0d40a2e9dda0a05883ab11374334f6c4de38adffd"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1f67c7038d560d92149c060157d623c542173016c4babc0c1913cca0564b9939"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9aad3c1755095ce347e26488214ef77e0485a3c34a50c5a5e2471dff60b9dd9c"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:14ff806850827afd6b07a5f32bd917fb7f45b046ba40c57abdb636674a8b559c"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8f9293864fe09b8149f0cc42ce56e3f0e54de883a9de90cd427f191c346eb2e1"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-win32.whl", hash = "sha256:715d3562f79d540f251b99ebd6d8baa547118974341db04f5ad06d5ea3eb8007"},
+ {file = "MarkupSafe-2.1.3-cp312-cp312-win_amd64.whl", hash = "sha256:1b8dd8c3fd14349433c79fa8abeb573a55fc0fdd769133baac1f5e07abf54aeb"},
{file = "MarkupSafe-2.1.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8e254ae696c88d98da6555f5ace2279cf7cd5b3f52be2b5cf97feafe883b58d2"},
{file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0932dc158471523c9637e807d9bfb93e06a95cbf010f1a38b98623b929ef2b"},
{file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9402b03f1a1b4dc4c19845e5c749e3ab82d5078d16a2a4c2cd2df62d57bb0707"},
@@ -544,6 +608,92 @@ files = [
{file = "mdurl-0.1.2.tar.gz", hash = "sha256:bb413d29f5eea38f31dd4754dd7377d4465116fb207585f97bf925588687c1ba"},
]
+[[package]]
+name = "mergedeep"
+version = "1.3.4"
+description = "A deep merge function for 🐍."
+optional = false
+python-versions = ">=3.6"
+files = [
+ {file = "mergedeep-1.3.4-py3-none-any.whl", hash = "sha256:70775750742b25c0d8f36c55aed03d24c3384d17c951b3175d898bd778ef0307"},
+ {file = "mergedeep-1.3.4.tar.gz", hash = "sha256:0096d52e9dad9939c3d975a774666af186eda617e6ca84df4c94dec30004f2a8"},
+]
+
+[[package]]
+name = "mkdocs"
+version = "1.5.2"
+description = "Project documentation with Markdown."
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "mkdocs-1.5.2-py3-none-any.whl", hash = "sha256:60a62538519c2e96fe8426654a67ee177350451616118a41596ae7c876bb7eac"},
+ {file = "mkdocs-1.5.2.tar.gz", hash = "sha256:70d0da09c26cff288852471be03c23f0f521fc15cf16ac89c7a3bfb9ae8d24f9"},
+]
+
+[package.dependencies]
+click = ">=7.0"
+colorama = {version = ">=0.4", markers = "platform_system == \"Windows\""}
+ghp-import = ">=1.0"
+importlib-metadata = {version = ">=4.3", markers = "python_version < \"3.10\""}
+jinja2 = ">=2.11.1"
+markdown = ">=3.2.1"
+markupsafe = ">=2.0.1"
+mergedeep = ">=1.3.4"
+packaging = ">=20.5"
+pathspec = ">=0.11.1"
+platformdirs = ">=2.2.0"
+pyyaml = ">=5.1"
+pyyaml-env-tag = ">=0.1"
+watchdog = ">=2.0"
+
+[package.extras]
+i18n = ["babel (>=2.9.0)"]
+min-versions = ["babel (==2.9.0)", "click (==7.0)", "colorama (==0.4)", "ghp-import (==1.0)", "importlib-metadata (==4.3)", "jinja2 (==2.11.1)", "markdown (==3.2.1)", "markupsafe (==2.0.1)", "mergedeep (==1.3.4)", "packaging (==20.5)", "pathspec (==0.11.1)", "platformdirs (==2.2.0)", "pyyaml (==5.1)", "pyyaml-env-tag (==0.1)", "typing-extensions (==3.10)", "watchdog (==2.0)"]
+
+[[package]]
+name = "mkdocs-material"
+version = "9.1.15"
+description = "Documentation that simply works"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "mkdocs_material-9.1.15-py3-none-any.whl", hash = "sha256:b49e12869ab464558e2dd3c5792da5b748a7e0c48ee83b4d05715f98125a7a39"},
+ {file = "mkdocs_material-9.1.15.tar.gz", hash = "sha256:8513ab847c9a541ed3d11a3a7eed556caf72991ee786c31c5aac6691a121088a"},
+]
+
+[package.dependencies]
+colorama = ">=0.4"
+jinja2 = ">=3.0"
+markdown = ">=3.2"
+mkdocs = ">=1.4.2"
+mkdocs-material-extensions = ">=1.1"
+pygments = ">=2.14"
+pymdown-extensions = ">=9.9.1"
+regex = ">=2022.4.24"
+requests = ">=2.26"
+
+[[package]]
+name = "mkdocs-material-extensions"
+version = "1.3"
+description = "Extension pack for Python Markdown and MkDocs Material."
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "mkdocs_material_extensions-1.3-py3-none-any.whl", hash = "sha256:0297cc48ba68a9fdd1ef3780a3b41b534b0d0df1d1181a44676fda5f464eeadc"},
+ {file = "mkdocs_material_extensions-1.3.tar.gz", hash = "sha256:f0446091503acb110a7cab9349cbc90eeac51b58d1caa92a704a81ca1e24ddbd"},
+]
+
+[[package]]
+name = "mkdocs-version-annotations"
+version = "1.0.0"
+description = "MkDocs plugin to add custom admonitions for documenting version differences"
+optional = false
+python-versions = ">=3.7,<4.0"
+files = [
+ {file = "mkdocs-version-annotations-1.0.0.tar.gz", hash = "sha256:6786024b37d27b330fda240b76ebec8e7ce48bd5a9d7a66e99804559d088dffa"},
+ {file = "mkdocs_version_annotations-1.0.0-py3-none-any.whl", hash = "sha256:385004eb4a7530dd87a227e08cd907ce7a8fe21fdf297720a4149c511bcf05f5"},
+]
+
[[package]]
name = "mypy-extensions"
version = "1.0.0"
@@ -661,6 +811,24 @@ typing-extensions = {version = ">=3.10.0", markers = "python_version < \"3.10\""
spelling = ["pyenchant (>=3.2,<4.0)"]
testutils = ["gitpython (>3)"]
+[[package]]
+name = "pymdown-extensions"
+version = "10.3.1"
+description = "Extension pack for Python Markdown."
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "pymdown_extensions-10.3.1-py3-none-any.whl", hash = "sha256:8cba67beb2a1318cdaf742d09dff7c0fc4cafcc290147ade0f8fb7b71522711a"},
+ {file = "pymdown_extensions-10.3.1.tar.gz", hash = "sha256:f6c79941498a458852853872e379e7bab63888361ba20992fc8b4f8a9b61735e"},
+]
+
+[package.dependencies]
+markdown = ">=3.2"
+pyyaml = "*"
+
+[package.extras]
+extra = ["pygments (>=2.12)"]
+
[[package]]
name = "pytest"
version = "7.4.0"
@@ -741,6 +909,7 @@ files = [
{file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938"},
{file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d"},
{file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba336e390cd8e4d1739f42dfe9bb83a3cc2e80f567d8805e11b46f4a943f5515"},
+ {file = "PyYAML-6.0.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:326c013efe8048858a6d312ddd31d56e468118ad4cdeda36c719bf5bb6192290"},
{file = "PyYAML-6.0.1-cp310-cp310-win32.whl", hash = "sha256:bd4af7373a854424dabd882decdc5579653d7868b8fb26dc7d0e99f823aa5924"},
{file = "PyYAML-6.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:fd1592b3fdf65fff2ad0004b5e363300ef59ced41c2e6b3a99d4089fa8c5435d"},
{file = "PyYAML-6.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007"},
@@ -748,8 +917,15 @@ files = [
{file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d"},
{file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc"},
{file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673"},
+ {file = "PyYAML-6.0.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e7d73685e87afe9f3b36c799222440d6cf362062f78be1013661b00c5c6f678b"},
{file = "PyYAML-6.0.1-cp311-cp311-win32.whl", hash = "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741"},
{file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"},
+ {file = "PyYAML-6.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28"},
+ {file = "PyYAML-6.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9"},
+ {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0"},
+ {file = "PyYAML-6.0.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4"},
+ {file = "PyYAML-6.0.1-cp312-cp312-win32.whl", hash = "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54"},
+ {file = "PyYAML-6.0.1-cp312-cp312-win_amd64.whl", hash = "sha256:0d3304d8c0adc42be59c5f8a4d9e3d7379e6955ad754aa9d6ab7a398b59dd1df"},
{file = "PyYAML-6.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47"},
{file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98"},
{file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c"},
@@ -766,6 +942,7 @@ files = [
{file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5"},
{file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696"},
{file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735"},
+ {file = "PyYAML-6.0.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:49a183be227561de579b4a36efbb21b3eab9651dd81b1858589f796549873dd6"},
{file = "PyYAML-6.0.1-cp38-cp38-win32.whl", hash = "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206"},
{file = "PyYAML-6.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62"},
{file = "PyYAML-6.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8"},
@@ -773,11 +950,123 @@ files = [
{file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5773183b6446b2c99bb77e77595dd486303b4faab2b086e7b17bc6bef28865f6"},
{file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b786eecbdf8499b9ca1d697215862083bd6d2a99965554781d0d8d1ad31e13a0"},
{file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1bf2925a1ecd43da378f4db9e4f799775d6367bdb94671027b73b393a7c42c"},
+ {file = "PyYAML-6.0.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:04ac92ad1925b2cff1db0cfebffb6ffc43457495c9b3c39d3fcae417d7125dc5"},
{file = "PyYAML-6.0.1-cp39-cp39-win32.whl", hash = "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c"},
{file = "PyYAML-6.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486"},
{file = "PyYAML-6.0.1.tar.gz", hash = "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43"},
]
+[[package]]
+name = "pyyaml-env-tag"
+version = "0.1"
+description = "A custom YAML tag for referencing environment variables in YAML files. "
+optional = false
+python-versions = ">=3.6"
+files = [
+ {file = "pyyaml_env_tag-0.1-py3-none-any.whl", hash = "sha256:af31106dec8a4d68c60207c1886031cbf839b68aa7abccdb19868200532c2069"},
+ {file = "pyyaml_env_tag-0.1.tar.gz", hash = "sha256:70092675bda14fdec33b31ba77e7543de9ddc88f2e5b99160396572d11525bdb"},
+]
+
+[package.dependencies]
+pyyaml = "*"
+
+[[package]]
+name = "regex"
+version = "2023.10.3"
+description = "Alternative regular expression module, to replace re."
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "regex-2023.10.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:4c34d4f73ea738223a094d8e0ffd6d2c1a1b4c175da34d6b0de3d8d69bee6bcc"},
+ {file = "regex-2023.10.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a8f4e49fc3ce020f65411432183e6775f24e02dff617281094ba6ab079ef0915"},
+ {file = "regex-2023.10.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4cd1bccf99d3ef1ab6ba835308ad85be040e6a11b0977ef7ea8c8005f01a3c29"},
+ {file = "regex-2023.10.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:81dce2ddc9f6e8f543d94b05d56e70d03a0774d32f6cca53e978dc01e4fc75b8"},
+ {file = "regex-2023.10.3-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9c6b4d23c04831e3ab61717a707a5d763b300213db49ca680edf8bf13ab5d91b"},
+ {file = "regex-2023.10.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c15ad0aee158a15e17e0495e1e18741573d04eb6da06d8b84af726cfc1ed02ee"},
+ {file = "regex-2023.10.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6239d4e2e0b52c8bd38c51b760cd870069f0bdf99700a62cd509d7a031749a55"},
+ {file = "regex-2023.10.3-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:4a8bf76e3182797c6b1afa5b822d1d5802ff30284abe4599e1247be4fd6b03be"},
+ {file = "regex-2023.10.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d9c727bbcf0065cbb20f39d2b4f932f8fa1631c3e01fcedc979bd4f51fe051c5"},
+ {file = "regex-2023.10.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:3ccf2716add72f80714b9a63899b67fa711b654be3fcdd34fa391d2d274ce767"},
+ {file = "regex-2023.10.3-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:107ac60d1bfdc3edb53be75e2a52aff7481b92817cfdddd9b4519ccf0e54a6ff"},
+ {file = "regex-2023.10.3-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:00ba3c9818e33f1fa974693fb55d24cdc8ebafcb2e4207680669d8f8d7cca79a"},
+ {file = "regex-2023.10.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:f0a47efb1dbef13af9c9a54a94a0b814902e547b7f21acb29434504d18f36e3a"},
+ {file = "regex-2023.10.3-cp310-cp310-win32.whl", hash = "sha256:36362386b813fa6c9146da6149a001b7bd063dabc4d49522a1f7aa65b725c7ec"},
+ {file = "regex-2023.10.3-cp310-cp310-win_amd64.whl", hash = "sha256:c65a3b5330b54103e7d21cac3f6bf3900d46f6d50138d73343d9e5b2900b2353"},
+ {file = "regex-2023.10.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:90a79bce019c442604662d17bf69df99090e24cdc6ad95b18b6725c2988a490e"},
+ {file = "regex-2023.10.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:c7964c2183c3e6cce3f497e3a9f49d182e969f2dc3aeeadfa18945ff7bdd7051"},
+ {file = "regex-2023.10.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ef80829117a8061f974b2fda8ec799717242353bff55f8a29411794d635d964"},
+ {file = "regex-2023.10.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5addc9d0209a9afca5fc070f93b726bf7003bd63a427f65ef797a931782e7edc"},
+ {file = "regex-2023.10.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c148bec483cc4b421562b4bcedb8e28a3b84fcc8f0aa4418e10898f3c2c0eb9b"},
+ {file = "regex-2023.10.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8d1f21af4c1539051049796a0f50aa342f9a27cde57318f2fc41ed50b0dbc4ac"},
+ {file = "regex-2023.10.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0b9ac09853b2a3e0d0082104036579809679e7715671cfbf89d83c1cb2a30f58"},
+ {file = "regex-2023.10.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:ebedc192abbc7fd13c5ee800e83a6df252bec691eb2c4bedc9f8b2e2903f5e2a"},
+ {file = "regex-2023.10.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:d8a993c0a0ffd5f2d3bda23d0cd75e7086736f8f8268de8a82fbc4bd0ac6791e"},
+ {file = "regex-2023.10.3-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:be6b7b8d42d3090b6c80793524fa66c57ad7ee3fe9722b258aec6d0672543fd0"},
+ {file = "regex-2023.10.3-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:4023e2efc35a30e66e938de5aef42b520c20e7eda7bb5fb12c35e5d09a4c43f6"},
+ {file = "regex-2023.10.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:0d47840dc05e0ba04fe2e26f15126de7c755496d5a8aae4a08bda4dd8d646c54"},
+ {file = "regex-2023.10.3-cp311-cp311-win32.whl", hash = "sha256:9145f092b5d1977ec8c0ab46e7b3381b2fd069957b9862a43bd383e5c01d18c2"},
+ {file = "regex-2023.10.3-cp311-cp311-win_amd64.whl", hash = "sha256:b6104f9a46bd8743e4f738afef69b153c4b8b592d35ae46db07fc28ae3d5fb7c"},
+ {file = "regex-2023.10.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:bff507ae210371d4b1fe316d03433ac099f184d570a1a611e541923f78f05037"},
+ {file = "regex-2023.10.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:be5e22bbb67924dea15039c3282fa4cc6cdfbe0cbbd1c0515f9223186fc2ec5f"},
+ {file = "regex-2023.10.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4a992f702c9be9c72fa46f01ca6e18d131906a7180950958f766c2aa294d4b41"},
+ {file = "regex-2023.10.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7434a61b158be563c1362d9071358f8ab91b8d928728cd2882af060481244c9e"},
+ {file = "regex-2023.10.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c2169b2dcabf4e608416f7f9468737583ce5f0a6e8677c4efbf795ce81109d7c"},
+ {file = "regex-2023.10.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a9e908ef5889cda4de038892b9accc36d33d72fb3e12c747e2799a0e806ec841"},
+ {file = "regex-2023.10.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:12bd4bc2c632742c7ce20db48e0d99afdc05e03f0b4c1af90542e05b809a03d9"},
+ {file = "regex-2023.10.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:bc72c231f5449d86d6c7d9cc7cd819b6eb30134bb770b8cfdc0765e48ef9c420"},
+ {file = "regex-2023.10.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:bce8814b076f0ce5766dc87d5a056b0e9437b8e0cd351b9a6c4e1134a7dfbda9"},
+ {file = "regex-2023.10.3-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:ba7cd6dc4d585ea544c1412019921570ebd8a597fabf475acc4528210d7c4a6f"},
+ {file = "regex-2023.10.3-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b0c7d2f698e83f15228ba41c135501cfe7d5740181d5903e250e47f617eb4292"},
+ {file = "regex-2023.10.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:5a8f91c64f390ecee09ff793319f30a0f32492e99f5dc1c72bc361f23ccd0a9a"},
+ {file = "regex-2023.10.3-cp312-cp312-win32.whl", hash = "sha256:ad08a69728ff3c79866d729b095872afe1e0557251da4abb2c5faff15a91d19a"},
+ {file = "regex-2023.10.3-cp312-cp312-win_amd64.whl", hash = "sha256:39cdf8d141d6d44e8d5a12a8569d5a227f645c87df4f92179bd06e2e2705e76b"},
+ {file = "regex-2023.10.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:4a3ee019a9befe84fa3e917a2dd378807e423d013377a884c1970a3c2792d293"},
+ {file = "regex-2023.10.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:76066d7ff61ba6bf3cb5efe2428fc82aac91802844c022d849a1f0f53820502d"},
+ {file = "regex-2023.10.3-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bfe50b61bab1b1ec260fa7cd91106fa9fece57e6beba05630afe27c71259c59b"},
+ {file = "regex-2023.10.3-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9fd88f373cb71e6b59b7fa597e47e518282455c2734fd4306a05ca219a1991b0"},
+ {file = "regex-2023.10.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b3ab05a182c7937fb374f7e946f04fb23a0c0699c0450e9fb02ef567412d2fa3"},
+ {file = "regex-2023.10.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dac37cf08fcf2094159922edc7a2784cfcc5c70f8354469f79ed085f0328ebdf"},
+ {file = "regex-2023.10.3-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:e54ddd0bb8fb626aa1f9ba7b36629564544954fff9669b15da3610c22b9a0991"},
+ {file = "regex-2023.10.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:3367007ad1951fde612bf65b0dffc8fd681a4ab98ac86957d16491400d661302"},
+ {file = "regex-2023.10.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:16f8740eb6dbacc7113e3097b0a36065a02e37b47c936b551805d40340fb9971"},
+ {file = "regex-2023.10.3-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:f4f2ca6df64cbdd27f27b34f35adb640b5d2d77264228554e68deda54456eb11"},
+ {file = "regex-2023.10.3-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:39807cbcbe406efca2a233884e169d056c35aa7e9f343d4e78665246a332f597"},
+ {file = "regex-2023.10.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:7eece6fbd3eae4a92d7c748ae825cbc1ee41a89bb1c3db05b5578ed3cfcfd7cb"},
+ {file = "regex-2023.10.3-cp37-cp37m-win32.whl", hash = "sha256:ce615c92d90df8373d9e13acddd154152645c0dc060871abf6bd43809673d20a"},
+ {file = "regex-2023.10.3-cp37-cp37m-win_amd64.whl", hash = "sha256:0f649fa32fe734c4abdfd4edbb8381c74abf5f34bc0b3271ce687b23729299ed"},
+ {file = "regex-2023.10.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:9b98b7681a9437262947f41c7fac567c7e1f6eddd94b0483596d320092004533"},
+ {file = "regex-2023.10.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:91dc1d531f80c862441d7b66c4505cd6ea9d312f01fb2f4654f40c6fdf5cc37a"},
+ {file = "regex-2023.10.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:82fcc1f1cc3ff1ab8a57ba619b149b907072e750815c5ba63e7aa2e1163384a4"},
+ {file = "regex-2023.10.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7979b834ec7a33aafae34a90aad9f914c41fd6eaa8474e66953f3f6f7cbd4368"},
+ {file = "regex-2023.10.3-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ef71561f82a89af6cfcbee47f0fabfdb6e63788a9258e913955d89fdd96902ab"},
+ {file = "regex-2023.10.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd829712de97753367153ed84f2de752b86cd1f7a88b55a3a775eb52eafe8a94"},
+ {file = "regex-2023.10.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:00e871d83a45eee2f8688d7e6849609c2ca2a04a6d48fba3dff4deef35d14f07"},
+ {file = "regex-2023.10.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:706e7b739fdd17cb89e1fbf712d9dc21311fc2333f6d435eac2d4ee81985098c"},
+ {file = "regex-2023.10.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:cc3f1c053b73f20c7ad88b0d1d23be7e7b3901229ce89f5000a8399746a6e039"},
+ {file = "regex-2023.10.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:6f85739e80d13644b981a88f529d79c5bdf646b460ba190bffcaf6d57b2a9863"},
+ {file = "regex-2023.10.3-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:741ba2f511cc9626b7561a440f87d658aabb3d6b744a86a3c025f866b4d19e7f"},
+ {file = "regex-2023.10.3-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:e77c90ab5997e85901da85131fd36acd0ed2221368199b65f0d11bca44549711"},
+ {file = "regex-2023.10.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:979c24cbefaf2420c4e377ecd1f165ea08cc3d1fbb44bdc51bccbbf7c66a2cb4"},
+ {file = "regex-2023.10.3-cp38-cp38-win32.whl", hash = "sha256:58837f9d221744d4c92d2cf7201c6acd19623b50c643b56992cbd2b745485d3d"},
+ {file = "regex-2023.10.3-cp38-cp38-win_amd64.whl", hash = "sha256:c55853684fe08d4897c37dfc5faeff70607a5f1806c8be148f1695be4a63414b"},
+ {file = "regex-2023.10.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:2c54e23836650bdf2c18222c87f6f840d4943944146ca479858404fedeb9f9af"},
+ {file = "regex-2023.10.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:69c0771ca5653c7d4b65203cbfc5e66db9375f1078689459fe196fe08b7b4930"},
+ {file = "regex-2023.10.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6ac965a998e1388e6ff2e9781f499ad1eaa41e962a40d11c7823c9952c77123e"},
+ {file = "regex-2023.10.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1c0e8fae5b27caa34177bdfa5a960c46ff2f78ee2d45c6db15ae3f64ecadde14"},
+ {file = "regex-2023.10.3-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:6c56c3d47da04f921b73ff9415fbaa939f684d47293f071aa9cbb13c94afc17d"},
+ {file = "regex-2023.10.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7ef1e014eed78ab650bef9a6a9cbe50b052c0aebe553fb2881e0453717573f52"},
+ {file = "regex-2023.10.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d29338556a59423d9ff7b6eb0cb89ead2b0875e08fe522f3e068b955c3e7b59b"},
+ {file = "regex-2023.10.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9c6d0ced3c06d0f183b73d3c5920727268d2201aa0fe6d55c60d68c792ff3588"},
+ {file = "regex-2023.10.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:994645a46c6a740ee8ce8df7911d4aee458d9b1bc5639bc968226763d07f00fa"},
+ {file = "regex-2023.10.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:66e2fe786ef28da2b28e222c89502b2af984858091675044d93cb50e6f46d7af"},
+ {file = "regex-2023.10.3-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:11175910f62b2b8c055f2b089e0fedd694fe2be3941b3e2633653bc51064c528"},
+ {file = "regex-2023.10.3-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:06e9abc0e4c9ab4779c74ad99c3fc10d3967d03114449acc2c2762ad4472b8ca"},
+ {file = "regex-2023.10.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:fb02e4257376ae25c6dd95a5aec377f9b18c09be6ebdefa7ad209b9137b73d48"},
+ {file = "regex-2023.10.3-cp39-cp39-win32.whl", hash = "sha256:3b2c3502603fab52d7619b882c25a6850b766ebd1b18de3df23b2f939360e1bd"},
+ {file = "regex-2023.10.3-cp39-cp39-win_amd64.whl", hash = "sha256:adbccd17dcaff65704c856bd29951c58a1bd4b2b0f8ad6b826dbd543fe740988"},
+ {file = "regex-2023.10.3.tar.gz", hash = "sha256:3fef4f844d2290ee0ba57addcec17eec9e3df73f10a2748485dfd6a3a188cc0f"},
+]
+
[[package]]
name = "requests"
version = "2.31.0"
@@ -941,6 +1230,45 @@ secure = ["certifi", "cryptography (>=1.9)", "idna (>=2.0.0)", "pyopenssl (>=17.
socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"]
zstd = ["zstandard (>=0.18.0)"]
+[[package]]
+name = "watchdog"
+version = "3.0.0"
+description = "Filesystem events monitoring"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "watchdog-3.0.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:336adfc6f5cc4e037d52db31194f7581ff744b67382eb6021c868322e32eef41"},
+ {file = "watchdog-3.0.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a70a8dcde91be523c35b2bf96196edc5730edb347e374c7de7cd20c43ed95397"},
+ {file = "watchdog-3.0.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:adfdeab2da79ea2f76f87eb42a3ab1966a5313e5a69a0213a3cc06ef692b0e96"},
+ {file = "watchdog-3.0.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:2b57a1e730af3156d13b7fdddfc23dea6487fceca29fc75c5a868beed29177ae"},
+ {file = "watchdog-3.0.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:7ade88d0d778b1b222adebcc0927428f883db07017618a5e684fd03b83342bd9"},
+ {file = "watchdog-3.0.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:7e447d172af52ad204d19982739aa2346245cc5ba6f579d16dac4bfec226d2e7"},
+ {file = "watchdog-3.0.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:9fac43a7466eb73e64a9940ac9ed6369baa39b3bf221ae23493a9ec4d0022674"},
+ {file = "watchdog-3.0.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:8ae9cda41fa114e28faf86cb137d751a17ffd0316d1c34ccf2235e8a84365c7f"},
+ {file = "watchdog-3.0.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:25f70b4aa53bd743729c7475d7ec41093a580528b100e9a8c5b5efe8899592fc"},
+ {file = "watchdog-3.0.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4f94069eb16657d2c6faada4624c39464f65c05606af50bb7902e036e3219be3"},
+ {file = "watchdog-3.0.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:7c5f84b5194c24dd573fa6472685b2a27cc5a17fe5f7b6fd40345378ca6812e3"},
+ {file = "watchdog-3.0.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3aa7f6a12e831ddfe78cdd4f8996af9cf334fd6346531b16cec61c3b3c0d8da0"},
+ {file = "watchdog-3.0.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:233b5817932685d39a7896b1090353fc8efc1ef99c9c054e46c8002561252fb8"},
+ {file = "watchdog-3.0.0-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:13bbbb462ee42ec3c5723e1205be8ced776f05b100e4737518c67c8325cf6100"},
+ {file = "watchdog-3.0.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:8f3ceecd20d71067c7fd4c9e832d4e22584318983cabc013dbf3f70ea95de346"},
+ {file = "watchdog-3.0.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:c9d8c8ec7efb887333cf71e328e39cffbf771d8f8f95d308ea4125bf5f90ba64"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_aarch64.whl", hash = "sha256:0e06ab8858a76e1219e68c7573dfeba9dd1c0219476c5a44d5333b01d7e1743a"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_armv7l.whl", hash = "sha256:d00e6be486affb5781468457b21a6cbe848c33ef43f9ea4a73b4882e5f188a44"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_i686.whl", hash = "sha256:c07253088265c363d1ddf4b3cdb808d59a0468ecd017770ed716991620b8f77a"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_ppc64.whl", hash = "sha256:5113334cf8cf0ac8cd45e1f8309a603291b614191c9add34d33075727a967709"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_ppc64le.whl", hash = "sha256:51f90f73b4697bac9c9a78394c3acbbd331ccd3655c11be1a15ae6fe289a8c83"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_s390x.whl", hash = "sha256:ba07e92756c97e3aca0912b5cbc4e5ad802f4557212788e72a72a47ff376950d"},
+ {file = "watchdog-3.0.0-py3-none-manylinux2014_x86_64.whl", hash = "sha256:d429c2430c93b7903914e4db9a966c7f2b068dd2ebdd2fa9b9ce094c7d459f33"},
+ {file = "watchdog-3.0.0-py3-none-win32.whl", hash = "sha256:3ed7c71a9dccfe838c2f0b6314ed0d9b22e77d268c67e015450a29036a81f60f"},
+ {file = "watchdog-3.0.0-py3-none-win_amd64.whl", hash = "sha256:4c9956d27be0bb08fc5f30d9d0179a855436e655f046d288e2bcc11adfae893c"},
+ {file = "watchdog-3.0.0-py3-none-win_ia64.whl", hash = "sha256:5d9f3a10e02d7371cd929b5d8f11e87d4bad890212ed3901f9b4d68767bee759"},
+ {file = "watchdog-3.0.0.tar.gz", hash = "sha256:4d98a320595da7a7c5a18fc48cb633c2e73cda78f93cac2ef42d42bf609a33f9"},
+]
+
+[package.extras]
+watchmedo = ["PyYAML (>=3.10)"]
+
[[package]]
name = "wrapt"
version = "1.15.0"
@@ -1043,7 +1371,22 @@ pyyaml = "*"
[package.extras]
dev = ["doc8", "flake8", "flake8-import-order", "rstcheck[sphinx]", "sphinx"]
+[[package]]
+name = "zipp"
+version = "3.17.0"
+description = "Backport of pathlib-compatible object wrapper for zip files"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "zipp-3.17.0-py3-none-any.whl", hash = "sha256:0e923e726174922dce09c53c59ad483ff7bbb8e572e00c7f7c46b88556409f31"},
+ {file = "zipp-3.17.0.tar.gz", hash = "sha256:84e64a1c28cf7e91ed2078bb8cc8c259cb19b76942096c8d7b84947690cabaf0"},
+]
+
+[package.extras]
+docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"]
+testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-ignore-flaky", "pytest-mypy (>=0.9.1)", "pytest-ruff"]
+
[metadata]
lock-version = "2.0"
python-versions = "^3.8"
-content-hash = "061386bb7b2c34c852255c1d22e19551e14f5100066927a6d66c2585ba45f3e4"
+content-hash = "6d577b6593a2cc508c68437af91310e209bcea81ff47f5abcd9872adc3af7c48"
diff --git a/pyproject.toml b/pyproject.toml
index 6d1313a..22a007b 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -26,6 +26,14 @@ pytest-cookies = "*"
ruff = "*"
yamllint = "*"
+[tool.poetry.group.docs.dependencies]
+# Rendering docs to HTML
+mkdocs = "1.5.2"
+# Material for MkDocs theme
+mkdocs-material = "9.1.15"
+# Render custom markdown for version added/changed/remove notes
+mkdocs-version-annotations = "1.0.0"
+
[tool.black]
line-length = 120
target-version = ['py38']