claude-dev-env 1.58.0 → 1.60.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (106) hide show
  1. package/CLAUDE.md +2 -2
  2. package/_shared/pr-loop/scripts/code_rules_gate.py +36 -3
  3. package/_shared/pr-loop/scripts/pr_loop_shared_constants/code_rules_gate_constants.py +6 -0
  4. package/_shared/pr-loop/scripts/pr_loop_shared_constants/reviews_disabled_constants.py +1 -0
  5. package/_shared/pr-loop/scripts/reviews_disabled.py +12 -0
  6. package/_shared/pr-loop/scripts/tests/test_code_rules_gate.py +265 -0
  7. package/_shared/pr-loop/scripts/tests/test_reviews_disabled.py +29 -0
  8. package/audit-rubrics/category_rubrics/category-b-selector-engine-compat.md +1 -1
  9. package/audit-rubrics/category_rubrics/category-e-dead-code.md +1 -0
  10. package/audit-rubrics/category_rubrics/category-o-docstring-vs-impl-drift.md +1 -1
  11. package/audit-rubrics/prompts/category-b-selector-engine-compat.md +2 -2
  12. package/bin/install.mjs +100 -27
  13. package/bin/install.test.mjs +133 -1
  14. package/docs/CODE_RULES.md +3 -3
  15. package/hooks/blocking/code_rules_annotations_length.py +153 -0
  16. package/hooks/blocking/code_rules_dead_dataclass_field.py +319 -0
  17. package/hooks/blocking/code_rules_dead_module_constant.py +321 -0
  18. package/hooks/blocking/code_rules_duplicate_body.py +439 -0
  19. package/hooks/blocking/code_rules_enforcer.py +190 -21
  20. package/hooks/blocking/code_rules_magic_values.py +98 -0
  21. package/hooks/blocking/code_rules_shared.py +41 -0
  22. package/hooks/blocking/code_rules_typeddict_stub.py +172 -0
  23. package/hooks/blocking/config/__init__.py +5 -0
  24. package/hooks/blocking/config/verified_commit_constants.py +106 -0
  25. package/hooks/blocking/destructive_command_blocker.py +1027 -12
  26. package/hooks/blocking/hook_prose_detector_consistency.py +150 -0
  27. package/hooks/blocking/subprocess_budget_completeness.py +380 -0
  28. package/hooks/blocking/test_code_rules_enforcer_annotations.py +225 -0
  29. package/hooks/blocking/test_code_rules_enforcer_cap_meta.py +1 -0
  30. package/hooks/blocking/test_code_rules_enforcer_cross_skill_duplicate.py +146 -0
  31. package/hooks/blocking/test_code_rules_enforcer_dead_dataclass_field.py +467 -0
  32. package/hooks/blocking/test_code_rules_enforcer_dead_module_constant.py +188 -0
  33. package/hooks/blocking/test_code_rules_enforcer_duplicate_body.py +330 -0
  34. package/hooks/blocking/test_code_rules_enforcer_duplicate_body_hook_routing.py +179 -0
  35. package/hooks/blocking/test_code_rules_enforcer_magic_slice_bounds.py +133 -0
  36. package/hooks/blocking/test_code_rules_enforcer_zero_payload_alias.py +415 -0
  37. package/hooks/blocking/test_code_rules_enforcer_zero_payload_alias_hook_routing.py +156 -0
  38. package/hooks/blocking/test_destructive_command_blocker.py +622 -3
  39. package/hooks/blocking/test_hook_prose_detector_consistency.py +265 -0
  40. package/hooks/blocking/test_subprocess_budget_completeness.py +588 -0
  41. package/hooks/blocking/test_verdict_directory_write_blocker.py +720 -0
  42. package/hooks/blocking/test_verification_verdict_store.py +278 -0
  43. package/hooks/blocking/test_verified_commit_gate.py +368 -0
  44. package/hooks/blocking/test_verified_commit_message_accuracy_blocker.py +131 -0
  45. package/hooks/blocking/test_verifier_verdict_minter.py +214 -0
  46. package/hooks/blocking/test_workflow_substitution_slot_blocker.py +242 -0
  47. package/hooks/blocking/verdict_directory_write_blocker.py +667 -0
  48. package/hooks/blocking/verification_verdict_store.py +446 -0
  49. package/hooks/blocking/verified_commit_gate.py +523 -0
  50. package/hooks/blocking/verified_commit_message_accuracy_blocker.py +152 -0
  51. package/hooks/blocking/verifier_verdict_minter.py +299 -0
  52. package/hooks/blocking/workflow_substitution_slot_blocker.py +159 -0
  53. package/hooks/diagnostic/test_hook_log_extractor.py +3 -3
  54. package/hooks/hooks.json +58 -1
  55. package/hooks/hooks_constants/blocking_check_limits.py +1 -0
  56. package/hooks/hooks_constants/code_rules_enforcer_constants.py +16 -0
  57. package/hooks/hooks_constants/dead_dataclass_field_constants.py +25 -0
  58. package/hooks/hooks_constants/dead_module_constant_constants.py +20 -0
  59. package/hooks/hooks_constants/destructive_command_segment_constants.py +178 -0
  60. package/hooks/hooks_constants/duplicate_function_body_constants.py +34 -0
  61. package/hooks/hooks_constants/hook_prose_detector_consistency_constants.py +30 -0
  62. package/hooks/hooks_constants/precommit_code_rules_gate_constants.py +1 -1
  63. package/hooks/hooks_constants/subprocess_budget_completeness_constants.py +5 -0
  64. package/hooks/hooks_constants/workflow_substitution_slot_blocker_constants.py +22 -0
  65. package/package.json +1 -1
  66. package/rules/docstring-prose-matches-implementation.md +43 -0
  67. package/rules/file-global-constants.md +7 -1
  68. package/rules/hook-prose-matches-detector.md +26 -0
  69. package/rules/no-cross-skill-duplicate-helpers.md +29 -0
  70. package/rules/no-inline-destructive-literals.md +11 -0
  71. package/rules/workflow-substitution-slots.md +7 -0
  72. package/skills/_shared/pr-loop/scripts/preflight_worktree.py +392 -0
  73. package/skills/_shared/pr-loop/scripts/skills_pr_loop_constants/preflight_constants.py +70 -0
  74. package/skills/_shared/pr-loop/scripts/test_preflight_worktree.py +263 -0
  75. package/skills/autoconverge/SKILL.md +67 -19
  76. package/skills/autoconverge/reference/closing-report.md +59 -17
  77. package/skills/autoconverge/reference/convergence.md +7 -3
  78. package/skills/autoconverge/reference/stop-conditions.md +7 -2
  79. package/skills/autoconverge/workflow/aggregate_runs.py +371 -0
  80. package/skills/autoconverge/workflow/autoconverge_report_constants/render_report_constants.py +193 -76
  81. package/skills/autoconverge/workflow/converge.clean-audit.test.mjs +76 -0
  82. package/skills/autoconverge/workflow/converge.contract.test.mjs +206 -206
  83. package/skills/autoconverge/workflow/converge.copilot-gate.test.mjs +265 -0
  84. package/skills/autoconverge/workflow/converge.mjs +234 -42
  85. package/skills/autoconverge/workflow/convergence_summary.py +110 -0
  86. package/skills/autoconverge/workflow/fixtures/wf_run/subagents/workflows/wf_881252e6-700/agent-ab1c2d3e4f5a6b7c8.jsonl +2 -0
  87. package/skills/autoconverge/workflow/fixtures/wf_run/workflows/wf_881252e6-700.json +7 -0
  88. package/skills/autoconverge/workflow/render_report.py +488 -397
  89. package/skills/autoconverge/workflow/test_aggregate_runs.py +134 -0
  90. package/skills/autoconverge/workflow/test_convergence_summary.py +132 -0
  91. package/skills/autoconverge/workflow/test_render_report.py +488 -259
  92. package/skills/pr-converge/reference/per-tick.md +28 -8
  93. package/skills/pr-converge/scripts/check_convergence.py +195 -64
  94. package/skills/pr-converge/scripts/test_check_convergence.py +173 -2
  95. package/skills/rebase/SKILL.md +2 -4
  96. package/skills/update/SKILL.md +37 -5
  97. package/system-prompts/software-engineer.xml +2 -6
  98. package/hooks/blocking/content_search_to_zoekt_redirector.py +0 -59
  99. package/hooks/blocking/content_search_zoekt_bash_block_reason.py +0 -25
  100. package/hooks/blocking/content_search_zoekt_block_payload.py +0 -21
  101. package/hooks/blocking/content_search_zoekt_indexed_paths.py +0 -24
  102. package/hooks/blocking/content_search_zoekt_indexed_roots_config.py +0 -131
  103. package/hooks/blocking/content_search_zoekt_redirect_guidance.py +0 -52
  104. package/hooks/blocking/test_content_search_to_zoekt_redirector_integration.py +0 -61
  105. package/hooks/blocking/test_content_search_to_zoekt_redirector_unit.py +0 -92
  106. package/hooks/blocking/test_content_search_zoekt_indexed_roots_config.py +0 -102
