35 Commits

Author SHA1 Message Date
kewl
290b852836 Update galaxy.yml
bump version
2025-08-22 21:13:35 +01:00
kewl
31828edaf9 Update galaxy.yml
add log
2025-08-22 20:53:45 +01:00
kewl
610e14b8c2 Update galaxy.yml
fix path
2025-08-22 20:51:19 +01:00
kewl
b8da3380f5 Update galaxy.yml
fixed path
2025-08-22 20:49:26 +01:00
kewl
984cb9fde1 Update galaxy.yml
finds tarball
2025-08-22 20:47:01 +01:00
kewl
0d4d6aa3e7 Update galaxy.yml
fix publish
2025-08-22 20:44:24 +01:00
kewl
d2f3fb5802 Update galaxy.yml
fixed deployment and improve version check
2025-08-22 20:41:09 +01:00
kewl
3ec32e68a3 Update galaxy.yml
updated key name
2025-08-22 20:31:43 +01:00
kewl
37d4b45acd Update galaxy.yml
v bump
2025-08-22 20:20:50 +01:00
kewl
ebb5e9aed1 Merge pull request #84 from mfinelli/fixgithubactions
Fix github actions
2025-08-22 20:17:52 +01:00
Mario Finelli
2ef95d29d6 Remove setup-python action
Ansible (and ansible-core) is already installed in the runner images
which should be enough to interact with ansible galaxy without needing
to install any additional packages.
2025-08-21 11:25:50 +02:00
Mario Finelli
c58c59c63d Fix output syntax
A while ago Github deprecated and removed the ::set-output syntax in
favor of appending to the GITHUB_OUTPUT.
2025-08-21 11:18:40 +02:00
Mario Finelli
b2ed7c1dc6 Upgrade to latest actions versions
We need to use the latest versions because older nodejs runtimes have
been deprecated and removed some time ago.

As a bonus, the latest version of the setup-python action supports
caching pip dependencies out-of-the-box (setting the appropriate cache
option) so we can eliminate a few of the manual steps to setup
dependency caching.

Finally, I also bumped the python version to 3.13 which is what
currently ships in the Arch repositories, not that it really matters.
2025-08-21 11:16:20 +02:00
Mario Finelli
a2267b88a3 Switch to ubuntu-latest runner
The ubuntu-20.04 runner is no longer available. Switch to latest so that
we don't need to keep updating this as time passes, we don't depend on
anything for a specific ubuntu version.
2025-08-21 11:15:43 +02:00
kewl
768c15ad24 Update galaxy.yml 2025-08-18 17:46:06 +01:00
kewl
0182711634 Merge pull request #82 from ammarlakis/detailed-return
Add installed and updated packages to output
2025-08-18 17:28:57 +01:00
kewl
73c68bfd79 Merge pull request #83 from ammarlakis/absent-state
Add package absent state
2025-08-18 17:23:57 +01:00
Ammar Lakis
21f86cbb6e Add installed and updated packages to output 2025-08-17 17:09:12 +02:00
Ammar Lakis
c348e578cb Add package absent state 2025-08-15 22:39:38 +02:00
kewl fft
fe050018db galaxy version updated to 0.11.1 2023-07-08 19:37:19 +01:00
kewl fft
326836ddbe galaxy version updated 2023-07-08 19:36:22 +01:00
kewl fft
583c7597a9 minor optimizations 2023-07-08 19:20:32 +01:00
kewl fft
cbf843e1db Replaced the changed_iter = changed_iter or ... statement with the changed_iter |= ... compound assignment operator for brevity 2023-07-08 18:56:02 +01:00
kewl fft
2a076dc6a0 optimize check_packages lists construction 2023-07-08 18:51:35 +01:00
kewl
8d177a7550 Merge pull request #78 from jp1995/master
fix for issue #77
2023-07-08 18:40:26 +01:00
kewl
b469b5a641 Merge pull request #73 from LorenzoBettini/LorenzoBettini-patch-1
"mode: 0644" in the README example
2023-07-08 18:30:26 +01:00
jp1995
63d3056a82 fix for issue #77 2023-07-07 18:15:52 +03:00
Lorenzo Bettini
91c6d70184 "mode: 0644" in the README example
When using ansible-lint it would complain about "risky-file-permissions: File permissions unset or incorrect".
Making the `mode` explicit avoids the problem
2022-05-24 14:29:56 +02:00
kewl
9b25b11cea Merge pull request #66 from gotmax23/master
Remove unnecessary files from collection tarball
2021-11-08 11:03:48 +00:00
kewl
2c090a49a8 Merge pull request #67 from LorenzoBettini/patch-1
Use become: yes in the example for creating aur_builder
2021-11-08 08:18:34 +00:00
Lorenzo Bettini
07e5fe301f Use become: yes in the example for creating aur_builder
It is my understanding that for creating the `aur_builder` you have to become root; maybe that's something one does in the whole playbook, but if one wants to specify that only when it's needed, the example could be improved as suggested, so that it's self-contained and can be simply copied and pasted.
2021-11-07 13:08:56 +01:00
Maxwell G
49a077790e Remove unnecessary files from collection tarball 2021-11-04 16:55:37 -05:00
kewl fft
87a15853fa proper version bump 2021-11-04 21:44:50 +00:00
kewl fft
7285f5aaaa correct version tag 2021-11-04 21:34:04 +00:00
kewl fft
cda508cfc0 correct version tag 2021-11-04 21:30:30 +00:00
4 changed files with 170 additions and 72 deletions

