all: children: zuul_unreachable: hosts: {} hosts: controller: ansible_connection: ssh ansible_host: 3.143.217.79 ansible_port: 22 ansible_python_interpreter: auto ansible_user: zuul-worker nodepool: az: us-east-2b cloud: AWS external_id: i-07b3b3bdf74babfd8 host_id: null interface_ip: 3.143.217.79 label: ansible-fedora-37-1vcpu private_ipv4: 172.16.85.179 private_ipv6: null provider: ansible-us-east-2 public_ipv4: 3.143.217.79 public_ipv6: null region: us-east-2 slot: null zuul_use_fetch_output: true vars: zuul: _inheritance_path: - '' - '' - '' - '' ansible_version: '8' artifacts: - branch: main change: '2344' job: build-ansible-collection metadata: type: zuul_manifest name: Zuul Manifest patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 project: ansible-collections/community.aws url: https://997c70408e6c47ac51dd-14b219e290455602ac0062c23a62d17d.ssl.cf1.rackcdn.com/ansible/839089371372424bb5b17cc9af405e12/zuul-manifest.json - branch: main change: '2344' job: build-ansible-collection metadata: type: ansible_collection version: 11.0.0 name: community.aws patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 project: ansible-collections/community.aws url: https://997c70408e6c47ac51dd-14b219e290455602ac0062c23a62d17d.ssl.cf1.rackcdn.com/ansible/839089371372424bb5b17cc9af405e12/artifacts/community-aws-11.0.0.tar.gz - branch: main change: '2344' job: build-ansible-collection metadata: type: ansible_collection version: 11.0.0 name: amazon.aws patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 project: ansible-collections/community.aws url: https://997c70408e6c47ac51dd-14b219e290455602ac0062c23a62d17d.ssl.cf1.rackcdn.com/ansible/839089371372424bb5b17cc9af405e12/artifacts/amazon-aws-11.0.0.tar.gz attempts: 1 branch: main build: 135f6ffd46974dcd858d532519eb74ef build_refs: - branch: main change: '2344' change_message: "feat: add DynamoDB query lookup plugin for AWS tables\n\n# Add DynamoDB Query Lookup Plugin\r\n\r\n## SUMMARY\r\n\r\nThis PR adds a new lookup plugin `dynamodb_query` that enables querying AWS DynamoDB tables directly from Ansible playbooks. The plugin provides a convenient way to retrieve configuration data, application settings, and other dynamic values stored in DynamoDB tables as part of infrastructure-as-code workflows.\r\n\r\nThe plugin supports both Query (efficient, partition key-based) and Scan (full table scan) operations, with comprehensive support for DynamoDB features including secondary indexes, filter expressions, projection expressions, and automatic pagination.\r\n\r\n### Key Capabilities\r\n\r\n- Query items by partition key with optional sort key conditions\r\n- Support for all DynamoDB sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Query Global Secondary Indexes (GSI) and Local Secondary Indexes (LSI)\r\n- Filter expressions with expression attribute names and values\r\n- Projection expressions to retrieve specific attributes\r\n- Automatic pagination with configurable limits\r\n- Full support for all DynamoDB data types including Sets (SS, NS, BS)\r\n\r\n### Use Cases\r\n\r\n- Retrieve application configuration from DynamoDB tables\r\n- Query deployment metadata and environment-specific settings\r\n- Fetch dynamic inventory data stored in DynamoDB\r\n- Access feature flags and runtime configuration\r\n- Retrieve secrets metadata (not the secrets themselves)\r\n\r\n## ISSUE TYPE\r\n\r\n- New Plugin Pull Request\r\n\r\n## COMPONENT NAME\r\n\r\n`dynamodb_query` (lookup plugin)\r\n\r\n## ADDITIONAL INFORMATION\r\n\r\n### Implementation Details\r\n\r\nThe plugin is built on top of `AWSLookupBase` and follows Ansible AWS collection best practices:\r\n\r\n- Uses `boto3` with automatic retry decorators for resilience\r\n- Proper error handling with informative error messages\r\n- Supports all standard AWS authentication methods (credentials, profiles, IAM roles)\r\n- Handles DynamoDB pagination automatically\r\n- Deserializes all DynamoDB data types to native Python types\r\n\r\n### Testing\r\n\r\nComprehensive test coverage has been implemented:\r\n\r\n#### Unit Tests (28 tests, ~90% code coverage)\r\n\r\n- **Basic query operations** - partition key, sort key, operators\r\n- **All sort key operators** - EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN (including validation)\r\n- **Filter expressions** - with expression attribute names/values\r\n- **Projection expressions** - string and list formats\r\n- **Scan operations** - full table scans\r\n- **Pagination handling** - automatic pagination and with limits\r\n- **Error handling** - ResourceNotFoundException, ValidationException, missing parameters\r\n- **Data type deserialization** - String, Number, Boolean, List, Map, NULL, Binary, Sets (SS, NS, BS)\r\n- **Secondary indexes** - GSI queries\r\n- **Advanced parameters** - limit, descending order, pagination with limit\r\n\r\n#### Integration Tests (34 tests)\r\n\r\n- All sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Global Secondary Index (GSI) and Local Secondary Index (LSI)\r\n- Filter expressions with built-in functions (`attribute_exists`, `size`, `contains`)\r\n- Projection expressions (string and list formats)\r\n- Expression attribute names and values\r\n- Pagination (automatic and with limit)\r\n- All DynamoDB data types including Sets (SS, NS, BS)\r\n- Empty results and error handling\r\n- Scan operations\r\n- Descending sort order\r\n\r\n### Documentation\r\n\r\nComplete documentation included:\r\n\r\n- Module documentation with parameter descriptions\r\n- 10+ usage examples covering common scenarios\r\n- Return value documentation\r\n\r\n### Example Usage\r\n\r\n#### Simple Query by Partition Key\r\n\r\n```yaml\r\n- name: Get user data\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n partition_key='user_id',\r\n \ partition_value='12345') }}\"\r\n```\r\n\r\n#### Query with Sort Key and Filter\r\n\r\n```yaml\r\n- name: Get recent active orders\r\n \ ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n sort_key='order_date',\r\n \ sort_value='2024-01-01',\r\n sort_operator='GE',\r\n \ filter_expression='#status = :active',\r\n expression_attribute_names={'#status': 'status'},\r\n expression_attribute_values={':active': 'active'}) }}\"\r\n```\r\n\r\n#### Query Using Global Secondary Index\r\n\r\n```yaml\r\n- name: Find user by email\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n index_name='email-index',\r\n \ partition_key='email',\r\n partition_value='[email protected]') }}\"\r\n```\r\n\r\n#### Query with Projection Expression\r\n\r\n```yaml\r\n- name: Get specific user attributes\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Users',\r\n \ partition_key='user_id',\r\n partition_value='12345',\r\n \ projection_expression=['user_id', 'name', 'email']) }}\"\r\n```\r\n\r\n#### Query with Limit and Descending Order\r\n\r\n```yaml\r\n- name: Get last 10 orders\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n limit=10,\r\n \ scan_index_forward=false) }}\"\r\n```\r\n\r\n#### Scan Operation\r\n\r\n```yaml\r\n- name: Scan small table with filter\r\n ansible.builtin.debug:\r\n \ msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Config',\r\n \ operation='scan',\r\n filter_expression='#env = :prod',\r\n expression_attribute_names={'#env': 'environment'},\r\n \ expression_attribute_values={':prod': 'production'},\r\n \ limit=100) }}\"\r\n```\r\n\r\n### Tested With\r\n\r\n- Python 3.11, 3.12, 3.13\r\n- ansible-core 2.14+\r\n- boto3 1.34.0+\r\n- botocore 1.34.0+\r\n\r\nAll sanity tests and integration tests pass successfully.\r\n\r\n### Related Documentation\r\n\r\n- [DynamoDB Query API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Query.html)\r\n- [DynamoDB Scan API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Scan.html)\r\n- [DynamoDB Data Types](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.NamingRulesDataTypes.html)\r\n- [Ansible Lookup Plugins](https://docs.ansible.com/ansible/latest/plugins/lookup.html)\r\n" change_url: https://github.com/ansible-collections/community.aws/pull/2344 commit_id: 31db89a1986e6df597d98574ba01018ed42b9a15 patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 project: canonical_hostname: github.com canonical_name: github.com/ansible-collections/community.aws name: ansible-collections/community.aws short_name: community.aws src_dir: src/github.com/ansible-collections/community.aws topic: null buildset: 68a34ca4caea4fc488c03f1e7b4c9d41 buildset_refs: - branch: main change: '2344' change_message: "feat: add DynamoDB query lookup plugin for AWS tables\n\n# Add DynamoDB Query Lookup Plugin\r\n\r\n## SUMMARY\r\n\r\nThis PR adds a new lookup plugin `dynamodb_query` that enables querying AWS DynamoDB tables directly from Ansible playbooks. The plugin provides a convenient way to retrieve configuration data, application settings, and other dynamic values stored in DynamoDB tables as part of infrastructure-as-code workflows.\r\n\r\nThe plugin supports both Query (efficient, partition key-based) and Scan (full table scan) operations, with comprehensive support for DynamoDB features including secondary indexes, filter expressions, projection expressions, and automatic pagination.\r\n\r\n### Key Capabilities\r\n\r\n- Query items by partition key with optional sort key conditions\r\n- Support for all DynamoDB sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Query Global Secondary Indexes (GSI) and Local Secondary Indexes (LSI)\r\n- Filter expressions with expression attribute names and values\r\n- Projection expressions to retrieve specific attributes\r\n- Automatic pagination with configurable limits\r\n- Full support for all DynamoDB data types including Sets (SS, NS, BS)\r\n\r\n### Use Cases\r\n\r\n- Retrieve application configuration from DynamoDB tables\r\n- Query deployment metadata and environment-specific settings\r\n- Fetch dynamic inventory data stored in DynamoDB\r\n- Access feature flags and runtime configuration\r\n- Retrieve secrets metadata (not the secrets themselves)\r\n\r\n## ISSUE TYPE\r\n\r\n- New Plugin Pull Request\r\n\r\n## COMPONENT NAME\r\n\r\n`dynamodb_query` (lookup plugin)\r\n\r\n## ADDITIONAL INFORMATION\r\n\r\n### Implementation Details\r\n\r\nThe plugin is built on top of `AWSLookupBase` and follows Ansible AWS collection best practices:\r\n\r\n- Uses `boto3` with automatic retry decorators for resilience\r\n- Proper error handling with informative error messages\r\n- Supports all standard AWS authentication methods (credentials, profiles, IAM roles)\r\n- Handles DynamoDB pagination automatically\r\n- Deserializes all DynamoDB data types to native Python types\r\n\r\n### Testing\r\n\r\nComprehensive test coverage has been implemented:\r\n\r\n#### Unit Tests (28 tests, ~90% code coverage)\r\n\r\n- **Basic query operations** - partition key, sort key, operators\r\n- **All sort key operators** - EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN (including validation)\r\n- **Filter expressions** - with expression attribute names/values\r\n- **Projection expressions** - string and list formats\r\n- **Scan operations** - full table scans\r\n- **Pagination handling** - automatic pagination and with limits\r\n- **Error handling** - ResourceNotFoundException, ValidationException, missing parameters\r\n- **Data type deserialization** - String, Number, Boolean, List, Map, NULL, Binary, Sets (SS, NS, BS)\r\n- **Secondary indexes** - GSI queries\r\n- **Advanced parameters** - limit, descending order, pagination with limit\r\n\r\n#### Integration Tests (34 tests)\r\n\r\n- All sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Global Secondary Index (GSI) and Local Secondary Index (LSI)\r\n- Filter expressions with built-in functions (`attribute_exists`, `size`, `contains`)\r\n- Projection expressions (string and list formats)\r\n- Expression attribute names and values\r\n- Pagination (automatic and with limit)\r\n- All DynamoDB data types including Sets (SS, NS, BS)\r\n- Empty results and error handling\r\n- Scan operations\r\n- Descending sort order\r\n\r\n### Documentation\r\n\r\nComplete documentation included:\r\n\r\n- Module documentation with parameter descriptions\r\n- 10+ usage examples covering common scenarios\r\n- Return value documentation\r\n\r\n### Example Usage\r\n\r\n#### Simple Query by Partition Key\r\n\r\n```yaml\r\n- name: Get user data\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n partition_key='user_id',\r\n \ partition_value='12345') }}\"\r\n```\r\n\r\n#### Query with Sort Key and Filter\r\n\r\n```yaml\r\n- name: Get recent active orders\r\n \ ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n sort_key='order_date',\r\n \ sort_value='2024-01-01',\r\n sort_operator='GE',\r\n \ filter_expression='#status = :active',\r\n expression_attribute_names={'#status': 'status'},\r\n expression_attribute_values={':active': 'active'}) }}\"\r\n```\r\n\r\n#### Query Using Global Secondary Index\r\n\r\n```yaml\r\n- name: Find user by email\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n index_name='email-index',\r\n \ partition_key='email',\r\n partition_value='[email protected]') }}\"\r\n```\r\n\r\n#### Query with Projection Expression\r\n\r\n```yaml\r\n- name: Get specific user attributes\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Users',\r\n \ partition_key='user_id',\r\n partition_value='12345',\r\n \ projection_expression=['user_id', 'name', 'email']) }}\"\r\n```\r\n\r\n#### Query with Limit and Descending Order\r\n\r\n```yaml\r\n- name: Get last 10 orders\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n limit=10,\r\n \ scan_index_forward=false) }}\"\r\n```\r\n\r\n#### Scan Operation\r\n\r\n```yaml\r\n- name: Scan small table with filter\r\n ansible.builtin.debug:\r\n \ msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Config',\r\n \ operation='scan',\r\n filter_expression='#env = :prod',\r\n expression_attribute_names={'#env': 'environment'},\r\n \ expression_attribute_values={':prod': 'production'},\r\n \ limit=100) }}\"\r\n```\r\n\r\n### Tested With\r\n\r\n- Python 3.11, 3.12, 3.13\r\n- ansible-core 2.14+\r\n- boto3 1.34.0+\r\n- botocore 1.34.0+\r\n\r\nAll sanity tests and integration tests pass successfully.\r\n\r\n### Related Documentation\r\n\r\n- [DynamoDB Query API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Query.html)\r\n- [DynamoDB Scan API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Scan.html)\r\n- [DynamoDB Data Types](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.NamingRulesDataTypes.html)\r\n- [Ansible Lookup Plugins](https://docs.ansible.com/ansible/latest/plugins/lookup.html)\r\n" change_url: https://github.com/ansible-collections/community.aws/pull/2344 commit_id: 31db89a1986e6df597d98574ba01018ed42b9a15 patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 project: canonical_hostname: github.com canonical_name: github.com/ansible-collections/community.aws name: ansible-collections/community.aws short_name: community.aws src_dir: src/github.com/ansible-collections/community.aws topic: null change: '2344' change_message: "feat: add DynamoDB query lookup plugin for AWS tables\n\n# Add DynamoDB Query Lookup Plugin\r\n\r\n## SUMMARY\r\n\r\nThis PR adds a new lookup plugin `dynamodb_query` that enables querying AWS DynamoDB tables directly from Ansible playbooks. The plugin provides a convenient way to retrieve configuration data, application settings, and other dynamic values stored in DynamoDB tables as part of infrastructure-as-code workflows.\r\n\r\nThe plugin supports both Query (efficient, partition key-based) and Scan (full table scan) operations, with comprehensive support for DynamoDB features including secondary indexes, filter expressions, projection expressions, and automatic pagination.\r\n\r\n### Key Capabilities\r\n\r\n- Query items by partition key with optional sort key conditions\r\n- Support for all DynamoDB sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Query Global Secondary Indexes (GSI) and Local Secondary Indexes (LSI)\r\n- Filter expressions with expression attribute names and values\r\n- Projection expressions to retrieve specific attributes\r\n- Automatic pagination with configurable limits\r\n- Full support for all DynamoDB data types including Sets (SS, NS, BS)\r\n\r\n### Use Cases\r\n\r\n- Retrieve application configuration from DynamoDB tables\r\n- Query deployment metadata and environment-specific settings\r\n- Fetch dynamic inventory data stored in DynamoDB\r\n- Access feature flags and runtime configuration\r\n- Retrieve secrets metadata (not the secrets themselves)\r\n\r\n## ISSUE TYPE\r\n\r\n- New Plugin Pull Request\r\n\r\n## COMPONENT NAME\r\n\r\n`dynamodb_query` (lookup plugin)\r\n\r\n## ADDITIONAL INFORMATION\r\n\r\n### Implementation Details\r\n\r\nThe plugin is built on top of `AWSLookupBase` and follows Ansible AWS collection best practices:\r\n\r\n- Uses `boto3` with automatic retry decorators for resilience\r\n- Proper error handling with informative error messages\r\n- Supports all standard AWS authentication methods (credentials, profiles, IAM roles)\r\n- Handles DynamoDB pagination automatically\r\n- Deserializes all DynamoDB data types to native Python types\r\n\r\n### Testing\r\n\r\nComprehensive test coverage has been implemented:\r\n\r\n#### Unit Tests (28 tests, ~90% code coverage)\r\n\r\n- **Basic query operations** - partition key, sort key, operators\r\n- **All sort key operators** - EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN (including validation)\r\n- **Filter expressions** - with expression attribute names/values\r\n- **Projection expressions** - string and list formats\r\n- **Scan operations** - full table scans\r\n- **Pagination handling** - automatic pagination and with limits\r\n- **Error handling** - ResourceNotFoundException, ValidationException, missing parameters\r\n- **Data type deserialization** - String, Number, Boolean, List, Map, NULL, Binary, Sets (SS, NS, BS)\r\n- **Secondary indexes** - GSI queries\r\n- **Advanced parameters** - limit, descending order, pagination with limit\r\n\r\n#### Integration Tests (34 tests)\r\n\r\n- All sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Global Secondary Index (GSI) and Local Secondary Index (LSI)\r\n- Filter expressions with built-in functions (`attribute_exists`, `size`, `contains`)\r\n- Projection expressions (string and list formats)\r\n- Expression attribute names and values\r\n- Pagination (automatic and with limit)\r\n- All DynamoDB data types including Sets (SS, NS, BS)\r\n- Empty results and error handling\r\n- Scan operations\r\n- Descending sort order\r\n\r\n### Documentation\r\n\r\nComplete documentation included:\r\n\r\n- Module documentation with parameter descriptions\r\n- 10+ usage examples covering common scenarios\r\n- Return value documentation\r\n\r\n### Example Usage\r\n\r\n#### Simple Query by Partition Key\r\n\r\n```yaml\r\n- name: Get user data\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n partition_key='user_id',\r\n \ partition_value='12345') }}\"\r\n```\r\n\r\n#### Query with Sort Key and Filter\r\n\r\n```yaml\r\n- name: Get recent active orders\r\n \ ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n sort_key='order_date',\r\n \ sort_value='2024-01-01',\r\n sort_operator='GE',\r\n \ filter_expression='#status = :active',\r\n expression_attribute_names={'#status': 'status'},\r\n expression_attribute_values={':active': 'active'}) }}\"\r\n```\r\n\r\n#### Query Using Global Secondary Index\r\n\r\n```yaml\r\n- name: Find user by email\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n index_name='email-index',\r\n \ partition_key='email',\r\n partition_value='[email protected]') }}\"\r\n```\r\n\r\n#### Query with Projection Expression\r\n\r\n```yaml\r\n- name: Get specific user attributes\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Users',\r\n \ partition_key='user_id',\r\n partition_value='12345',\r\n \ projection_expression=['user_id', 'name', 'email']) }}\"\r\n```\r\n\r\n#### Query with Limit and Descending Order\r\n\r\n```yaml\r\n- name: Get last 10 orders\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n limit=10,\r\n \ scan_index_forward=false) }}\"\r\n```\r\n\r\n#### Scan Operation\r\n\r\n```yaml\r\n- name: Scan small table with filter\r\n ansible.builtin.debug:\r\n \ msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Config',\r\n \ operation='scan',\r\n filter_expression='#env = :prod',\r\n expression_attribute_names={'#env': 'environment'},\r\n \ expression_attribute_values={':prod': 'production'},\r\n \ limit=100) }}\"\r\n```\r\n\r\n### Tested With\r\n\r\n- Python 3.11, 3.12, 3.13\r\n- ansible-core 2.14+\r\n- boto3 1.34.0+\r\n- botocore 1.34.0+\r\n\r\nAll sanity tests and integration tests pass successfully.\r\n\r\n### Related Documentation\r\n\r\n- [DynamoDB Query API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Query.html)\r\n- [DynamoDB Scan API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Scan.html)\r\n- [DynamoDB Data Types](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.NamingRulesDataTypes.html)\r\n- [Ansible Lookup Plugins](https://docs.ansible.com/ansible/latest/plugins/lookup.html)\r\n" change_url: https://github.com/ansible-collections/community.aws/pull/2344 child_jobs: [] commit_id: 31db89a1986e6df597d98574ba01018ed42b9a15 event_id: 76cfe9f0-a605-11f0-8a4f-89c19f6ef3dc executor: hostname: ze04.softwarefactory-project.io inventory_file: /var/lib/zuul/builds/135f6ffd46974dcd858d532519eb74ef/ansible/inventory.yaml log_root: /var/lib/zuul/builds/135f6ffd46974dcd858d532519eb74ef/work/logs result_data_file: /var/lib/zuul/builds/135f6ffd46974dcd858d532519eb74ef/work/results.json src_root: /var/lib/zuul/builds/135f6ffd46974dcd858d532519eb74ef/work/src work_root: /var/lib/zuul/builds/135f6ffd46974dcd858d532519eb74ef/work items: - branch: main change: '2344' change_message: "feat: add DynamoDB query lookup plugin for AWS tables\n\n# Add DynamoDB Query Lookup Plugin\r\n\r\n## SUMMARY\r\n\r\nThis PR adds a new lookup plugin `dynamodb_query` that enables querying AWS DynamoDB tables directly from Ansible playbooks. The plugin provides a convenient way to retrieve configuration data, application settings, and other dynamic values stored in DynamoDB tables as part of infrastructure-as-code workflows.\r\n\r\nThe plugin supports both Query (efficient, partition key-based) and Scan (full table scan) operations, with comprehensive support for DynamoDB features including secondary indexes, filter expressions, projection expressions, and automatic pagination.\r\n\r\n### Key Capabilities\r\n\r\n- Query items by partition key with optional sort key conditions\r\n- Support for all DynamoDB sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Query Global Secondary Indexes (GSI) and Local Secondary Indexes (LSI)\r\n- Filter expressions with expression attribute names and values\r\n- Projection expressions to retrieve specific attributes\r\n- Automatic pagination with configurable limits\r\n- Full support for all DynamoDB data types including Sets (SS, NS, BS)\r\n\r\n### Use Cases\r\n\r\n- Retrieve application configuration from DynamoDB tables\r\n- Query deployment metadata and environment-specific settings\r\n- Fetch dynamic inventory data stored in DynamoDB\r\n- Access feature flags and runtime configuration\r\n- Retrieve secrets metadata (not the secrets themselves)\r\n\r\n## ISSUE TYPE\r\n\r\n- New Plugin Pull Request\r\n\r\n## COMPONENT NAME\r\n\r\n`dynamodb_query` (lookup plugin)\r\n\r\n## ADDITIONAL INFORMATION\r\n\r\n### Implementation Details\r\n\r\nThe plugin is built on top of `AWSLookupBase` and follows Ansible AWS collection best practices:\r\n\r\n- Uses `boto3` with automatic retry decorators for resilience\r\n- Proper error handling with informative error messages\r\n- Supports all standard AWS authentication methods (credentials, profiles, IAM roles)\r\n- Handles DynamoDB pagination automatically\r\n- Deserializes all DynamoDB data types to native Python types\r\n\r\n### Testing\r\n\r\nComprehensive test coverage has been implemented:\r\n\r\n#### Unit Tests (28 tests, ~90% code coverage)\r\n\r\n- **Basic query operations** - partition key, sort key, operators\r\n- **All sort key operators** - EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN (including validation)\r\n- **Filter expressions** - with expression attribute names/values\r\n- **Projection expressions** - string and list formats\r\n- **Scan operations** - full table scans\r\n- **Pagination handling** - automatic pagination and with limits\r\n- **Error handling** - ResourceNotFoundException, ValidationException, missing parameters\r\n- **Data type deserialization** - String, Number, Boolean, List, Map, NULL, Binary, Sets (SS, NS, BS)\r\n- **Secondary indexes** - GSI queries\r\n- **Advanced parameters** - limit, descending order, pagination with limit\r\n\r\n#### Integration Tests (34 tests)\r\n\r\n- All sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)\r\n- Global Secondary Index (GSI) and Local Secondary Index (LSI)\r\n- Filter expressions with built-in functions (`attribute_exists`, `size`, `contains`)\r\n- Projection expressions (string and list formats)\r\n- Expression attribute names and values\r\n- Pagination (automatic and with limit)\r\n- All DynamoDB data types including Sets (SS, NS, BS)\r\n- Empty results and error handling\r\n- Scan operations\r\n- Descending sort order\r\n\r\n### Documentation\r\n\r\nComplete documentation included:\r\n\r\n- Module documentation with parameter descriptions\r\n- 10+ usage examples covering common scenarios\r\n- Return value documentation\r\n\r\n### Example Usage\r\n\r\n#### Simple Query by Partition Key\r\n\r\n```yaml\r\n- name: Get user data\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n partition_key='user_id',\r\n \ partition_value='12345') }}\"\r\n```\r\n\r\n#### Query with Sort Key and Filter\r\n\r\n```yaml\r\n- name: Get recent active orders\r\n \ ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n sort_key='order_date',\r\n \ sort_value='2024-01-01',\r\n sort_operator='GE',\r\n \ filter_expression='#status = :active',\r\n expression_attribute_names={'#status': 'status'},\r\n expression_attribute_values={':active': 'active'}) }}\"\r\n```\r\n\r\n#### Query Using Global Secondary Index\r\n\r\n```yaml\r\n- name: Find user by email\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Users',\r\n index_name='email-index',\r\n \ partition_key='email',\r\n partition_value='[email protected]') }}\"\r\n```\r\n\r\n#### Query with Projection Expression\r\n\r\n```yaml\r\n- name: Get specific user attributes\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Users',\r\n \ partition_key='user_id',\r\n partition_value='12345',\r\n \ projection_expression=['user_id', 'name', 'email']) }}\"\r\n```\r\n\r\n#### Query with Limit and Descending Order\r\n\r\n```yaml\r\n- name: Get last 10 orders\r\n ansible.builtin.debug:\r\n msg: \"{{ lookup('community.aws.dynamodb_query',\r\n \ table_name='Orders',\r\n partition_key='user_id',\r\n \ partition_value='12345',\r\n limit=10,\r\n \ scan_index_forward=false) }}\"\r\n```\r\n\r\n#### Scan Operation\r\n\r\n```yaml\r\n- name: Scan small table with filter\r\n ansible.builtin.debug:\r\n \ msg: \"{{ lookup('community.aws.dynamodb_query',\r\n table_name='Config',\r\n \ operation='scan',\r\n filter_expression='#env = :prod',\r\n expression_attribute_names={'#env': 'environment'},\r\n \ expression_attribute_values={':prod': 'production'},\r\n \ limit=100) }}\"\r\n```\r\n\r\n### Tested With\r\n\r\n- Python 3.11, 3.12, 3.13\r\n- ansible-core 2.14+\r\n- boto3 1.34.0+\r\n- botocore 1.34.0+\r\n\r\nAll sanity tests and integration tests pass successfully.\r\n\r\n### Related Documentation\r\n\r\n- [DynamoDB Query API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Query.html)\r\n- [DynamoDB Scan API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Scan.html)\r\n- [DynamoDB Data Types](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.NamingRulesDataTypes.html)\r\n- [Ansible Lookup Plugins](https://docs.ansible.com/ansible/latest/plugins/lookup.html)\r\n" change_url: https://github.com/ansible-collections/community.aws/pull/2344 commit_id: 31db89a1986e6df597d98574ba01018ed42b9a15 patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 project: canonical_hostname: github.com canonical_name: github.com/ansible-collections/community.aws name: ansible-collections/community.aws short_name: community.aws src_dir: src/github.com/ansible-collections/community.aws topic: null job: ansible-galaxy-importer jobtags: [] max_attempts: 3 message: feat: add DynamoDB query lookup plugin for AWS tables