@@ -0,0 +1,392 @@
1
+ """Preflight guard for the working directory and worktree before a PR-convergence run.
2
+
3
+ Usage:
4
+ python preflight_worktree.py --owner <O> --repo <R> --mode <strict|classify>
5
+
6
+ Modes:
7
+ strict — autoconverge: the working directory must be the PR's own repo so
8
+ EnterWorktree can create and enter the branch worktree. Any other
9
+ state aborts (exit 1).
10
+ classify — pr-converge: emit the environment classification so the caller can
11
+ route. same_repo and different_repo both succeed (exit 0); a
12
+ re-rooted session (no git work tree, or no readable origin) aborts
13
+ (exit 1).
14
+
15
+ Output (stdout):
16
+ A line 'PREFLIGHT_OUTCOME=<same_repo|different_repo|re_rooted>', a
17
+ human-readable summary, and, on abort, a recovery instruction.
18
+
19
+ Exit codes:
20
+ 0 — safe to continue for the given mode
21
+ 1 — abort (the caller must stop and recover)
22
+ """
23
+
24
+ from __future__ import annotations
25
+
26
+ import argparse
27
+ import subprocess
28
+ import sys
29
+ from dataclasses import dataclass
30
+ from pathlib import Path
31
+
32
+ _self_dir = Path(__file__).resolve().parent
33
+ if str(_self_dir) not in sys.path:
34
+ sys.path.insert(0, str(_self_dir))
35
+
36
+ from skills_pr_loop_constants.preflight_constants import (
37
+ ABORT_DIFFERENT_REPO_STRICT_TEMPLATE,
38
+ ABORT_RE_ROOTED_TEMPLATE,
39
+ ABORT_WORKTREE_BROKEN_TEMPLATE,
40
+ ALL_GIT_IS_INSIDE_WORK_TREE_ARGS,
41
+ ALL_GIT_REMOTE_GET_URL_ARGS,
42
+ ALL_GIT_WORKTREE_LIST_ARGS,
43
+ ALL_PREFLIGHT_MODES,
44
+ CWD_IDENTITY_UNKNOWN,
45
+ EXIT_PREFLIGHT_ABORT,
46
+ EXIT_PREFLIGHT_OK,
47
+ GIT_DIRECTORY_FLAG,
48
+ GIT_EXECUTABLE,
49
+ GIT_INSIDE_WORK_TREE_TRUE,
50
+ GIT_SUBPROCESS_TIMEOUT_SECONDS,
51
+ MODE_ARG_FLAG,
52
+ MODE_STRICT,
53
+ OUTCOME_DIFFERENT_REPO,
54
+ OUTCOME_MARKER_TEMPLATE,
55
+ OUTCOME_RE_ROOTED,
56
+ OUTCOME_SAME_REPO,
57
+ OWNER_ARG_FLAG,
58
+ PREFLIGHT_CLI_DESCRIPTION,
59
+ REMOTE_URL_IDENTITY_PATTERN,
60
+ REPO_ARG_FLAG,
61
+ ROUTE_DIFFERENT_REPO_TEMPLATE,
62
+ SUMMARY_DIFFERENT_REPO_TEMPLATE,
63
+ SUMMARY_RE_ROOTED_TEMPLATE,
64
+ SUMMARY_SAME_REPO_TEMPLATE,
65
+ )
66
+
67
+
68
+ @dataclass(frozen=True)
69
+ class RepoIdentity:
70
+ """A GitHub repository identity parsed from a git remote URL.
71
+
72
+ Attributes:
73
+ owner: Repository owner (login or org), lower-cased for comparison.
74
+ repo: Repository name, lower-cased for comparison.
75
+ """
76
+
77
+ owner: str
78
+ repo: str
79
+
80
+
81
+ @dataclass(frozen=True)
82
+ class PreflightVerdict:
83
+ """Classification of the working directory against a target PR repo.
84
+
85
+ Attributes:
86
+ outcome: One of OUTCOME_SAME_REPO, OUTCOME_DIFFERENT_REPO, or
87
+ OUTCOME_RE_ROOTED.
88
+ cwd_identity: Parsed identity of the working directory's origin, or
89
+ None when there is no git work tree, no origin, or an unparseable
90
+ origin URL.
91
+ has_healthy_worktree_machinery: True when 'git worktree list' runs
92
+ cleanly in the working directory.
93
+ """
94
+
95
+ outcome: str
96
+ cwd_identity: RepoIdentity | None
97
+ has_healthy_worktree_machinery: bool
98
+
99
+
100
+ def parse_repo_identity(remote_url: str) -> RepoIdentity | None:
101
+ """Parse a GitHub owner/repo from a git remote URL.
102
+
103
+ Accepts the https, git@ (scp-like), and ssh:// forms and drops a trailing
104
+ '.git'. Comparison is case-insensitive, so the returned owner and repo are
105
+ lower-cased.
106
+
107
+ Args:
108
+ remote_url: The remote URL from 'git remote get-url origin'.
109
+
110
+ Returns:
111
+ A RepoIdentity, or None when the URL is not a GitHub remote.
112
+ """
113
+ match = REMOTE_URL_IDENTITY_PATTERN.search(remote_url.strip())
114
+ if match is None:
115
+ return None
116
+ return RepoIdentity(
117
+ owner=match.group("owner").lower(),
118
+ repo=match.group("repo").lower(),
119
+ )
120
+
121
+
122
+ def _run_git(
123
+ working_directory: Path, all_git_arguments: tuple[str, ...]
124
+ ) -> subprocess.CompletedProcess[str]:
125
+ """Run a git subcommand in a working directory with a bounded timeout.
126
+
127
+ Args:
128
+ working_directory: Directory to run git in (passed via 'git -C').
129
+ all_git_arguments: The git subcommand and its arguments.
130
+
131
+ Returns:
132
+ The completed process with captured text stdout and stderr.
133
+ """
134
+ return subprocess.run(
135
+ [
136
+ GIT_EXECUTABLE,
137
+ GIT_DIRECTORY_FLAG,
138
+ str(working_directory),
139
+ *all_git_arguments,
140
+ ],
141
+ capture_output=True,
142
+ text=True,
143
+ check=False,
144
+ timeout=GIT_SUBPROCESS_TIMEOUT_SECONDS,
145
+ )
146
+
147
+
148
+ def is_inside_work_tree(working_directory: Path) -> bool:
149
+ """Report whether the working directory is inside a git work tree.
150
+
151
+ Args:
152
+ working_directory: Directory to test.
153
+
154
+ Returns:
155
+ True when 'git rev-parse --is-inside-work-tree' prints 'true'.
156
+ """
157
+ completed = _run_git(working_directory, ALL_GIT_IS_INSIDE_WORK_TREE_ARGS)
158
+ return completed.returncode == 0 and (
159
+ completed.stdout.strip() == GIT_INSIDE_WORK_TREE_TRUE
160
+ )
161
+
162
+
163
+ def worktree_machinery_is_healthy(working_directory: Path) -> bool:
164
+ """Report whether git worktree machinery works in the working directory.
165
+
166
+ EnterWorktree relies on 'git worktree' to create and enter the branch
167
+ worktree; a failing worktree list means it cannot.
168
+
169
+ Args:
170
+ working_directory: Directory to test.
171
+
172
+ Returns:
173
+ True when 'git worktree list' exits cleanly.
174
+ """
175
+ completed = _run_git(working_directory, ALL_GIT_WORKTREE_LIST_ARGS)
176
+ return completed.returncode == 0
177
+
178
+
179
+ def classify_environment(
180
+ working_directory: Path, pr_identity: RepoIdentity
181
+ ) -> PreflightVerdict:
182
+ """Classify the working directory against the target PR repository.
183
+
184
+ Args:
185
+ working_directory: The session's current working directory.
186
+ pr_identity: The PR's owner/repo (lower-cased for comparison).
187
+
188
+ Returns:
189
+ A PreflightVerdict naming the outcome and the worktree-machinery health.
190
+ """
191
+ if not is_inside_work_tree(working_directory):
192
+ return PreflightVerdict(OUTCOME_RE_ROOTED, None, False)
193
+ has_healthy_machinery = worktree_machinery_is_healthy(working_directory)
194
+ remote = _run_git(working_directory, ALL_GIT_REMOTE_GET_URL_ARGS)
195
+ if remote.returncode != 0:
196
+ return PreflightVerdict(OUTCOME_RE_ROOTED, None, has_healthy_machinery)
197
+ cwd_identity = parse_repo_identity(remote.stdout)
198
+ if cwd_identity is None:
199
+ return PreflightVerdict(OUTCOME_DIFFERENT_REPO, None, has_healthy_machinery)
200
+ if cwd_identity == pr_identity:
201
+ return PreflightVerdict(OUTCOME_SAME_REPO, cwd_identity, has_healthy_machinery)
202
+ return PreflightVerdict(OUTCOME_DIFFERENT_REPO, cwd_identity, has_healthy_machinery)
203
+
204
+
205
+ def decide_exit_code(verdict: PreflightVerdict, mode: str) -> int:
206
+ """Decide the process exit code from a classification and the mode.
207
+
208
+ Args:
209
+ verdict: The classification of the working directory.
210
+ mode: MODE_STRICT (autoconverge) or MODE_CLASSIFY (pr-converge).
211
+
212
+ Returns:
213
+ EXIT_PREFLIGHT_OK when it is safe to continue, else EXIT_PREFLIGHT_ABORT.
214
+ """
215
+ if verdict.outcome == OUTCOME_RE_ROOTED:
216
+ return EXIT_PREFLIGHT_ABORT
217
+ if verdict.outcome == OUTCOME_SAME_REPO:
218
+ if not verdict.has_healthy_worktree_machinery:
219
+ return EXIT_PREFLIGHT_ABORT
220
+ return EXIT_PREFLIGHT_OK
221
+ if mode == MODE_STRICT:
222
+ return EXIT_PREFLIGHT_ABORT
223
+ return EXIT_PREFLIGHT_OK
224
+
225
+
226
+ def _cwd_identity_labels(verdict: PreflightVerdict) -> tuple[str, str]:
227
+ """Return display labels for the working directory's owner and repo.
228
+
229
+ Args:
230
+ verdict: The classification of the working directory.
231
+
232
+ Returns:
233
+ The lower-cased owner and repo, or the unknown placeholder for each
234
+ when the working directory identity could not be parsed.
235
+ """
236
+ if verdict.cwd_identity is None:
237
+ return CWD_IDENTITY_UNKNOWN, CWD_IDENTITY_UNKNOWN
238
+ return verdict.cwd_identity.owner, verdict.cwd_identity.repo
239
+
240
+
241
+ def _same_repo_lines(
242
+ verdict: PreflightVerdict, working_directory: Path, pr_identity: RepoIdentity
243
+ ) -> list[str]:
244
+ """Build the report lines for the same-repo outcome.
245
+
246
+ Args:
247
+ verdict: The classification of the working directory.
248
+ working_directory: The session's current working directory.
249
+ pr_identity: The PR's owner/repo.
250
+
251
+ Returns:
252
+ A summary line, plus an abort line when the worktree machinery is broken.
253
+ """
254
+ report_lines = [
255
+ SUMMARY_SAME_REPO_TEMPLATE.format(
256
+ owner=pr_identity.owner, repo=pr_identity.repo
257
+ )
258
+ ]
259
+ if not verdict.has_healthy_worktree_machinery:
260
+ report_lines.append(
261
+ ABORT_WORKTREE_BROKEN_TEMPLATE.format(cwd=working_directory)
262
+ )
263
+ return report_lines
264
+
265
+
266
+ def _different_repo_lines(
267
+ verdict: PreflightVerdict, mode: str, pr_identity: RepoIdentity
268
+ ) -> list[str]:
269
+ """Build the report lines for the different-repo outcome.
270
+
271
+ Args:
272
+ verdict: The classification of the working directory.
273
+ mode: MODE_STRICT (autoconverge) or MODE_CLASSIFY (pr-converge).
274
+ pr_identity: The PR's owner/repo.
275
+
276
+ Returns:
277
+ A summary line, plus an abort line under strict mode or a route line
278
+ under classify mode.
279
+ """
280
+ cwd_owner, cwd_repo = _cwd_identity_labels(verdict)
281
+ report_lines = [
282
+ SUMMARY_DIFFERENT_REPO_TEMPLATE.format(
283
+ cwd_owner=cwd_owner,
284
+ cwd_repo=cwd_repo,
285
+ owner=pr_identity.owner,
286
+ repo=pr_identity.repo,
287
+ )
288
+ ]
289
+ if mode == MODE_STRICT:
290
+ report_lines.append(
291
+ ABORT_DIFFERENT_REPO_STRICT_TEMPLATE.format(
292
+ cwd_owner=cwd_owner,
293
+ cwd_repo=cwd_repo,
294
+ owner=pr_identity.owner,
295
+ repo=pr_identity.repo,
296
+ )
297
+ )
298
+ else:
299
+ report_lines.append(
300
+ ROUTE_DIFFERENT_REPO_TEMPLATE.format(
301
+ owner=pr_identity.owner, repo=pr_identity.repo
302
+ )
303
+ )
304
+ return report_lines
305
+
306
+
307
+ def _re_rooted_lines(working_directory: Path, pr_identity: RepoIdentity) -> list[str]:
308
+ """Build the report lines for the re-rooted outcome.
309
+
310
+ Args:
311
+ working_directory: The session's current working directory.
312
+ pr_identity: The PR's owner/repo.
313
+
314
+ Returns:
315
+ A summary line and an abort line with the recovery instruction.
316
+ """
317
+ return [
318
+ SUMMARY_RE_ROOTED_TEMPLATE.format(cwd=working_directory),
319
+ ABORT_RE_ROOTED_TEMPLATE.format(owner=pr_identity.owner, repo=pr_identity.repo),
320
+ ]
321
+
322
+
323
+ def build_report_lines(
324
+ verdict: PreflightVerdict,
325
+ mode: str,
326
+ working_directory: Path,
327
+ pr_identity: RepoIdentity,
328
+ ) -> list[str]:
329
+ """Build the stdout report lines for a classification.
330
+
331
+ Args:
332
+ verdict: The classification of the working directory.
333
+ mode: MODE_STRICT (autoconverge) or MODE_CLASSIFY (pr-converge).
334
+ working_directory: The session's current working directory.
335
+ pr_identity: The PR's owner/repo.
336
+
337
+ Returns:
338
+ Lines to print: the machine-readable outcome marker first, then a
339
+ summary and a recovery or routing instruction when relevant.
340
+ """
341
+ report_lines = [OUTCOME_MARKER_TEMPLATE.format(outcome=verdict.outcome)]
342
+ if verdict.outcome == OUTCOME_SAME_REPO:
343
+ report_lines.extend(_same_repo_lines(verdict, working_directory, pr_identity))
344
+ elif verdict.outcome == OUTCOME_DIFFERENT_REPO:
345
+ report_lines.extend(_different_repo_lines(verdict, mode, pr_identity))
346
+ else:
347
+ report_lines.extend(_re_rooted_lines(working_directory, pr_identity))
348
+ return report_lines
349
+
350
+
351
+ def main(all_arguments: list[str]) -> int:
352
+ """Classify the working directory and report whether the run can continue.
353
+
354
+ Args:
355
+ all_arguments: The argument vector (sys.argv[1:] in normal use).
356
+
357
+ Returns:
358
+ The process exit code (EXIT_PREFLIGHT_OK or EXIT_PREFLIGHT_ABORT).
359
+ """
360
+ parser = argparse.ArgumentParser(description=PREFLIGHT_CLI_DESCRIPTION)
361
+ parser.add_argument(OWNER_ARG_FLAG, required=True)
362
+ parser.add_argument(REPO_ARG_FLAG, required=True)
363
+ parser.add_argument(MODE_ARG_FLAG, required=True, choices=ALL_PREFLIGHT_MODES)
364
+ parsed_arguments = parser.parse_args(all_arguments)
365
+ pr_identity = RepoIdentity(
366
+ owner=parsed_arguments.owner.lower(),
367
+ repo=parsed_arguments.repo.lower(),
368
+ )
369
+ working_directory = Path.cwd()
370
+ try:
371
+ verdict = classify_environment(working_directory, pr_identity)
372
+ except subprocess.TimeoutExpired:
373
+ print(
374
+ OUTCOME_MARKER_TEMPLATE.format(outcome=OUTCOME_RE_ROOTED),
375
+ file=sys.stdout,
376
+ )
377
+ print(
378
+ ABORT_RE_ROOTED_TEMPLATE.format(
379
+ owner=pr_identity.owner, repo=pr_identity.repo
380
+ ),
381
+ file=sys.stdout,
382
+ )
383
+ return EXIT_PREFLIGHT_ABORT
384
+ for each_report_line in build_report_lines(
385
+ verdict, parsed_arguments.mode, working_directory, pr_identity
386
+ ):
387
+ print(each_report_line, file=sys.stdout)
388
+ return decide_exit_code(verdict, parsed_arguments.mode)
389
+
390
+
391
+ if __name__ == "__main__":
392
+ raise SystemExit(main(sys.argv[1:]))
@@ -0,0 +1,70 @@
1
+ """Constants for the worktree/cwd preflight guard shared by pr-loop skills."""
2
+
3
+ import re
4
+
5
+ OUTCOME_SAME_REPO = "same_repo"
6
+ OUTCOME_DIFFERENT_REPO = "different_repo"
7
+ OUTCOME_RE_ROOTED = "re_rooted"
8
+
9
+ MODE_STRICT = "strict"
10
+ MODE_CLASSIFY = "classify"
11
+ ALL_PREFLIGHT_MODES = (MODE_STRICT, MODE_CLASSIFY)
12
+
13
+ EXIT_PREFLIGHT_OK = 0
14
+ EXIT_PREFLIGHT_ABORT = 1
15
+
16
+ GIT_EXECUTABLE = "git"
17
+ GIT_DIRECTORY_FLAG = "-C"
18
+ ALL_GIT_IS_INSIDE_WORK_TREE_ARGS = ("rev-parse", "--is-inside-work-tree")
19
+ ALL_GIT_REMOTE_GET_URL_ARGS = ("remote", "get-url", "origin")
20
+ ALL_GIT_WORKTREE_LIST_ARGS = ("worktree", "list")
21
+ GIT_INSIDE_WORK_TREE_TRUE = "true"
22
+ GIT_SUBPROCESS_TIMEOUT_SECONDS = 30
23
+
24
+ REMOTE_URL_IDENTITY_PATTERN = re.compile(
25
+ r"github\.com[:/](?P<owner>[^/]+)/(?P<repo>[^/]+?)(?:\.git)?/?$"
26
+ )
27
+
28
+ CWD_IDENTITY_UNKNOWN = "unknown"
29
+
30
+ OWNER_ARG_FLAG = "--owner"
31
+ REPO_ARG_FLAG = "--repo"
32
+ MODE_ARG_FLAG = "--mode"
33
+
34
+ PREFLIGHT_CLI_DESCRIPTION = (
35
+ "Verify the working directory and worktree before a PR-convergence run."
36
+ )
37
+
38
+ OUTCOME_MARKER_TEMPLATE = "PREFLIGHT_OUTCOME={outcome}"
39
+
40
+ SUMMARY_SAME_REPO_TEMPLATE = (
41
+ "OK: the working directory is the PR repo {owner}/{repo}; EnterWorktree can "
42
+ "create and enter the branch worktree."
43
+ )
44
+ SUMMARY_DIFFERENT_REPO_TEMPLATE = (
45
+ "The working directory repo {cwd_owner}/{cwd_repo} differs from the PR repo "
46
+ "{owner}/{repo}."
47
+ )
48
+ SUMMARY_RE_ROOTED_TEMPLATE = (
49
+ "The working directory {cwd} is not a usable checkout of a GitHub repo "
50
+ "(no git work tree, or no readable origin remote)."
51
+ )
52
+ ABORT_DIFFERENT_REPO_STRICT_TEMPLATE = (
53
+ "ABORT: autoconverge runs inside the PR's own repo, but this session is "
54
+ "rooted in {cwd_owner}/{cwd_repo}. Start the session from a checkout of "
55
+ "{owner}/{repo} and re-run."
56
+ )
57
+ ABORT_RE_ROOTED_TEMPLATE = (
58
+ "ABORT: this session is not rooted in a git checkout (a resumed or "
59
+ "background session can re-root to the home directory). Start the session "
60
+ "from a checkout of {owner}/{repo} and re-run."
61
+ )
62
+ ABORT_WORKTREE_BROKEN_TEMPLATE = (
63
+ "ABORT: the git worktree machinery in {cwd} is broken (git worktree list "
64
+ "failed), so EnterWorktree cannot create the branch worktree. Run "
65
+ "'git worktree prune' in {cwd} and re-run."
66
+ )
67
+ ROUTE_DIFFERENT_REPO_TEMPLATE = (
68
+ "ROUTE: resolve the PR worktree for {owner}/{repo} and cd into it before "
69
+ "any local work (cross-repo PR)."
70
+ )