1
0
Fork 0
mirror of https://github.com/ansible-collections/community.general.git synced 2026-02-04 07:51:50 +00:00
community.general/plugins/modules/composer.py
2025-11-01 13:46:53 +01:00

279 lines
8.9 KiB
Python

#!/usr/bin/python
# Copyright (c) 2014, Dimitrios Tydeas Mengidis <tydeas.dr@gmail.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 annotations
DOCUMENTATION = r"""
module: composer
author:
- "Dimitrios Tydeas Mengidis (@dmtrs)"
- "René Moser (@resmo)"
short_description: Dependency Manager for PHP
description:
- Composer is a tool for dependency management in PHP. It allows you to declare the dependent libraries your project needs
and it installs them in your project for you.
extends_documentation_fragment:
- community.general.attributes
attributes:
check_mode:
support: full
diff_mode:
support: none
options:
command:
type: str
description:
- Composer command like V(install), V(update) and so on.
default: install
arguments:
type: str
description:
- Composer arguments like required package, version and so on.
default: ''
executable:
type: path
description:
- Path to PHP executable on the remote host, if PHP is not in E(PATH).
aliases: [php_path]
working_dir:
type: path
description:
- Directory of your project (see C(--working-dir)). This is required when the command is not run globally.
- This is ignored if O(global_command=true).
global_command:
description:
- Runs the specified command globally.
type: bool
default: false
prefer_source:
description:
- Forces installation from package sources when possible (see C(--prefer-source)).
default: false
type: bool
prefer_dist:
description:
- Forces installation from package dist even for dev versions (see C(--prefer-dist)).
default: false
type: bool
no_dev:
description:
- Disables installation of require-dev packages (see C(--no-dev)).
default: true
type: bool
no_scripts:
description:
- Skips the execution of all scripts defined in composer.json (see C(--no-scripts)).
default: false
type: bool
no_plugins:
description:
- Disables all plugins (see C(--no-plugins)).
default: false
type: bool
optimize_autoloader:
description:
- Optimize autoloader during autoloader dump (see C(--optimize-autoloader)).
- Convert PSR-0/4 autoloading to classmap to get a faster autoloader.
- Recommended especially for production, but can take a bit of time to run.
default: true
type: bool
classmap_authoritative:
description:
- Autoload classes from classmap only.
- Implicitly enable optimize_autoloader.
- Recommended especially for production, but can take a bit of time to run.
default: false
type: bool
apcu_autoloader:
description:
- Uses APCu to cache found/not-found classes.
default: false
type: bool
ignore_platform_reqs:
description:
- Ignore C(php), C(hhvm), C(lib-*) and C(ext-*) requirements and force the installation even if the local machine does not fulfill
these.
default: false
type: bool
composer_executable:
type: path
description:
- Path to composer executable on the remote host, if composer is not in E(PATH) or a custom composer is needed.
version_added: 3.2.0
requirements:
- php
- composer installed in bin path (recommended C(/usr/local/bin)) or specified in O(composer_executable)
notes:
- Default options that are always appended in each execution are C(--no-ansi), C(--no-interaction) and C(--no-progress)
if available.
- We received reports about issues on macOS if composer was installed by Homebrew. Please use the official install method
to avoid issues.
"""
EXAMPLES = r"""
- name: Download and installs all libs and dependencies outlined in the /path/to/project/composer.lock
community.general.composer:
command: install
working_dir: /path/to/project
- name: Install a new package
community.general.composer:
command: require
arguments: my/package
working_dir: /path/to/project
- name: Clone and install a project with all dependencies
community.general.composer:
command: create-project
arguments: package/package /path/to/project ~1.0
working_dir: /path/to/project
prefer_dist: true
- name: Install a package globally
community.general.composer:
command: require
global_command: true
arguments: my/package
"""
import re
import shlex
from ansible.module_utils.basic import AnsibleModule
def parse_out(string):
return re.sub(r"\s+", " ", string).strip()
def has_changed(string):
for no_change in ["Nothing to install or update", "Nothing to install, update or remove"]:
if no_change in string:
return False
return True
def get_available_options(module, command="install"):
# get all available options from a composer command using composer help to json
rc, out, err = composer_command(module, ["help", command], arguments=["--no-interaction", "--format=json"])
if rc != 0:
output = parse_out(err)
module.fail_json(msg=output)
command_help_json = module.from_json(out)
return command_help_json["definition"]["options"]
def composer_command(module, command, arguments=None, options=None):
if options is None:
options = []
if arguments is None:
arguments = []
global_command = module.params["global_command"]
if global_command:
global_arg = ["global"]
else:
global_arg = []
options.extend(["--working-dir", module.params["working_dir"]])
if module.params["executable"] is None:
php_path = module.get_bin_path("php", True, ["/usr/local/bin"])
else:
php_path = module.params["executable"]
if module.params["composer_executable"] is None:
composer_path = module.get_bin_path("composer", True, ["/usr/local/bin"])
else:
composer_path = module.params["composer_executable"]
cmd = [php_path, composer_path] + global_arg + command + options + arguments
return module.run_command(cmd)
def main():
module = AnsibleModule(
argument_spec=dict(
command=dict(default="install", type="str"),
arguments=dict(default="", type="str"),
executable=dict(type="path", aliases=["php_path"]),
working_dir=dict(type="path"),
global_command=dict(default=False, type="bool"),
prefer_source=dict(default=False, type="bool"),
prefer_dist=dict(default=False, type="bool"),
no_dev=dict(default=True, type="bool"),
no_scripts=dict(default=False, type="bool"),
no_plugins=dict(default=False, type="bool"),
apcu_autoloader=dict(default=False, type="bool"),
optimize_autoloader=dict(default=True, type="bool"),
classmap_authoritative=dict(default=False, type="bool"),
ignore_platform_reqs=dict(default=False, type="bool"),
composer_executable=dict(type="path"),
),
required_if=[("global_command", False, ["working_dir"])],
supports_check_mode=True,
)
# Get composer command with fallback to default
command = module.params["command"]
if re.search(r"\s", command):
module.fail_json(msg="Use the 'arguments' param for passing arguments with the 'command'")
arguments = shlex.split(module.params["arguments"])
available_options = get_available_options(module=module, command=command)
options = []
# Default options
default_options = [
"no-ansi",
"no-interaction",
"no-progress",
]
for option in default_options:
if option in available_options:
option = f"--{option}"
options.append(option)
option_params = {
"prefer_source": "prefer-source",
"prefer_dist": "prefer-dist",
"no_dev": "no-dev",
"no_scripts": "no-scripts",
"no_plugins": "no-plugins",
"apcu_autoloader": "acpu-autoloader",
"optimize_autoloader": "optimize-autoloader",
"classmap_authoritative": "classmap-authoritative",
"ignore_platform_reqs": "ignore-platform-reqs",
}
for param, option in option_params.items():
if module.params.get(param) and option in available_options:
option = f"--{option}"
options.append(option)
if module.check_mode:
if "dry-run" in available_options:
options.append("--dry-run")
else:
module.exit_json(skipped=True, msg=f"command '{command}' does not support check mode, skipping")
rc, out, err = composer_command(module, [command], arguments, options)
if rc != 0:
output = parse_out(err)
module.fail_json(msg=output, stdout=err)
else:
# Composer version > 1.0.0-alpha9 now use stderr for standard notification messages
output = parse_out(out + err)
module.exit_json(changed=has_changed(output), msg=output, stdout=out + err)
if __name__ == "__main__":
main()