View File

@@ -1,9 +1,8 @@
---
name: Build and Publish Collection to Ansible Galaxy name: Build and Publish Collection to Ansible Galaxy
'on': on:
release: release:
types: [ published ] types: [published]
workflow_dispatch: workflow_dispatch:
defaults: defaults:
@@ -12,44 +11,43 @@ defaults:
jobs: jobs:
release: release:
runs-on: ubuntu-20.04 runs-on: ubuntu-latest
steps: steps:
- name: Checkout git repo - name: Checkout repo
uses: actions/checkout@v2 uses: actions/checkout@v5
with: with:
path: collections/ansible_collections/kewlfft/aur path: collections/ansible_collections/kewlfft/aur
- name: Ensure that `version:` in `galaxy.yml` matches `GITHUB_REF` - name: Verify version in galaxy.yml matches release tag
id: version
run: | run: |
[ "$(cat galaxy.yml | grep version: | awk '{print $2}')" = $(awk -F '/' '{print substr($3, 2)}' <<< ${GITHUB_REF}) ] || exit 1 tag_version="${GITHUB_REF##*/}" # strip refs/tags/
echo ::set-output name=version::$(awk -F '/' '{print substr($3, 2)}' <<< ${GITHUB_REF}) file_version=$(awk '/^version:/ {print $2}' galaxy.yml)
- name: "Set up Python 3.9" if [ "$file_version" != "$tag_version" ]; then
uses: actions/setup-python@v2 echo "❌ galaxy.yml version ($file_version) does not match tag ($tag_version)"
exit 1
fi
- name: Build collection
run: |
path=$(ansible-galaxy collection build | awk '/Created collection/ {print $NF}')
echo "TARBALL=$path" >> $GITHUB_ENV
echo "📦 Built collection at $path"
- name: Upload built collection artifact
uses: actions/upload-artifact@v4
with: with:
python-version: "3.9" name: ansible-collection-tarball
path: ${{ env.TARBALL }}
- name: Generate cache keys - name: Debug tarball before publish
id: keys
run: | run: |
echo ::set-output name=pip_cache_dir::$(pip cache dir) echo "🔎 Checking tarball path..."
# This ensures that the cache is invalidated after a week echo "TARBALL=$TARBALL"
echo ::set-output name=date::$(date +%Y_%g) ls -l "$(dirname "$TARBALL")"
- name: Pip Cache - name: Publish collection
uses: actions/cache@v2
with:
path: ${{ steps.keys.outputs.pip_cache_dir }}
key: ansible-pip-${{ steps.keys.outputs.date }}
- name: Install pip packages
run: | run: |
pip3 install -U pip echo "📤 Publishing $TARBALL ..."
pip3 install wheel ansible-galaxy collection publish "$TARBALL" \
pip3 install ansible-core --api-key "${{ secrets.ANSIBLE_GALAXY_API_KEY }}"
- name: Deploy collection
run: |
ansible-galaxy collection build
ansible-galaxy collection publish --api-key ${{ secrets.GALAXY_API_KEY }} "./kewlfft-aur-${{ steps.version.outputs.version }}.tar.gz"

