mirror of
https://github.com/kewlfft/ansible-aur.git
synced 2025-09-17 08:30:39 +03:00
Compare commits
37 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
7affd00193 | ||
![]() |
8e6f0f49bd | ||
![]() |
bb73f524ca | ||
![]() |
62193e882c | ||
![]() |
290b852836 | ||
![]() |
31828edaf9 | ||
![]() |
610e14b8c2 | ||
![]() |
b8da3380f5 | ||
![]() |
984cb9fde1 | ||
![]() |
0d4d6aa3e7 | ||
![]() |
d2f3fb5802 | ||
![]() |
3ec32e68a3 | ||
![]() |
37d4b45acd | ||
![]() |
ebb5e9aed1 | ||
![]() |
2ef95d29d6 | ||
![]() |
c58c59c63d | ||
![]() |
b2ed7c1dc6 | ||
![]() |
a2267b88a3 | ||
![]() |
768c15ad24 | ||
![]() |
0182711634 | ||
![]() |
73c68bfd79 | ||
![]() |
21f86cbb6e | ||
![]() |
c348e578cb | ||
![]() |
fe050018db | ||
![]() |
326836ddbe | ||
![]() |
583c7597a9 | ||
![]() |
cbf843e1db | ||
![]() |
2a076dc6a0 | ||
![]() |
8d177a7550 | ||
![]() |
b469b5a641 | ||
![]() |
63d3056a82 | ||
![]() |
91c6d70184 | ||
![]() |
9b25b11cea | ||
![]() |
2c090a49a8 | ||
![]() |
07e5fe301f | ||
![]() |
49a077790e | ||
![]() |
87a15853fa |
72
.github/workflows/galaxy.yml
vendored
72
.github/workflows/galaxy.yml
vendored
@@ -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,49 @@ 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: Debug workflow version
|
||||||
id: version
|
run: echo "=== RUNNING UPDATED WORKFLOW ==="
|
||||||
run: |
|
|
||||||
[ "$(cat galaxy.yml | grep version: | awk '{print $2}')" = $(awk -F '/' '{print substr($3, 2)}' <<< ${GITHUB_REF}) ] || exit 1
|
|
||||||
echo ::set-output name=version::$(awk -F '/' '{print substr($3, 2)}' <<< ${GITHUB_REF})
|
|
||||||
|
|
||||||
- name: "Set up Python 3.9"
|
- name: Verify version in galaxy.yml matches release tag
|
||||||
uses: actions/setup-python@v2
|
run: |
|
||||||
|
tag_version="${GITHUB_REF##*/}" # e.g., v0.12.2
|
||||||
|
tag_version="${tag_version#v}" # strip leading 'v' if present
|
||||||
|
file_version=$(awk '/^version:/ {print $2}' galaxy.yml)
|
||||||
|
|
||||||
|
if [ "${GITHUB_EVENT_NAME}" != "workflow_dispatch" ] && [ "$file_version" != "$tag_version" ]; then
|
||||||
|
echo "❌ galaxy.yml version ($file_version) does not match tag ($tag_version)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ Version check passed: $file_version"
|
||||||
|
|
||||||
|
- 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
|
if: github.event_name != 'workflow_dispatch' # skip publish for manual testing
|
||||||
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"
|
|
||||||
|
17
README.md
17
README.md
@@ -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
|
||||||
|
@@ -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.9.3
|
version: 0.13.0
|
||||||
|
|
||||||
# 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
|
||||||
|
@@ -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,11 +295,12 @@ 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':
|
||||||
@@ -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,7 +463,16 @@ 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:
|
||||||
|
if params['state'] == 'absent':
|
||||||
|
check_packages_absent(module, params['name'])
|
||||||
|
else:
|
||||||
check_packages(module, params['name'])
|
check_packages(module, params['name'])
|
||||||
|
else:
|
||||||
|
if params['state'] == 'absent':
|
||||||
|
remove_packages(module,
|
||||||
|
params['name'],
|
||||||
|
use,
|
||||||
|
params['extra_args'])
|
||||||
else:
|
else:
|
||||||
install_packages(module,
|
install_packages(module,
|
||||||
params['name'],
|
params['name'],
|
||||||
|
Reference in New Issue
Block a user