# Add DynamoDB Query Lookup Plugin

## SUMMARY

This PR adds a new lookup plugin `dynamodb_query` that enables querying AWS DynamoDB tables directly from Ansible playbooks. The plugin provides a convenient way to retrieve configuration data, application settings, and other dynamic values stored in DynamoDB tables as part of infrastructure-as-code workflows.

The plugin supports both Query (efficient, partition key-based) and Scan (full table scan) operations, with comprehensive support for DynamoDB features including secondary indexes, filter expressions, projection expressions, and automatic pagination.

### Key Capabilities

- Query items by partition key with optional sort key conditions
- Support for all DynamoDB sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)
- Query Global Secondary Indexes (GSI) and Local Secondary Indexes (LSI)
- Filter expressions with expression attribute names and values
- Projection expressions to retrieve specific attributes
- Automatic pagination with configurable limits
- Full support for all DynamoDB data types including Sets (SS, NS, BS)

### Use Cases

- Retrieve application configuration from DynamoDB tables
- Query deployment metadata and environment-specific settings
- Fetch dynamic inventory data stored in DynamoDB
- Access feature flags and runtime configuration
- Retrieve secrets metadata (not the secrets themselves)

## ISSUE TYPE

- New Plugin Pull Request

## COMPONENT NAME

`dynamodb_query` (lookup plugin)

