diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/debputy/analysis/debian_dir.py | 54 | ||||
-rw-r--r-- | src/debputy/commands/debputy_cmd/lint_and_lsp_cmds.py | 10 | ||||
-rw-r--r-- | src/debputy/lsp/lsp_debian_control_reference_data.py | 42 | ||||
-rw-r--r-- | src/debputy/lsp/lsp_features.py | 23 |
4 files changed, 85 insertions, 44 deletions
diff --git a/src/debputy/analysis/debian_dir.py b/src/debputy/analysis/debian_dir.py index 68a5c1b..d63b0c9 100644 --- a/src/debputy/analysis/debian_dir.py +++ b/src/debputy/analysis/debian_dir.py @@ -15,6 +15,7 @@ from typing import ( Iterator, TypedDict, NotRequired, + FrozenSet, ) from debputy.analysis import REFERENCE_DATA_TABLE @@ -277,12 +278,49 @@ def _kpf_install_pattern( return ppkpf.info.get("install_pattern") +def _parse_dh_cmd_list( + cmd_list: Optional[List[Union[Mapping[str, Any], object]]] +) -> Iterable[str]: + if not isinstance(cmd_list, list): + return + + for command in cmd_list: + if not isinstance(command, dict): + continue + command_name = command.get("command") + if isinstance(command_name, str): + yield command_name + + +def _resolve_active_and_inactive_dh_commands( + dh_rules_addons: Iterable[str], +) -> Tuple[FrozenSet[str], FrozenSet[str]]: + cmd = ["dh_assistant", "list-commands", "--output-format=json"] + if dh_rules_addons: + addons = ",".join(dh_rules_addons) + cmd.append(f"--with={addons}") + try: + output = subprocess.check_output( + cmd, + stderr=subprocess.DEVNULL, + ) + except (subprocess.CalledProcessError, FileNotFoundError): + return frozenset(), frozenset() + else: + result = json.loads(output) + active_commands = frozenset(_parse_dh_cmd_list(result.get("commands"))) + disabled_commands = frozenset( + _parse_dh_cmd_list(result.get("disabled-commands")) + ) + return active_commands, disabled_commands + + def _resolve_debhelper_config_files( debian_dir: VirtualPath, binary_packages: Mapping[str, BinaryPackage], debputy_plugin_metadata: DebputyPluginMetadata, dh_ppf_docs: Dict[str, PluginProvidedKnownPackagingFile], - dh_rules_addons: Iterable[str], + dh_rules_addons: Sequence[str], dh_compat_level: int, saw_dh: bool, ) -> Tuple[List[PackagerProvidedFile], Optional[object], int]: @@ -311,6 +349,7 @@ def _resolve_debhelper_config_files( "config-files", [] ) issues = result.get("issues") + active_commands, _ = _resolve_active_and_inactive_dh_commands(dh_rules_addons) for config_file in config_files: if not isinstance(config_file, dict): continue @@ -369,6 +408,8 @@ def _resolve_debhelper_config_files( else: continue is_active = command.get("is-active", True) + if is_active is None and command_name in active_commands: + is_active = True if not isinstance(is_active, bool): continue if is_active: @@ -396,6 +437,13 @@ def _resolve_debhelper_config_files( "packageless_is_fallback_for_all_packages", False, ) + has_active_command = ( + ppkpf.info.get("has_active_command", False) if saw_dh else False + ) + if not has_active_command: + dh_cmds = ppkpf.info.get("debhelper_commands") + if dh_cmds: + has_active_command = any(c in active_commands for c in dh_cmds) dh_ppfs[stem] = _fake_PPFClassSpec( debputy_plugin_metadata, stem, @@ -404,9 +452,7 @@ def _resolve_debhelper_config_files( default_priority=default_priority, post_formatting_rewrite=post_formatting_rewrite, packageless_is_fallback_for_all_packages=packageless_is_fallback_for_all_packages, - has_active_command=( - ppkpf.info.get("has_active_command", False) if saw_dh else False - ), + has_active_command=has_active_command, ) all_dh_ppfs = list( flatten_ppfs( diff --git a/src/debputy/commands/debputy_cmd/lint_and_lsp_cmds.py b/src/debputy/commands/debputy_cmd/lint_and_lsp_cmds.py index e8e380e..46b536b 100644 --- a/src/debputy/commands/debputy_cmd/lint_and_lsp_cmds.py +++ b/src/debputy/commands/debputy_cmd/lint_and_lsp_cmds.py @@ -246,9 +246,13 @@ def lsp_describe_features(context: CommandContext) -> None: from debputy.lsp.lsp_self_check import assert_can_start_lsp - assert_can_start_lsp() - - from debputy.lsp.lsp_features import describe_lsp_features + try: + from debputy.lsp.lsp_features import describe_lsp_features + except ImportError: + assert_can_start_lsp() + raise AssertionError( + "Cannot load the language server features but `assert_can_start_lsp` did not fail" + ) describe_lsp_features(context) diff --git a/src/debputy/lsp/lsp_debian_control_reference_data.py b/src/debputy/lsp/lsp_debian_control_reference_data.py index 9a589bf..1e32d3c 100644 --- a/src/debputy/lsp/lsp_debian_control_reference_data.py +++ b/src/debputy/lsp/lsp_debian_control_reference_data.py @@ -2874,12 +2874,13 @@ BINARY_FIELDS = _fields( Conflicts: bar ``` - Please check the description of the **Breaks** field for when you would use **Breaks** vs. - **Conflicts**. + Please see https://wiki.debian.org/PackageTransition for when to uses **Breaks**, **Conflicts**, + or/and **Replaces**. Note if a package conflicts with itself (indirectly or via **Provides**), then it is using a special rule for **Conflicts**. See section - 7.6.2 "[Replacing whole packages, forcing their removal]" in the Debian Policy Manual. + 7.6.2 "[Replacing whole packages, forcing their removal]" in the Debian Policy Manual for the + details. [Replacing whole packages, forcing their removal]: https://www.debian.org/doc/debian-policy/ch-relationships.html#replacing-whole-packages-forcing-their-removal """ @@ -2902,31 +2903,12 @@ BINARY_FIELDS = _fields( Breaks: bar (<= 1.0~) ```` - **Breaks vs. Conflicts**: + Please see https://wiki.debian.org/PackageTransition for when to uses **Breaks**, **Conflicts**, or/and + **Replaces**. - * I moved files from **foo** to **bar** in version X, what should I do? - - Add `Breaks: foo (<< X~)` + `Replaces: foo (<< X~)` to **bar** - - * Upgrading **bar** while **foo** is version X or less causes **foo** or **bar** to break. - How do I solve this? - - Add `Breaks: foo (<< X~)` to **bar** - - * The **foo** and **bar** packages provide the same functionality (interface) but different - implementations and there can be at most one of them. What should I do? - - See section 7.6.2 [Replacing whole packages, forcing their removal] in the Debian Policy Manual. - - * How to handle when **foo** and **bar** packages are unrelated but happen to provide the same binary? - - Attempt to resolve the name conflict by renaming the clashing files in question on either (or both) sides. - - Note the use of *~* in version numbers in the answers are generally used to ensure this works correctly in + Note the use of *~* in version numbers in the example is used to ensure this works correctly in case of a backports (in the Debian archive), where the package is rebuilt with the "~bpo" suffix in its version. - - [Replacing whole packages, forcing their removal]: https://www.debian.org/doc/debian-policy/ch-relationships.html#replacing-whole-packages-forcing-their-removal """ ), ), @@ -2939,7 +2921,7 @@ BINARY_FIELDS = _fields( This package either replaces another package or overwrites files that used to be provided by another package. - **Attention**: The `Replaces` field is **always** used with either `Breaks` or `Conflicts` field. + **Attention**: The `Replaces` field should **always** used with either `Breaks` or `Conflicts` field. **Example**: ``` @@ -2952,8 +2934,12 @@ BINARY_FIELDS = _fields( Breaks: foo (<< 1.2-3~) ``` - Please check the description of the `Breaks` field for when you would use `Breaks` vs. `Conflicts`. - It also covers common uses of `Replaces`. + Please see https://wiki.debian.org/PackageTransition for when to uses **Breaks**, **Conflicts**, + or/and **Replaces**. + + Note the use of *~* in version numbers in the example is used to ensure this works correctly in + case of a backports (in the Debian archive), where the package is rebuilt with the "~bpo" suffix in its + version. """ ), ), diff --git a/src/debputy/lsp/lsp_features.py b/src/debputy/lsp/lsp_features.py index 800f738..c513523 100644 --- a/src/debputy/lsp/lsp_features.py +++ b/src/debputy/lsp/lsp_features.py @@ -320,18 +320,23 @@ def describe_lsp_features(context: CommandContext) -> None: print("General features:") for self_check in LSP_CHECKS: is_ok = self_check.test() - assert not self_check.is_mandatory or is_ok - if self_check.is_mandatory: - continue if is_ok: print(f" * {self_check.feature}: {fo.colored('enabled', fg='green')}") else: - disabled = fo.colored( - "disabled", - fg="yellow", - bg="black", - style="bold", - ) + if self_check.is_mandatory: + disabled = fo.colored( + "missing", + fg="red", + bg="black", + style="bold", + ) + else: + disabled = fo.colored( + "disabled", + fg="yellow", + bg="black", + style="bold", + ) if self_check.how_to_fix: print(f" * {self_check.feature}: {disabled}") |