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