mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-04-05 10:10:31 -07:00
Add 'to_prettytable' filter plugin and tests
This commit is contained in:
parent
dc205f6d29
commit
94e7200e42
3 changed files with 317 additions and 0 deletions
182
plugins/filter/to_prettytable.py
Normal file
182
plugins/filter/to_prettytable.py
Normal file
|
@ -0,0 +1,182 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright (c) 2025, Timur Gadiev <timur@example.com>
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
from __future__ import (absolute_import, division, print_function)
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
DOCUMENTATION = '''
|
||||||
|
name: to_prettytable
|
||||||
|
short_description: Format a list of dictionaries as an ASCII table
|
||||||
|
version_added: "8.0.0"
|
||||||
|
author: Timur Gadiev (@tgadiev)
|
||||||
|
description:
|
||||||
|
- This filter takes a list of dictionaries and formats it as an ASCII table using the I(prettytable) Python library.
|
||||||
|
requirements:
|
||||||
|
- prettytable
|
||||||
|
options:
|
||||||
|
_input:
|
||||||
|
description: A list of dictionaries to format.
|
||||||
|
type: list
|
||||||
|
elements: dictionary
|
||||||
|
required: true
|
||||||
|
column_order:
|
||||||
|
description: List of column names to specify the order of columns in the table.
|
||||||
|
type: list
|
||||||
|
elements: string
|
||||||
|
header_names:
|
||||||
|
description: List of custom header names to use instead of dictionary keys.
|
||||||
|
type: list
|
||||||
|
elements: string
|
||||||
|
column_alignments:
|
||||||
|
description: Dictionary of column alignments. Keys are column names, values are alignments.
|
||||||
|
type: dictionary
|
||||||
|
suboptions:
|
||||||
|
alignment:
|
||||||
|
description: Alignment for the column. Must be one of C(left), C(center), C(right), C(l), C(c), or C(r).
|
||||||
|
type: string
|
||||||
|
choices: [left, center, right, l, c, r]
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = '''
|
||||||
|
- name: Display a list of users as a table
|
||||||
|
vars:
|
||||||
|
users:
|
||||||
|
- name: Alice
|
||||||
|
age: 25
|
||||||
|
role: admin
|
||||||
|
- name: Bob
|
||||||
|
age: 30
|
||||||
|
role: user
|
||||||
|
debug:
|
||||||
|
msg: "{{ users | community.general.to_prettytable }}"
|
||||||
|
|
||||||
|
- name: Display a table with custom column ordering
|
||||||
|
debug:
|
||||||
|
msg: "{{ users | community.general.to_prettytable('role', 'name', 'age') }}"
|
||||||
|
|
||||||
|
- name: Display a table with custom headers
|
||||||
|
debug:
|
||||||
|
msg: "{{ users | community.general.to_prettytable(header_names=['User Name', 'User Age', 'User Role']) }}"
|
||||||
|
|
||||||
|
- name: Display a table with custom alignments
|
||||||
|
debug:
|
||||||
|
msg: "{{ users | community.general.to_prettytable(column_alignments={'name': 'center', 'age': 'right', 'role': 'left'}) }}"
|
||||||
|
|
||||||
|
- name: Combine multiple options
|
||||||
|
debug:
|
||||||
|
msg: "{{ users | community.general.to_prettytable(
|
||||||
|
column_order=['role', 'name', 'age'],
|
||||||
|
header_names=['Position', 'Full Name', 'Years'],
|
||||||
|
column_alignments={'name': 'center', 'age': 'right', 'role': 'left'}) }}"
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = '''
|
||||||
|
_value:
|
||||||
|
description: The formatted ASCII table.
|
||||||
|
type: string
|
||||||
|
'''
|
||||||
|
|
||||||
|
try:
|
||||||
|
import prettytable
|
||||||
|
HAS_PRETTYTABLE = True
|
||||||
|
except ImportError:
|
||||||
|
HAS_PRETTYTABLE = False
|
||||||
|
|
||||||
|
from ansible.errors import AnsibleFilterError
|
||||||
|
from ansible.module_utils._text import to_text
|
||||||
|
from ansible.module_utils.six import string_types
|
||||||
|
|
||||||
|
|
||||||
|
def to_prettytable(data, *args, **kwargs):
|
||||||
|
"""Convert a list of dictionaries to an ASCII table.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
data: List of dictionaries to format
|
||||||
|
*args: Optional list of column names to specify column order
|
||||||
|
**kwargs: Optional keyword arguments:
|
||||||
|
- column_order: List of column names to specify the order
|
||||||
|
- header_names: List of custom header names
|
||||||
|
- column_alignments: Dict of column alignments (left, center, right)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
String containing the ASCII table
|
||||||
|
"""
|
||||||
|
if not HAS_PRETTYTABLE:
|
||||||
|
raise AnsibleFilterError(
|
||||||
|
'You need to install "prettytable" Python module to use this filter'
|
||||||
|
)
|
||||||
|
|
||||||
|
if not isinstance(data, list):
|
||||||
|
raise AnsibleFilterError(
|
||||||
|
"Expected a list of dictionaries, got a string"
|
||||||
|
if isinstance(data, string_types)
|
||||||
|
else f"Expected a list of dictionaries, got {type(data).__name__}"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Handle empty data
|
||||||
|
if not data:
|
||||||
|
return "++\n++"
|
||||||
|
|
||||||
|
# Check that all items are dictionaries
|
||||||
|
if not all(isinstance(item, dict) for item in data):
|
||||||
|
invalid_item = next(item for item in data if not isinstance(item, dict))
|
||||||
|
raise AnsibleFilterError(
|
||||||
|
"All items in the list must be dictionaries, got a string"
|
||||||
|
if isinstance(invalid_item, string_types)
|
||||||
|
else f"All items in the list must be dictionaries, got {type(invalid_item).__name__}"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Handle positional argument column order
|
||||||
|
column_order = kwargs.get('column_order', None)
|
||||||
|
if args and not column_order:
|
||||||
|
column_order = list(args)
|
||||||
|
|
||||||
|
# Create the table and configure it
|
||||||
|
table = prettytable.PrettyTable()
|
||||||
|
|
||||||
|
# Determine field names
|
||||||
|
field_names = column_order or list(data[0].keys())
|
||||||
|
|
||||||
|
# Set headers
|
||||||
|
header_names = kwargs.get('header_names', None)
|
||||||
|
table.field_names = header_names if header_names else field_names
|
||||||
|
|
||||||
|
# Configure alignments
|
||||||
|
_configure_alignments(table, field_names, kwargs.get('column_alignments', {}))
|
||||||
|
|
||||||
|
# Add rows
|
||||||
|
rows = [[item.get(col, "") for col in field_names] for item in data]
|
||||||
|
table.add_rows(rows)
|
||||||
|
|
||||||
|
return to_text(table)
|
||||||
|
|
||||||
|
|
||||||
|
def _configure_alignments(table, field_names, column_alignments):
|
||||||
|
"""Configure column alignments for the table.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
table: The PrettyTable instance to configure
|
||||||
|
field_names: List of field names to align
|
||||||
|
column_alignments: Dict of column alignments
|
||||||
|
"""
|
||||||
|
valid_alignments = {"left", "center", "right", "l", "c", "r"}
|
||||||
|
|
||||||
|
if not isinstance(column_alignments, dict):
|
||||||
|
return
|
||||||
|
|
||||||
|
for col_name, alignment in column_alignments.items():
|
||||||
|
if col_name in field_names:
|
||||||
|
alignment = alignment.lower()
|
||||||
|
if alignment in valid_alignments:
|
||||||
|
table.align[col_name] = alignment[0]
|
||||||
|
|
||||||
|
|
||||||
|
class FilterModule(object):
|
||||||
|
"""Ansible core jinja2 filters."""
|
||||||
|
|
||||||
|
def filters(self):
|
||||||
|
return {
|
||||||
|
'to_prettytable': to_prettytable
|
||||||
|
}
|
5
tests/integration/targets/filter_to_prettytable/aliases
Normal file
5
tests/integration/targets/filter_to_prettytable/aliases
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# Copyright (c) Ansible Project
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
azp/posix/1
|
130
tests/integration/targets/filter_to_prettytable/tasks/main.yml
Normal file
130
tests/integration/targets/filter_to_prettytable/tasks/main.yml
Normal file
|
@ -0,0 +1,130 @@
|
||||||
|
---
|
||||||
|
####################################################################
|
||||||
|
# WARNING: These are designed specifically for Ansible tests #
|
||||||
|
# and should not be used as examples of how to write Ansible roles #
|
||||||
|
####################################################################
|
||||||
|
|
||||||
|
# Copyright (c) 2025, Timur Gadiev (@tgadiev)
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
- name: Install required libs
|
||||||
|
pip:
|
||||||
|
name: prettytable
|
||||||
|
state: present
|
||||||
|
delegate_to: localhost
|
||||||
|
become: false
|
||||||
|
|
||||||
|
- name: Set test data
|
||||||
|
set_fact:
|
||||||
|
test_data:
|
||||||
|
- name: Alice
|
||||||
|
age: 25
|
||||||
|
role: admin
|
||||||
|
- name: Bob
|
||||||
|
age: 30
|
||||||
|
role: user
|
||||||
|
data_for_align:
|
||||||
|
- date: 2023-01-01
|
||||||
|
description: Office supplies
|
||||||
|
amount: 123.45
|
||||||
|
|
||||||
|
# Test basic functionality
|
||||||
|
- name: Test basic table creation
|
||||||
|
set_fact:
|
||||||
|
basic_table: '{{ test_data | community.general.to_prettytable }}'
|
||||||
|
|
||||||
|
- name: Verify basic table output
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- basic_table | trim is defined
|
||||||
|
- basic_table | trim is not none
|
||||||
|
|
||||||
|
# Test column ordering
|
||||||
|
- name: Test column ordering
|
||||||
|
set_fact:
|
||||||
|
ordered_table: "{{ test_data | community.general.to_prettytable(column_order=['role', 'name', 'age']) }}"
|
||||||
|
|
||||||
|
- name: Verify ordered table output
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- ordered_table | trim is defined
|
||||||
|
- ordered_table | trim is not none
|
||||||
|
- ordered_table | trim != basic_table | trim
|
||||||
|
|
||||||
|
# Test custom headers
|
||||||
|
- name: Test custom headers
|
||||||
|
set_fact:
|
||||||
|
headers_table: "{{ test_data | community.general.to_prettytable(header_names=['User Name', 'User Age', 'User Role']) }}"
|
||||||
|
|
||||||
|
- name: Verify custom headers output
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- headers_table | trim is defined
|
||||||
|
- headers_table | trim is not none
|
||||||
|
- "headers_table | trim is search('User Name')"
|
||||||
|
- "headers_table | trim is search('User Age')"
|
||||||
|
- "headers_table | trim is search('User Role')"
|
||||||
|
|
||||||
|
# Test alignments
|
||||||
|
- name: Test column alignments
|
||||||
|
set_fact:
|
||||||
|
aligned_table: "{{ data_for_align | community.general.to_prettytable(column_alignments={'amount': 'right', 'description': 'left', 'date': 'center'}) }}"
|
||||||
|
|
||||||
|
- name: Verify aligned table output
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- aligned_table | trim is defined
|
||||||
|
- aligned_table | trim is not none
|
||||||
|
|
||||||
|
# Test combined options
|
||||||
|
- name: Test combined options
|
||||||
|
set_fact:
|
||||||
|
combined_table: "{{ test_data | community.general.to_prettytable(
|
||||||
|
column_order=['role', 'name', 'age'],
|
||||||
|
header_names=['Position', 'Full Name', 'Years'],
|
||||||
|
column_alignments={'name': 'center', 'age': 'right', 'role': 'left'}) }}"
|
||||||
|
|
||||||
|
- name: Verify combined table output
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- combined_table | trim is defined
|
||||||
|
- combined_table | trim is not none
|
||||||
|
- "combined_table | trim is search('Position')"
|
||||||
|
- "combined_table | trim is search('Full Name')"
|
||||||
|
- "combined_table | trim is search('Years')"
|
||||||
|
|
||||||
|
# Test empty data
|
||||||
|
- name: Test empty data list
|
||||||
|
set_fact:
|
||||||
|
empty_table: "{{ [] | community.general.to_prettytable }}"
|
||||||
|
|
||||||
|
- name: Verify empty table output
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- empty_table | trim == "++\n++" | trim
|
||||||
|
|
||||||
|
# Test error conditions
|
||||||
|
- name: Test non-list input (expect failure)
|
||||||
|
block:
|
||||||
|
- set_fact:
|
||||||
|
invalid_table: "{{ 'not_a_list' | community.general.to_prettytable }}"
|
||||||
|
register: failure_result
|
||||||
|
ignore_errors: true
|
||||||
|
- name: Verify error message for non-list input
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- failure_result is failed
|
||||||
|
- "'Expected a list of dictionaries, got a string' in failure_result.msg"
|
||||||
|
|
||||||
|
- name: Test list with non-dictionary items (expect failure)
|
||||||
|
block:
|
||||||
|
- set_fact:
|
||||||
|
invalid_table: "{{ ['not_a_dict', 'also_not_a_dict'] | community.general.to_prettytable }}"
|
||||||
|
register: failure_result
|
||||||
|
ignore_errors: true
|
||||||
|
- name: Verify error message for non-dictionary items
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- failure_result is failed
|
||||||
|
- "'All items in the list must be dictionaries' in failure_result.msg"
|
Loading…
Add table
Reference in a new issue