Skip to content

Commit

Permalink
Detect the EXAMPLES format and use it (#225)
Browse files Browse the repository at this point in the history
* Detect the EXAMPLES format and use it.

* Use field validator instead of root_validator.
  • Loading branch information
felixfontein authored Dec 9, 2023
1 parent bd726b6 commit 4fa1813
Show file tree
Hide file tree
Showing 24 changed files with 50 additions and 4 deletions.
2 changes: 2 additions & 0 deletions changelogs/fragments/225-examples-format.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
bugfixes:
- "When ``EXAMPLES`` has the format specified by ``# fmt: <format>``, this value is used to determine the code block type (https://github.com/ansible-community/antsibull-docs/pull/225)."
Original file line number Diff line number Diff line change
Expand Up @@ -331,7 +331,7 @@ See Also
Examples
--------

.. code-block:: yaml+jinja
.. code-block:: @{ examples_format | rst_format(for_sphinx=true) }@

@{ examples | indent(4, True) }@

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -260,7 +260,7 @@ See Also
Examples
--------

.. code-block:: yaml
.. code-block:: @{ examples_format | rst_format }@

@{ examples | indent(4, True) }@

Expand Down
2 changes: 2 additions & 0 deletions src/antsibull_docs/jinja2/environment.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
plugin_shortname,
remove_options_from_list,
rst_fmt,
rst_format,
rst_xline,
suboption_depth,
to_ini_value,
Expand Down Expand Up @@ -153,6 +154,7 @@ def doc_environment(
env.filters["collection_name"] = collection_name
env.filters["plugin_shortname"] = plugin_shortname
env.filters["suboption_depth"] = suboption_depth
env.filters["rst_format"] = rst_format
if collection_url is not None:
env.filters["collection_url"] = collection_url
if collection_install is not None:
Expand Down
13 changes: 13 additions & 0 deletions src/antsibull_docs/jinja2/filters.py
Original file line number Diff line number Diff line change
Expand Up @@ -238,3 +238,16 @@ def suboption_depth(
if subdata and isinstance(subdata, Mapping):
subdepth = max(subdepth, suboption_depth(subdata.items(), subkey))
return subdepth + 1


def rst_format(fmt: str, for_sphinx: bool = False) -> str:
if fmt == "ini":
return "ini"
if fmt == "toml":
return "toml"
if fmt == "yaml":
return "yaml+jinja" if for_sphinx else "yaml"
if fmt == "json":
return "json"
# Unknown format: no syntax highlighting
return "text"
12 changes: 12 additions & 0 deletions src/antsibull_docs/schemas/docs/plugin.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
# to initialize the attributes when data is loaded into them.
# pyre-ignore-all-errors[13]

import re
import typing as t

import pydantic as p
Expand All @@ -32,6 +33,8 @@

_SENTINEL = object()

_EXAMPLES_FMT_RE = re.compile(r"^# fmt:\s+(\S+)")


class OptionCliSchema(BaseModel):
name: str = REQUIRED_CLI_F
Expand Down Expand Up @@ -207,6 +210,15 @@ class PluginDocSchema(BaseModel):

class PluginExamplesSchema(BaseModel):
examples: str = ""
examples_format: str = "yaml"

@p.validator("examples_format", always=True)
# pylint:disable=no-self-argument
def extract_examples_format(cls, value: t.Any, values: dict[str, t.Any]):
if isinstance(examples := values.get("examples"), str):
if fmt_match := _EXAMPLES_FMT_RE.match(examples.lstrip()):
value = fmt_match.group(1)
return value

@p.validator("examples", pre=True)
# pylint:disable=no-self-argument
Expand Down
1 change: 1 addition & 0 deletions src/antsibull_docs/write_docs/plugins.py
Original file line number Diff line number Diff line change
Expand Up @@ -236,6 +236,7 @@ def create_plugin_rst(
plugin_name=plugin_name,
doc=plugin_record["doc"],
examples=plugin_record["examples"],
examples_format=plugin_record["examples_format"],
returndocs=plugin_record["return"],
nonfatal_errors=nonfatal_errors,
edit_on_github_url=edit_on_github_url,
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_become_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_cache_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "\nexample: >\n To enable, add this to your ansible.cfg file in the defaults block\n [defaults]\n callback_whitelist = aws_resource_actions\nsample output: >\n#\n# AWS ACTIONS: ['s3:PutBucketAcl', 's3:HeadObject', 's3:DeleteObject', 's3:PutObjectAcl', 's3:CreateMultipartUpload',\n# 's3:DeleteBucket', 's3:GetObject', 's3:DeleteObjects', 's3:CreateBucket', 's3:CompleteMultipartUpload',\n# 's3:ListObjectsV2', 's3:HeadBucket', 's3:UploadPart', 's3:PutObject']\n#\nsample output: >\n#\n# AWS ACTIONS: ['ec2:DescribeVpcAttribute', 'ec2:DescribeVpcClassicLink', 'ec2:ModifyVpcAttribute', 'ec2:CreateTags',\n# 'sts:GetCallerIdentity', 'ec2:DescribeSecurityGroups', 'ec2:DescribeTags', 'ec2:DescribeVpcs', 'ec2:CreateVpc']\n#\n",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_cliconf_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -497,6 +497,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_filter_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,7 @@
"version_added": "1.9",
"version_added_collection": "ansible.builtin"
},
"examples_format": "yaml",
"examples": "# set first 10 volumes rw, rest as dp\nvolume_mode: \"{{ (item|int < 11)|ternary('rw', 'dp') }}\"\n\n# choose correct vpc subnet id, note that vars as values are evaluated even if not returned\nvpc_subnet_id: \"{{ (ec2_subnet_type == 'public') | ternary(ec2_vpc_public_subnet_id, ec2_vpc_private_subnet_id) }}\"\n\n- name: service-foo, use systemd module unless upstart is present, then use old service module\n service:\n state: restarted\n enabled: yes\n use: \"{{ (ansible_service_mgr == 'upstart') | ternary('service', 'systemd') }}\"\n",
"metadata": null,
"return_": {
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_httpapi_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -665,6 +665,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "\n# Minimal example using environment vars or instance role credentials\n# Fetch all hosts in us-east-1, the hostname is the public DNS if it exists, otherwise the private IP address\nplugin: aws_ec2\nregions:\n - us-east-1\n\n# Example using filters, ignoring permission errors, and specifying the hostname precedence\nplugin: aws_ec2\nboto_profile: aws_profile\n# Populate inventory with instances in these regions\nregions:\n - us-east-1\n - us-east-2\nfilters:\n # All instances with their `Environment` tag set to `dev`\n tag:Environment: dev\n # All dev and QA hosts\n tag:Environment:\n - dev\n - qa\n instance.group-id: sg-xxxxxxxx\n# Ignores 403 errors rather than failing\nstrict_permissions: False\n# Note: I(hostnames) sets the inventory_hostname. To modify ansible_host without modifying\n# inventory_hostname use compose (see example below).\nhostnames:\n - tag:Name=Tag1,Name=Tag2 # Return specific hosts only\n - tag:CustomDNSName\n - dns-name\n - private-ip-address\n\n# Example using constructed features to create groups and set ansible_host\nplugin: aws_ec2\nregions:\n - us-east-1\n - us-west-1\n# keyed_groups may be used to create custom groups\nstrict: False\nkeyed_groups:\n # Add e.g. x86_64 hosts to an arch_x86_64 group\n - prefix: arch\n key: 'architecture'\n # Add hosts to tag_Name_Value groups for each Name/Value tag pair\n - prefix: tag\n key: tags\n # Add hosts to e.g. instance_type_z3_tiny\n - prefix: instance_type\n key: instance_type\n # Create security_groups_sg_abcd1234 group for each SG\n - key: 'security_groups|json_query(\"[].group_id\")'\n prefix: 'security_groups'\n # Create a group for each value of the Application tag\n - key: tags.Application\n separator: ''\n # Create a group per region e.g. aws_region_us_east_2\n - key: placement.region\n prefix: aws_region\n # Create a group (or groups) based on the value of a custom tag \"Role\" and add them to a metagroup called \"project\"\n - key: tags['Role']\n prefix: foo\n parent_group: \"project\"\n# Set individual variables with compose\ncompose:\n # Use the private IP address to connect to the host\n # (note: this does not modify inventory_hostname, which is set via I(hostnames))\n ansible_host: private_ip_address\n",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_lookup_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -275,6 +275,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "\n - debug:\n msg: 'key contains {{item}}'\n with_consul_kv:\n - 'key/to/retrieve'\n\n - name: Parameters can be provided after the key be more specific about what to retrieve\n debug:\n msg: 'key contains {{item}}'\n with_consul_kv:\n - 'key/to recurse=true token=E6C060A9-26FB-407A-B83E-12DDAFCB4D98'\n\n - name: retrieving a KV from a remote cluster on non default port\n debug:\n msg: \"{{ lookup('consul_kv', 'my/key', host='10.10.10.10', port='2000') }}\"\n",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_module_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -250,6 +250,7 @@
"version_added": "0.9",
"version_added_collection": "foo.bar"
},
"examples_format": "yaml",
"examples": "\n- name: Add host to group 'just_created' with variable foo=42\n add_host:\n name: '{{ ip_from_ec2 }}'\n groups: just_created\n foo: 42\n\n- name: Add host to multiple groups\n add_host:\n hostname: '{{ new_ip }}'\n groups:\n - group1\n - group2\n\n- name: Add a host with a non-standard port local to your machines\n add_host:\n name: '{{ new_ip }}:{{ new_port }}'\n\n- name: Add a host alias that we reach through a tunnel (Ansible 1.9 and older)\n add_host:\n hostname: '{{ new_ip }}'\n ansible_ssh_host: '{{ inventory_hostname }}'\n ansible_ssh_port: '{{ new_port }}'\n\n- name: Add a host alias that we reach through a tunnel (Ansible 2.0 and newer)\n add_host:\n hostname: '{{ new_ip }}'\n ansible_host: '{{ inventory_hostname }}'\n ansible_port: '{{ new_port }}'\n\n- name: Ensure inventory vars are set to the same value as the inventory_hostname has (close to pre Ansible 2.4 behaviour)\n add_host:\n hostname: charlie\n inventory_dir: '{{ inventory_dir }}'\n",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_netconf_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
"version_added": "historical",
"version_added_collection": ""
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_shell_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -223,6 +223,7 @@
"version_added": "historical",
"version_added_collection": "foo.bar"
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
"version_added": "2.0",
"version_added_collection": "foo.bar"
},
"examples_format": "yaml",
"examples": "",
"metadata": {
"status": [
Expand Down
1 change: 1 addition & 0 deletions tests/functional/schema/good_data/one_test_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@
"version_added": "1.9",
"version_added_collection": "ansible.builtin"
},
"examples_format": "yaml",
"examples": "# test 'status' to know how to respond\n{{ (taskresults is changed }}\n",
"metadata": null,
"return_": {
Expand Down
2 changes: 1 addition & 1 deletion tests/functional/schema/good_data/one_vars.json
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,7 @@
"version_added": "2.4",
"version_added_collection": "foo.bar"
},
"examples": null,
"examples": " # fmt: ini\n[foo]\nbar = bam",
"metadata": {
"status": [
"preview"
Expand Down
3 changes: 2 additions & 1 deletion tests/functional/schema/good_data/one_vars_results.json
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,8 @@
"version_added": "2.4",
"version_added_collection": "foo.bar"
},
"examples": "",
"examples_format": "ini",
"examples": " # fmt: ini\n[foo]\nbar = bam",
"metadata": {
"status": [
"preview"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1207,6 +1207,7 @@
"version_added": "historical",
"version_added_collection": "ansible.builtin"
},
"examples_format": "yaml",
"examples": "",
"metadata": null,
"return_": {}
Expand Down

0 comments on commit 4fa1813

Please sign in to comment.