View File

@@ -67,7 +67,7 @@ The following helpers are supported and automatically selected, if present, in t
| Parameter | Choices/**Default** | Comments | | Parameter | Choices/**Default** | Comments |
| -------------- | ------------------------------------------------------------ | -------------------------------------------------------------------------------------------------------------------------------------------- | | -------------- | ------------------------------------------------------------ | -------------------------------------------------------------------------------------------------------------------------------------------- |
| name | | Name or list of names of the package(s) to install or upgrade. | | name | | Name or list of names of the package(s) to install or upgrade. |
| state | **present**, latest | Desired state of the package, 'present' skips operations if the package is already installed. | | state | **present**, latest, absent | Desired state of the package, 'present' skips operations if the package is already installed. |
| upgrade | yes, **no** | Whether or not to upgrade whole system. | | upgrade | yes, **no** | Whether or not to upgrade whole system. |
| update_cache | yes, **no** | Whether or not to refresh the packages cache | | update_cache | yes, **no** | Whether or not to refresh the packages cache |
| use | **auto**, yay, paru, pacaur, trizen, pikaur, aurman, makepkg | The tool to use, 'auto' uses the first known helper found and makepkg as a fallback. | | use | **auto**, yay, paru, pacaur, trizen, pikaur, aurman, makepkg | The tool to use, 'auto' uses the first known helper found and makepkg as a fallback. |
@@ -97,16 +97,19 @@ This user can be created in an Ansible task with the following actions:
```yaml ```yaml
- name: Create the `aur_builder` user - name: Create the `aur_builder` user
become: yes
ansible.builtin.user: ansible.builtin.user:
name: aur_builder name: aur_builder
create_home: yes create_home: yes
group: wheel group: wheel
- name: Allow the `aur_builder` user to run `sudo pacman` without a password - name: Allow the `aur_builder` user to run `sudo pacman` without a password
become: yes
ansible.builtin.lineinfile: ansible.builtin.lineinfile:
path: /etc/sudoers.d/11-install-aur_builder path: /etc/sudoers.d/11-install-aur_builder
line: 'aur_builder ALL=(ALL) NOPASSWD: /usr/bin/pacman' line: 'aur_builder ALL=(ALL) NOPASSWD: /usr/bin/pacman'
create: yes create: yes
mode: 0644
validate: 'visudo -cf %s' validate: 'visudo -cf %s'
``` ```
@@ -164,4 +167,14 @@ Use it in a task, as in the following examples:
state: present state: present
become: yes become: yes
become_user: aur_builder become_user: aur_builder
```
# Remove AUR packages (state=absent)
- name: Remove package_name_1 and package_name_2 using yay
kewlfft.aur.aur:
use: yay
state: absent
name:
- package_name_1
- package_name_2
become: yes
become_user: aur_builder

View File

@@ -8,7 +8,7 @@ namespace: kewlfft
name: aur name: aur
# The version of the collection. Must be compatible with semantic versioning # The version of the collection. Must be compatible with semantic versioning
version: 0.10.0 version: 0.12.2
# The path to the Markdown (.md) readme file. This path is relative to the root of the collection # The path to the Markdown (.md) readme file. This path is relative to the root of the collection
readme: README.md readme: README.md
@@ -65,4 +65,6 @@ issues: https://github.com/kewlfft/ansible-aur/issues
# artifact. A pattern is matched from the relative path of the file or directory of the collection directory. This # artifact. A pattern is matched from the relative path of the file or directory of the collection directory. This
# uses 'fnmatch' to match the files or directories. Some directories and files like 'galaxy.yml', '*.pyc', '*.retry', # uses 'fnmatch' to match the files or directories. Some directories and files like 'galaxy.yml', '*.pyc', '*.retry',
# and '.git' are always filtered # and '.git' are always filtered
build_ignore: [] build_ignore:
- .github
- .gitignore

View File

@@ -31,7 +31,7 @@ options:
description: description:
- Desired state of the package. - Desired state of the package.
default: present default: present
choices: [ present, latest ] choices: [ present, latest, absent ]
upgrade: upgrade:
description: description:
@@ -97,6 +97,14 @@ msg:
description: action that has been taken description: action that has been taken
helper: helper:
description: the helper that was actually used description: the helper that was actually used
installed:
description: list of packages that were newly installed
returned: always
type: list
updated:
description: list of packages that were upgraded
returned: always
type: list
''' '''
EXAMPLES = ''' EXAMPLES = '''
@@ -123,6 +131,15 @@ use_cmd_local_pkgbuild = {
'makepkg': ['makepkg', '--syncdeps', '--install', '--noconfirm', '--needed'] 'makepkg': ['makepkg', '--syncdeps', '--install', '--noconfirm', '--needed']
} }
use_remove_cmd = {
'yay': ['yay', '-R', '--noconfirm'],
'paru': ['paru', '-R', '--noconfirm'],
'pacaur': ['pacaur', '-R', '--noconfirm'],
'trizen': ['trizen', '-R', '--noconfirm'],
'pikaur': ['pikaur', '-R', '--noconfirm'],
'aurman': ['aurman', '-R', '--noconfirm'],
}
has_aur_option = ['yay', 'paru', 'pacaur', 'trizen', 'pikaur', 'aurman'] has_aur_option = ['yay', 'paru', 'pacaur', 'trizen', 'pikaur', 'aurman']
@@ -138,18 +155,8 @@ def check_packages(module, packages):
""" """
Inform the user what would change if the module were run Inform the user what would change if the module were run
""" """
would_be_changed = [] would_be_changed = [package for package in packages if not package_installed(module, package)]
diff = { diff = {'before': '', 'after': '\n'.join(package for package in would_be_changed if module._diff)}
'before': '',
'after': '',
}
for package in packages:
installed = package_installed(module, package)
if not installed:
would_be_changed.append(package)
if module._diff:
diff['after'] += package + "\n"
if would_be_changed: if would_be_changed:
status = True status = True
@@ -166,6 +173,28 @@ def check_packages(module, packages):
module.exit_json(changed=status, msg=message, diff=diff) module.exit_json(changed=status, msg=message, diff=diff)
def check_packages_absent(module, packages):
"""
Inform the user what would change if the module were run with state=absent
"""
would_be_changed = [package for package in packages if package_installed(module, package)]
diff = {'before': '\n'.join(pkg for pkg in would_be_changed if module._diff), 'after': ''}
if would_be_changed:
status = True
if len(packages) > 1:
message = '{} package(s) would be removed'.format(len(would_be_changed))
else:
message = 'package would be removed'
else:
status = False
if len(packages) > 1:
message = 'all packages are already absent'
else:
message = 'package is already absent'
module.exit_json(changed=status, msg=message, diff=diff)
def build_command_prefix(use, extra_args, skip_pgp_check=False, ignore_arch=False, aur_only=False, local_pkgbuild=None, update_cache=False): def build_command_prefix(use, extra_args, skip_pgp_check=False, ignore_arch=False, aur_only=False, local_pkgbuild=None, update_cache=False):
""" """
Create the prefix of a command that can be used by the install and upgrade functions. Create the prefix of a command that can be used by the install and upgrade functions.
@@ -231,11 +260,10 @@ def check_upgrade(module, use):
Inform user how many packages would be upgraded Inform user how many packages would be upgraded
""" """
rc, stdout, stderr = module.run_command([use, '-Qu'], check_rc=True) rc, stdout, stderr = module.run_command([use, '-Qu'], check_rc=True)
data = stdout.split('\n') num_packages = sum(1 for line in stdout.splitlines() if line.strip())
data.remove('')
module.exit_json( module.exit_json(
changed=len(data) > 0, changed=num_packages > 0,
msg="{} package(s) would be upgraded".format(len(data)), msg=f"{num_packages} package(s) would be upgraded",
helper=use, helper=use,
) )
@@ -267,13 +295,14 @@ def install_packages(module, packages, use, extra_args, state, skip_pgp_check, i
else: else:
assert use in use_cmd assert use in use_cmd
changed_iter = False installed_pkgs = []
updated_pkgs = []
for package in packages: for package in packages:
if state == 'present': was_installed = package_installed(module, package)
if package_installed(module, package): if state == 'present' and was_installed:
rc = 0 rc = 0
continue continue
if use == 'makepkg': if use == 'makepkg':
rc, out, err = install_with_makepkg(module, package, extra_args, skip_pgp_check, ignore_arch, local_pkgbuild) rc, out, err = install_with_makepkg(module, package, extra_args, skip_pgp_check, ignore_arch, local_pkgbuild)
elif local_pkgbuild: elif local_pkgbuild:
@@ -283,7 +312,14 @@ def install_packages(module, packages, use, extra_args, state, skip_pgp_check, i
command.append(package) command.append(package)
rc, out, err = module.run_command(command, check_rc=True) rc, out, err = module.run_command(command, check_rc=True)
changed_iter = changed_iter or not (out == '' or '-- skipping' in out or 'nothing to do' in out.lower()) changed_pkg = not (out == '' or 'up-to-date -- skipping' in out or 'nothing to do' in out.lower())
if changed_pkg:
if was_installed:
updated_pkgs.append(package)
else:
installed_pkgs.append(package)
changed_iter = bool(installed_pkgs or updated_pkgs)
message = 'installed package(s)' if changed_iter else 'package(s) already installed' message = 'installed package(s)' if changed_iter else 'package(s) already installed'
@@ -292,6 +328,46 @@ def install_packages(module, packages, use, extra_args, state, skip_pgp_check, i
msg=message if not rc else err, msg=message if not rc else err,
helper=use, helper=use,
rc=rc, rc=rc,
installed=installed_pkgs,
updated=updated_pkgs,
)
def remove_packages(module, packages, use, extra_args):
"""
Remove the specified packages
"""
# Determine the base command. If a helper other than makepkg is selected, use it.
# If use is 'makepkg' (which has no remove capability), fallback to pacman directly.
def base_remove_cmd(selected_use):
if selected_use == 'makepkg':
return def_lang + ['pacman', '-R', '--noconfirm'], 'pacman'
else:
assert selected_use in use_remove_cmd
return def_lang + use_remove_cmd[selected_use], selected_use
changed_iter = False
rc = 0
used_helper = use
for package in packages:
if not package_installed(module, package):
continue
command, helper = base_remove_cmd(use)
used_helper = helper
if extra_args:
command += shlex.split(extra_args)
command.append(package)
rc, out, err = module.run_command(command, check_rc=True)
changed_iter |= (rc == 0)
message = 'removed package(s)' if changed_iter else 'package(s) already absent'
module.exit_json(
changed=changed_iter,
msg=message if not rc else err,
helper=used_helper,
rc=rc,
) )
@@ -303,7 +379,7 @@ def make_module():
}, },
'state': { 'state': {
'default': 'present', 'default': 'present',
'choices': ['present', 'latest'], 'choices': ['present', 'latest', 'absent'],
}, },
'upgrade': { 'upgrade': {
'type': 'bool', 'type': 'bool',
@@ -387,18 +463,27 @@ def apply_module(module, use):
upgrade(module, use, params['extra_args'], params['aur_only'], params['update_cache']) upgrade(module, use, params['extra_args'], params['aur_only'], params['update_cache'])
else: else:
if module.check_mode: if module.check_mode:
check_packages(module, params['name']) if params['state'] == 'absent':
check_packages_absent(module, params['name'])
else:
check_packages(module, params['name'])
else: else:
install_packages(module, if params['state'] == 'absent':
params['name'], remove_packages(module,
use, params['name'],
params['extra_args'], use,
params['state'], params['extra_args'])
params['skip_pgp_check'], else:
params['ignore_arch'], install_packages(module,
params['aur_only'], params['name'],
params['local_pkgbuild'], use,
params['update_cache']) params['extra_args'],
params['state'],
params['skip_pgp_check'],
params['ignore_arch'],
params['aur_only'],
params['local_pkgbuild'],
params['update_cache'])
def main(): def main():