## ADDITIONAL INFORMATION

### Implementation Details

The plugin is built on top of `AWSLookupBase` and follows Ansible AWS collection best practices:

- Uses `boto3` with automatic retry decorators for resilience
- Proper error handling with informative error messages
- Supports all standard AWS authentication methods (credentials, profiles, IAM roles)
- Handles DynamoDB pagination automatically
- Deserializes all DynamoDB data types to native Python types

### Testing

Comprehensive test coverage has been implemented:

#### Unit Tests (28 tests, ~90% code coverage)

- **Basic query operations** - partition key, sort key, operators
- **All sort key operators** - EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN (including validation)
- **Filter expressions** - with expression attribute names/values
- **Projection expressions** - string and list formats
- **Scan operations** - full table scans
- **Pagination handling** - automatic pagination and with limits
- **Error handling** - ResourceNotFoundException, ValidationException, missing parameters
- **Data type deserialization** - String, Number, Boolean, List, Map, NULL, Binary, Sets (SS, NS, BS)
- **Secondary indexes** - GSI queries
- **Advanced parameters** - limit, descending order, pagination with limit

#### Integration Tests (34 tests)

- All sort key operators (EQ, LT, LE, GT, GE, BEGINS_WITH, BETWEEN)
- Global Secondary Index (GSI) and Local Secondary Index (LSI)
- Filter expressions with built-in functions (`attribute_exists`, `size`, `contains`)
- Projection expressions (string and list formats)
- Expression attribute names and values
- Pagination (automatic and with limit)
- All DynamoDB data types including Sets (SS, NS, BS)
- Empty results and error handling
- Scan operations
- Descending sort order

### Documentation

Complete documentation included:

- Module documentation with parameter descriptions
- 10+ usage examples covering common scenarios
- Return value documentation

### Example Usage

#### Simple Query by Partition Key

```yaml
- name: Get user data
  ansible.builtin.debug:
    msg: "{{ lookup('community.aws.dynamodb_query',
                     table_name='Users',
                     partition_key='user_id',
                     partition_value='12345') }}"
```

#### Query with Sort Key and Filter

```yaml
- name: Get recent active orders
  ansible.builtin.debug:
    msg: "{{ lookup('community.aws.dynamodb_query',
                     table_name='Orders',
                     partition_key='user_id',
                     partition_value='12345',
                     sort_key='order_date',
                     sort_value='2024-01-01',
                     sort_operator='GE',
                     filter_expression='#status = :active',
                     expression_attribute_names={'#status': 'status'},
                     expression_attribute_values={':active': 'active'}) }}"
```

#### Query Using Global Secondary Index

```yaml
- name: Find user by email
  ansible.builtin.debug:
    msg: "{{ lookup('community.aws.dynamodb_query',
                     table_name='Users',
                     index_name='email-index',
                     partition_key='email',
                     partition_value='[email protected]') }}"
```

#### Query with Projection Expression

```yaml
- name: Get specific user attributes
  ansible.builtin.debug:
    msg: "{{ lookup('community.aws.dynamodb_query',
                     table_name='Users',
                     partition_key='user_id',
                     partition_value='12345',
                     projection_expression=['user_id', 'name', 'email']) }}"
```

#### Query with Limit and Descending Order

```yaml
- name: Get last 10 orders
  ansible.builtin.debug:
    msg: "{{ lookup('community.aws.dynamodb_query',
                     table_name='Orders',
                     partition_key='user_id',
                     partition_value='12345',
                     limit=10,
                     scan_index_forward=false) }}"
```

#### Scan Operation

```yaml
- name: Scan small table with filter
  ansible.builtin.debug:
    msg: "{{ lookup('community.aws.dynamodb_query',
                     table_name='Config',
                     operation='scan',
                     filter_expression='#env = :prod',
                     expression_attribute_names={'#env': 'environment'},
                     expression_attribute_values={':prod': 'production'},
                     limit=100) }}"
```

### Tested With

- Python 3.11, 3.12, 3.13
- ansible-core 2.14+
- boto3 1.34.0+
- botocore 1.34.0+

All sanity tests and integration tests pass successfully.

### Related Documentation

- [DynamoDB Query API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Query.html)
- [DynamoDB Scan API](https://docs.aws.amazon.com/amazondynamodb/latest/APIReference/API_Scan.html)
- [DynamoDB Data Types](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.NamingRulesDataTypes.html)
- [Ansible Lookup Plugins](https://docs.ansible.com/ansible/latest/plugins/lookup.html)
 patchset: 31db89a1986e6df597d98574ba01018ed42b9a15 pipeline: third-party-check playbook_context: playbook_projects: trusted/project_0/github.com/ansible/zuul-config: canonical_name: github.com/ansible/zuul-config checkout: master commit: 9cec676fdc5b2a7fbf401767643a3c48545c082f trusted/project_1/opendev.org/zuul/zuul-jobs: canonical_name: opendev.org/zuul/zuul-jobs checkout: master commit: 3f62739c27168ebe05c65ba9b26a90fe6a6268df untrusted/project_0/github.com/ansible/ansible-zuul-jobs: canonical_name: github.com/ansible/ansible-zuul-jobs checkout: master commit: f22f22e05a919f76989979ce0a538febe2c4e73f untrusted/project_1/github.com/ansible/zuul-config: canonical_name: github.com/ansible/zuul-config checkout: master commit: 9cec676fdc5b2a7fbf401767643a3c48545c082f untrusted/project_2/opendev.org/zuul/zuul-jobs: canonical_name: opendev.org/zuul/zuul-jobs checkout: master commit: 3f62739c27168ebe05c65ba9b26a90fe6a6268df playbooks: - path: untrusted/project_0/github.com/ansible/ansible-zuul-jobs/playbooks/ansible-galaxy-importer/run.yaml roles: - checkout: master checkout_description: playbook branch link_name: ansible/playbook_0/role_0/zuul-jobs link_target: untrusted/project_0/github.com/ansible/ansible-zuul-jobs role_path: ansible/playbook_0/role_0/zuul-jobs/roles - checkout: master checkout_description: project default branch link_name: ansible/playbook_0/role_1/zuul-config link_target: untrusted/project_1/github.com/ansible/zuul-config role_path: ansible/playbook_0/role_1/zuul-config/roles - checkout: master checkout_description: project default branch link_name: ansible/playbook_0/role_2/zuul-jobs link_target: untrusted/project_2/opendev.org/zuul/zuul-jobs role_path: ansible/playbook_0/role_2/zuul-jobs/roles post_review: false project: canonical_hostname: github.com canonical_name: github.com/ansible-collections/community.aws name: ansible-collections/community.aws short_name: community.aws src_dir: src/github.com/ansible-collections/community.aws projects: github.com/ansible-collections/community.aws: canonical_hostname: github.com canonical_name: github.com/ansible-collections/community.aws checkout: main checkout_description: zuul branch commit: 5aa6a0b9586dca32b757beb4efbe14e6d4352562 name: ansible-collections/community.aws required: false short_name: community.aws src_dir: src/github.com/ansible-collections/community.aws github.com/ansible-network/releases: canonical_hostname: github.com canonical_name: github.com/ansible-network/releases checkout: master checkout_description: project default branch commit: 646b310655c531e4904be07f4ff8fc3a29addd09 name: ansible-network/releases required: true short_name: releases src_dir: src/github.com/ansible-network/releases ref: refs/pull/2344/head resources: {} tenant: ansible timeout: 1800 topic: null voting: false zuul_use_fetch_output: true