Compare commits
2 Commits
fix/issue-
...
fd7a98b263
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fd7a98b263 | ||
|
|
d0b100fca8 |
@@ -5,7 +5,7 @@
|
|||||||
1. Create a branch for your work: `git checkout -b fix/issue-N-name` or `git checkout -b feat/issue-N-feature-name`
|
1. Create a branch for your work: `git checkout -b fix/issue-N-name` or `git checkout -b feat/issue-N-feature-name`
|
||||||
2. Make changes and commit with clear messages
|
2. Make changes and commit with clear messages
|
||||||
3. Open a Pull Request for review
|
3. Open a Pull Request for review
|
||||||
4. Do not merge directly to `main` or `develop` for reviewable changes
|
4. Do not merge directly to `main` for reviewable changes
|
||||||
5. After approval, squash and merge
|
5. After approval, squash and merge
|
||||||
|
|
||||||
## Guidelines
|
## Guidelines
|
||||||
@@ -14,53 +14,13 @@
|
|||||||
- Keep PRs focused and reasonably sized
|
- Keep PRs focused and reasonably sized
|
||||||
- Document any non-obvious decisions
|
- Document any non-obvious decisions
|
||||||
- Test changes before submitting
|
- Test changes before submitting
|
||||||
- See [VERSIONING.md](VERSIONING.md) for backport compatibility rules
|
|
||||||
|
|
||||||
## Branches
|
## Branches
|
||||||
|
|
||||||
### Primary Branches
|
- `main` — stable, reviewed content only
|
||||||
|
- `develop` — experimental work for 0.2.x
|
||||||
- `main` — stable 0.1.x releases, production-ready code
|
|
||||||
- `develop` — experimental 0.2.x work, next major version
|
|
||||||
|
|
||||||
### Feature Branches
|
|
||||||
|
|
||||||
- `fix/*` — bug fixes
|
- `fix/*` — bug fixes
|
||||||
- `feat/*` — new features
|
- `feat/*` — new features
|
||||||
- `docs/*` — documentation updates
|
- `docs/*` — documentation updates
|
||||||
- `refactor/*` — code refactoring (no behavior change)
|
- `refactor/*` — refactoring
|
||||||
|
- `research/*` — new research notes
|
||||||
## Branch Model
|
|
||||||
|
|
||||||
```
|
|
||||||
main (0.1.x stable)
|
|
||||||
└── v0.1.0, v0.1.1, v0.1.2, ...
|
|
||||||
|
|
||||||
develop (0.2.x experimental)
|
|
||||||
└── (next major version work)
|
|
||||||
```
|
|
||||||
|
|
||||||
### Which Branch to Target?
|
|
||||||
|
|
||||||
| Change Type | Target Branch | Backport? |
|
|
||||||
|-------------|---------------|-----------|
|
|
||||||
| Bug fix | `main` | N/A |
|
|
||||||
| Documentation | `main` | N/A |
|
|
||||||
| New feature (backport-compatible) | `main` | Can cherry-pick to `develop` |
|
|
||||||
| Experimental feature | `develop` | No |
|
|
||||||
| Breaking change | `develop` | No |
|
|
||||||
|
|
||||||
## Backport Compatibility
|
|
||||||
|
|
||||||
Before merging, consider if your change is backport-compatible:
|
|
||||||
|
|
||||||
- **YES**: Bug fixes, docs, adding new optional inputs
|
|
||||||
- **NO**: Changing behavior, changing defaults, removing features
|
|
||||||
|
|
||||||
See [VERSIONING.md](VERSIONING.md) for full policy.
|
|
||||||
|
|
||||||
## Release Process
|
|
||||||
|
|
||||||
1. Bug fixes and docs → directly to `main`
|
|
||||||
2. New features → `develop` or feature branches → `develop`
|
|
||||||
3. When `develop` is stable enough → merge to `main` for release
|
|
||||||
|
|||||||
@@ -1,71 +0,0 @@
|
|||||||
# Versioning Policy
|
|
||||||
|
|
||||||
## Branch Strategy
|
|
||||||
|
|
||||||
Kugetsu uses a dual-branch model:
|
|
||||||
|
|
||||||
| Branch | Purpose | Version | Stability |
|
|
||||||
|--------|---------|---------|-----------|
|
|
||||||
| `main` | Stable releases | 0.1.x | Production-ready |
|
|
||||||
| `develop` | Experimental work | 0.2.x | Active development |
|
|
||||||
|
|
||||||
### Branch Definitions
|
|
||||||
|
|
||||||
- **`main`**: Contains the latest stable 0.1.x releases. All changes here should be production-ready and backport-compatible when possible.
|
|
||||||
|
|
||||||
- **`develop`**: Contains work for the next major version (0.2.x). This branch may contain experimental features that could change or be removed.
|
|
||||||
|
|
||||||
## Version Format
|
|
||||||
|
|
||||||
Versions follow [Semantic Versioning](https://semver.org/):
|
|
||||||
```
|
|
||||||
MAJOR.MINOR.PATCH
|
|
||||||
```
|
|
||||||
|
|
||||||
- **MAJOR**: Incompatible API/behavior changes
|
|
||||||
- **MINOR**: New functionality (backward-compatible)
|
|
||||||
- **PATCH**: Bug fixes (backward-compatible)
|
|
||||||
|
|
||||||
## Backport Compatibility
|
|
||||||
|
|
||||||
### Backport-Compatible Changes (0.1.x)
|
|
||||||
- Bug fixes
|
|
||||||
- Documentation updates
|
|
||||||
- Performance improvements
|
|
||||||
- Adding new inputs/options (must have sensible defaults)
|
|
||||||
- Changes that only affect 0.2.x-specific features
|
|
||||||
|
|
||||||
### NOT Backport-Compatible
|
|
||||||
- Removing or renaming existing options
|
|
||||||
- Changing default values of existing options
|
|
||||||
- Changing behavior of existing commands
|
|
||||||
- Introducing breaking changes to the API/shell interface
|
|
||||||
|
|
||||||
## Deprecation Policy
|
|
||||||
|
|
||||||
When introducing breaking changes:
|
|
||||||
|
|
||||||
1. **Deprecate in minor X**: Add warning messages, document the change
|
|
||||||
2. **Remove in major X+1**: The breaking change is removed in the next major version
|
|
||||||
|
|
||||||
Example:
|
|
||||||
- Option `--old-flag` deprecated in v0.1.5
|
|
||||||
- Option `--old-flag` removed in v1.0.0 (not v0.2.0)
|
|
||||||
|
|
||||||
## What Constitutes a Version Bump
|
|
||||||
|
|
||||||
| Change Type | Version Bump |
|
|
||||||
|-------------|--------------|
|
|
||||||
| Add new command/option | MINOR |
|
|
||||||
| Bug fix | PATCH |
|
|
||||||
| Change default value | MINOR (may warrant PATCH) |
|
|
||||||
| Add new required input | MAJOR |
|
|
||||||
| Remove deprecated feature | MAJOR |
|
|
||||||
| Change behavior of existing command | MINOR (needs deprecation first) |
|
|
||||||
|
|
||||||
## Release Process
|
|
||||||
|
|
||||||
1. Changes are developed on feature branches
|
|
||||||
2. PRs are opened against `main` for 0.1.x changes, or `develop` for 0.2.x
|
|
||||||
3. After review and approval, changes are squash-merged
|
|
||||||
4. Releases are tagged from `main` after significant changes
|
|
||||||
@@ -1,111 +0,0 @@
|
|||||||
# Changelog
|
|
||||||
|
|
||||||
All notable changes to kugetsu are documented here.
|
|
||||||
|
|
||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
|
|
||||||
|
|
||||||
## [Unreleased]
|
|
||||||
|
|
||||||
## [v0.2.1] - 2026-04-03
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Prevent excess agent spawning with flock + sequential processing
|
|
||||||
|
|
||||||
## [v0.2.0] - 2026-03-30
|
|
||||||
|
|
||||||
### Added
|
|
||||||
- Queue system with background daemon
|
|
||||||
- Agent timeout handling
|
|
||||||
- Context dump/load for session isolation
|
|
||||||
- PR tracking and safe destroy
|
|
||||||
|
|
||||||
## [v0.1.13] - 2026-03-29
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Add missing closing parenthesis in process_queue Python extraction
|
|
||||||
|
|
||||||
## [v0.1.12] - 2026-03-25
|
|
||||||
|
|
||||||
### Added
|
|
||||||
- Post-comment helper for PM agent
|
|
||||||
|
|
||||||
## [v0.1.11] - 2026-03-20
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Wrap cmd_continue in subshell with cd for correct worktree dir
|
|
||||||
|
|
||||||
## [v0.1.10] - 2026-03-15
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- destroy --base now also deletes PM agent session
|
|
||||||
|
|
||||||
## [v0.1.9] - 2026-03-10
|
|
||||||
|
|
||||||
### Added
|
|
||||||
- init creates base session in ~/.kugetsu-worktrees
|
|
||||||
- Adds context to forked sessions
|
|
||||||
- Clears logs on init
|
|
||||||
|
|
||||||
## [v0.1.8] - 2026-03-05
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- destroy --base and --pm-agent actually delete opencode sessions
|
|
||||||
|
|
||||||
## [v0.1.7] - 2026-02-28
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Warn if init run from non-empty directory
|
|
||||||
|
|
||||||
## [v0.1.6] - 2026-02-20
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Detect session via DB query instead of opencode session list
|
|
||||||
|
|
||||||
## [v0.1.5] - 2026-02-15
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Update forked session permissions after detection
|
|
||||||
|
|
||||||
## [v0.1.4] - 2026-02-10
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Call fix_session_permissions before forking
|
|
||||||
|
|
||||||
## [v0.1.3] - 2026-02-05
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Session detection ordering bug and debugging
|
|
||||||
|
|
||||||
## [v0.1.2] - 2026-01-28
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Improve session detection in cmd_start with retry logic and logging
|
|
||||||
|
|
||||||
## [v0.1.1] - 2026-01-20
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Use cd + worktree inside parent dir instead of --dir flag
|
|
||||||
|
|
||||||
## [v0.1.0] - 2026-01-15
|
|
||||||
|
|
||||||
### Added
|
|
||||||
- KUGETSU_VERBOSITY for PM agent output control
|
|
||||||
- Initial documented release
|
|
||||||
|
|
||||||
[Unreleased]: https://git.fbrns.co/shoko/kugetsu/compare/v0.2.1...HEAD
|
|
||||||
[v0.2.1]: https://git.fbrns.co/shoko/kugetsu/compare/v0.2.0...v0.2.1
|
|
||||||
[v0.2.0]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.13...v0.2.0
|
|
||||||
[v0.1.13]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.12...v0.1.13
|
|
||||||
[v0.1.12]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.11...v0.1.12
|
|
||||||
[v0.1.11]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.10...v0.1.11
|
|
||||||
[v0.1.10]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.9...v0.1.10
|
|
||||||
[v0.1.9]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.8...v0.1.9
|
|
||||||
[v0.1.8]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.7...v0.1.8
|
|
||||||
[v0.1.7]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.6...v0.1.7
|
|
||||||
[v0.1.6]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.5...v0.1.6
|
|
||||||
[v0.1.5]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.4...v0.1.5
|
|
||||||
[v0.1.4]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.3...v0.1.4
|
|
||||||
[v0.1.3]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.2...v0.1.3
|
|
||||||
[v0.1.2]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.1...v0.1.2
|
|
||||||
[v0.1.1]: https://git.fbrns.co/shoko/kugetsu/compare/v0.1.0...v0.1.1
|
|
||||||
[v0.1.0]: https://git.fbrns.co/shoko/kugetsu/releases/tag/v0.1.0
|
|
||||||
@@ -1075,6 +1075,18 @@ cmd_status() {
|
|||||||
return
|
return
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
local opencode_sessions=$(opencode session list 2>/dev/null | grep -oP '^ses_\w+' || true)
|
||||||
|
|
||||||
|
if ! echo "$opencode_sessions" | grep -q "^${base}$"; then
|
||||||
|
echo "error: base session '$base' not found in opencode"
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! echo "$opencode_sessions" | grep -q "^${pm_agent}$"; then
|
||||||
|
echo "error: pm_agent session '$pm_agent' not found in opencode"
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
|
||||||
echo "ok"
|
echo "ok"
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1145,6 +1157,11 @@ parse_issue_ref_from_message() {
|
|||||||
owner=$(echo "$full_path" | cut -d'/' -f2)
|
owner=$(echo "$full_path" | cut -d'/' -f2)
|
||||||
repo=$(echo "$full_path" | cut -d'/' -f3)
|
repo=$(echo "$full_path" | cut -d'/' -f3)
|
||||||
issue_number=$(echo "$full_path" | grep -oE '[0-9]+$' | head -1)
|
issue_number=$(echo "$full_path" | grep -oE '[0-9]+$' | head -1)
|
||||||
|
elif echo "$message" | grep -qE '[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#[0-9]+'; then
|
||||||
|
gitserver=$(echo "$message" | grep -oE '[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+' | head -1)
|
||||||
|
owner=$(echo "$gitserver" | cut -d'/' -f2)
|
||||||
|
repo=$(echo "$gitserver" | cut -d'/' -f3)
|
||||||
|
issue_number=$(echo "$message" | grep -oE '#[0-9]+' | grep -oE '[0-9]+' | head -1)
|
||||||
elif echo "$message" | grep -qE '[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#([0-9]+)'; then
|
elif echo "$message" | grep -qE '[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#([0-9]+)'; then
|
||||||
owner=$(echo "$message" | grep -oE '[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#' | sed 's/#$//' | cut -d'/' -f1)
|
owner=$(echo "$message" | grep -oE '[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#' | sed 's/#$//' | cut -d'/' -f1)
|
||||||
repo=$(echo "$message" | grep -oE '[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#' | sed 's/#$//' | cut -d'/' -f2)
|
repo=$(echo "$message" | grep -oE '[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+#' | sed 's/#$//' | cut -d'/' -f2)
|
||||||
|
|||||||
@@ -7,6 +7,8 @@
|
|||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
KUGETSU="./skills/kugetsu/scripts/kugetsu"
|
KUGETSU="./skills/kugetsu/scripts/kugetsu"
|
||||||
|
TEST_KUGETSU_DIR="/tmp/test-kugetsu-$$"
|
||||||
|
export KUGETSU_DIR="$TEST_KUGETSU_DIR"
|
||||||
TEST_ISSUE_REF="github.com/shoko/kugetsu#14"
|
TEST_ISSUE_REF="github.com/shoko/kugetsu#14"
|
||||||
TEST_DISCUSS_REF="github.com/shoko/kugetsu#-discuss"
|
TEST_DISCUSS_REF="github.com/shoko/kugetsu#-discuss"
|
||||||
TEST_BASE_SESSION_ID="ses_test_base_123"
|
TEST_BASE_SESSION_ID="ses_test_base_123"
|
||||||
@@ -18,28 +20,28 @@ PASS=0
|
|||||||
FAIL=0
|
FAIL=0
|
||||||
|
|
||||||
cleanup() {
|
cleanup() {
|
||||||
rm -rf ~/.kugetsu/sessions/* ~/.kugetsu/worktrees/* ~/.kugetsu/index.json 2>/dev/null || true
|
rm -rf "$TEST_KUGETSU_DIR" 2>/dev/null || true
|
||||||
}
|
}
|
||||||
|
|
||||||
setup_mock_base() {
|
setup_mock_base() {
|
||||||
mkdir -p ~/.kugetsu/sessions ~/.kugetsu/worktrees
|
mkdir -p "$TEST_KUGETSU_DIR/sessions" "$TEST_KUGETSU_DIR/worktrees"
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > "$TEST_KUGETSU_DIR/index.json" << EOF
|
||||||
{
|
{
|
||||||
"base": "$TEST_BASE_SESSION_ID",
|
"base": "$TEST_BASE_SESSION_ID",
|
||||||
"pm_agent": "$TEST_PM_AGENT_SESSION_ID",
|
"pm_agent": "$TEST_PM_AGENT_SESSION_ID",
|
||||||
"issues": {}
|
"issues": {}
|
||||||
}
|
}
|
||||||
EOF
|
EOF
|
||||||
cat > ~/.kugetsu/sessions/$TEST_BASE_SESSION_FILE << EOF
|
cat > "$TEST_KUGETSU_DIR/sessions/$TEST_BASE_SESSION_FILE" << EOF
|
||||||
{"type": "base", "opencode_session_id": "$TEST_BASE_SESSION_ID", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
{"type": "base", "opencode_session_id": "$TEST_BASE_SESSION_ID", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
||||||
EOF
|
EOF
|
||||||
cat > ~/.kugetsu/sessions/$TEST_PM_AGENT_SESSION_FILE << EOF
|
cat > "$TEST_KUGETSU_DIR/sessions/$TEST_PM_AGENT_SESSION_FILE" << EOF
|
||||||
{"type": "pm_agent", "opencode_session_id": "$TEST_PM_AGENT_SESSION_ID", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
{"type": "pm_agent", "opencode_session_id": "$TEST_PM_AGENT_SESSION_ID", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
||||||
EOF
|
EOF
|
||||||
}
|
}
|
||||||
|
|
||||||
setup_mock_forked() {
|
setup_mock_forked() {
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > "$TEST_KUGETSU_DIR/index.json" << EOF
|
||||||
{
|
{
|
||||||
"base": "$TEST_BASE_SESSION_ID",
|
"base": "$TEST_BASE_SESSION_ID",
|
||||||
"pm_agent": "$TEST_PM_AGENT_SESSION_ID",
|
"pm_agent": "$TEST_PM_AGENT_SESSION_ID",
|
||||||
@@ -48,7 +50,7 @@ setup_mock_forked() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
EOF
|
EOF
|
||||||
cat > ~/.kugetsu/sessions/$TEST_FORKED_SESSION_FILE << EOF
|
cat > "$TEST_KUGETSU_DIR/sessions/$TEST_FORKED_SESSION_FILE" << EOF
|
||||||
{"type": "forked", "issue_ref": "$TEST_ISSUE_REF", "opencode_session_id": "ses_forked_789", "worktree_path": "/tmp/test-worktree", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
{"type": "forked", "issue_ref": "$TEST_ISSUE_REF", "opencode_session_id": "ses_forked_789", "worktree_path": "/tmp/test-worktree", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
||||||
EOF
|
EOF
|
||||||
}
|
}
|
||||||
@@ -112,16 +114,16 @@ echo ""
|
|||||||
|
|
||||||
# Test 3b: start fails without pm-agent
|
# Test 3b: start fails without pm-agent
|
||||||
echo "--- Test: start without pm-agent session ---"
|
echo "--- Test: start without pm-agent session ---"
|
||||||
rm -f ~/.kugetsu/index.json ~/.kugetsu/sessions/*
|
rm -f $TEST_KUGETSU_DIR/index.json $TEST_KUGETSU_DIR/sessions/*
|
||||||
mkdir -p ~/.kugetsu/sessions
|
mkdir -p $TEST_KUGETSU_DIR/sessions
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > $TEST_KUGETSU_DIR/index.json << EOF
|
||||||
{
|
{
|
||||||
"base": "$TEST_BASE_SESSION_ID",
|
"base": "$TEST_BASE_SESSION_ID",
|
||||||
"pm_agent": null,
|
"pm_agent": null,
|
||||||
"issues": {}
|
"issues": {}
|
||||||
}
|
}
|
||||||
EOF
|
EOF
|
||||||
cat > ~/.kugetsu/sessions/$TEST_BASE_SESSION_FILE << EOF
|
cat > $TEST_KUGETSU_DIR/sessions/$TEST_BASE_SESSION_FILE << EOF
|
||||||
{"type": "base", "opencode_session_id": "$TEST_BASE_SESSION_ID", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
{"type": "base", "opencode_session_id": "$TEST_BASE_SESSION_ID", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}
|
||||||
EOF
|
EOF
|
||||||
OUTPUT=$($KUGETSU start github.com/shoko/kugetsu#14 "test" 2>&1 || true)
|
OUTPUT=$($KUGETSU start github.com/shoko/kugetsu#14 "test" 2>&1 || true)
|
||||||
@@ -176,7 +178,7 @@ echo ""
|
|||||||
|
|
||||||
# Test 6c: index.json has pm_agent field
|
# Test 6c: index.json has pm_agent field
|
||||||
echo "--- Test: index.json has pm_agent field ---"
|
echo "--- Test: index.json has pm_agent field ---"
|
||||||
if grep -q '"pm_agent"' ~/.kugetsu/index.json; then
|
if grep -q '"pm_agent"' $TEST_KUGETSU_DIR/index.json; then
|
||||||
pass "index.json has pm_agent field"
|
pass "index.json has pm_agent field"
|
||||||
else
|
else
|
||||||
fail "index.json missing pm_agent field"
|
fail "index.json missing pm_agent field"
|
||||||
@@ -227,12 +229,12 @@ echo ""
|
|||||||
echo "--- Test: destroy --pm-agent -y ---"
|
echo "--- Test: destroy --pm-agent -y ---"
|
||||||
setup_mock_base
|
setup_mock_base
|
||||||
OUTPUT=$($KUGETSU destroy --pm-agent -y 2>&1 || true)
|
OUTPUT=$($KUGETSU destroy --pm-agent -y 2>&1 || true)
|
||||||
if [ -f ~/.kugetsu/sessions/$TEST_PM_AGENT_SESSION_FILE ]; then
|
if [ -f $TEST_KUGETSU_DIR/sessions/$TEST_PM_AGENT_SESSION_FILE ]; then
|
||||||
fail "destroy --pm-agent -y removes pm-agent file"
|
fail "destroy --pm-agent -y removes pm-agent file"
|
||||||
else
|
else
|
||||||
pass "destroy --pm-agent -y removes pm-agent file"
|
pass "destroy --pm-agent -y removes pm-agent file"
|
||||||
fi
|
fi
|
||||||
if grep -q '"pm_agent": null' ~/.kugetsu/index.json; then
|
if grep -q '"pm_agent": null' $TEST_KUGETSU_DIR/index.json; then
|
||||||
pass "destroy --pm-agent -y sets pm_agent to null in index"
|
pass "destroy --pm-agent -y sets pm_agent to null in index"
|
||||||
else
|
else
|
||||||
fail "destroy --pm-agent -y should set pm_agent to null"
|
fail "destroy --pm-agent -y should set pm_agent to null"
|
||||||
@@ -243,7 +245,7 @@ echo ""
|
|||||||
echo "--- Test: destroy --base -y ---"
|
echo "--- Test: destroy --base -y ---"
|
||||||
setup_mock_base
|
setup_mock_base
|
||||||
OUTPUT=$($KUGETSU destroy --base -y 2>&1 || true)
|
OUTPUT=$($KUGETSU destroy --base -y 2>&1 || true)
|
||||||
if [ -f ~/.kugetsu/sessions/$TEST_BASE_SESSION_FILE ]; then
|
if [ -f $TEST_KUGETSU_DIR/sessions/$TEST_BASE_SESSION_FILE ]; then
|
||||||
fail "destroy --base -y removes base file"
|
fail "destroy --base -y removes base file"
|
||||||
else
|
else
|
||||||
pass "destroy --base -y removes base file"
|
pass "destroy --base -y removes base file"
|
||||||
@@ -292,7 +294,7 @@ echo ""
|
|||||||
|
|
||||||
# Test 15: worktree path in session file
|
# Test 15: worktree path in session file
|
||||||
echo "--- Test: worktree_path in session file ---"
|
echo "--- Test: worktree_path in session file ---"
|
||||||
if grep -q "worktree_path" ~/.kugetsu/sessions/$TEST_FORKED_SESSION_FILE; then
|
if grep -q "worktree_path" $TEST_KUGETSU_DIR/sessions/$TEST_FORKED_SESSION_FILE; then
|
||||||
pass "session file contains worktree_path"
|
pass "session file contains worktree_path"
|
||||||
else
|
else
|
||||||
fail "session file missing worktree_path"
|
fail "session file missing worktree_path"
|
||||||
@@ -303,7 +305,7 @@ echo ""
|
|||||||
echo "--- Test: prune with orphaned worktree ---"
|
echo "--- Test: prune with orphaned worktree ---"
|
||||||
cleanup
|
cleanup
|
||||||
setup_mock_base
|
setup_mock_base
|
||||||
mkdir -p ~/.kugetsu/worktrees/orphaned-worktree
|
mkdir -p $TEST_KUGETSU_DIR/worktrees/orphaned-worktree
|
||||||
OUTPUT=$($KUGETSU prune 2>&1 || true)
|
OUTPUT=$($KUGETSU prune 2>&1 || true)
|
||||||
if echo "$OUTPUT" | grep -q "orphaned worktree"; then
|
if echo "$OUTPUT" | grep -q "orphaned worktree"; then
|
||||||
pass "prune detects orphaned worktree"
|
pass "prune detects orphaned worktree"
|
||||||
@@ -315,7 +317,7 @@ echo ""
|
|||||||
# Test 17: prune --force removes orphaned worktrees
|
# Test 17: prune --force removes orphaned worktrees
|
||||||
echo "--- Test: prune --force removes orphaned worktrees ---"
|
echo "--- Test: prune --force removes orphaned worktrees ---"
|
||||||
OUTPUT=$($KUGETSU prune --force 2>&1 || true)
|
OUTPUT=$($KUGETSU prune --force 2>&1 || true)
|
||||||
if [ -d ~/.kugetsu/worktrees/orphaned-worktree ]; then
|
if [ -d $TEST_KUGETSU_DIR/worktrees/orphaned-worktree ]; then
|
||||||
fail "prune --force should remove orphaned worktree"
|
fail "prune --force should remove orphaned worktree"
|
||||||
else
|
else
|
||||||
pass "prune --force removes orphaned worktree"
|
pass "prune --force removes orphaned worktree"
|
||||||
@@ -332,10 +334,10 @@ echo ""
|
|||||||
echo "--- Test: destroy removes worktree ---"
|
echo "--- Test: destroy removes worktree ---"
|
||||||
cleanup
|
cleanup
|
||||||
setup_mock_forked
|
setup_mock_forked
|
||||||
# remove_worktree_for_issue derives path from issue ref: ~/.kugetsu/worktrees/github.com-shoko-kugetsu-14
|
# remove_worktree_for_issue derives path from issue ref: $TEST_KUGETSU_DIR/worktrees/github.com-shoko-kugetsu-14
|
||||||
mkdir -p ~/.kugetsu/worktrees/github.com-shoko-kugetsu-14
|
mkdir -p $TEST_KUGETSU_DIR/worktrees/github.com-shoko-kugetsu-14
|
||||||
OUTPUT=$($KUGETSU destroy github.com/shoko/kugetsu#14 -y 2>&1 || true)
|
OUTPUT=$($KUGETSU destroy github.com/shoko/kugetsu#14 -y 2>&1 || true)
|
||||||
if [ -d ~/.kugetsu/worktrees/github.com-shoko-kugetsu-14 ]; then
|
if [ -d $TEST_KUGETSU_DIR/worktrees/github.com-shoko-kugetsu-14 ]; then
|
||||||
fail "destroy should remove worktree"
|
fail "destroy should remove worktree"
|
||||||
else
|
else
|
||||||
pass "destroy removes worktree"
|
pass "destroy removes worktree"
|
||||||
@@ -345,7 +347,7 @@ echo ""
|
|||||||
# Test 20: session file properly formatted for v2.2
|
# Test 20: session file properly formatted for v2.2
|
||||||
echo "--- Test: session file format v2.2 ---"
|
echo "--- Test: session file format v2.2 ---"
|
||||||
setup_mock_forked
|
setup_mock_forked
|
||||||
SESSION_CONTENT=$(cat ~/.kugetsu/sessions/$TEST_FORKED_SESSION_FILE)
|
SESSION_CONTENT=$(cat $TEST_KUGETSU_DIR/sessions/$TEST_FORKED_SESSION_FILE)
|
||||||
if echo "$SESSION_CONTENT" | grep -q '"type": "forked"' && \
|
if echo "$SESSION_CONTENT" | grep -q '"type": "forked"' && \
|
||||||
echo "$SESSION_CONTENT" | grep -q '"worktree_path"'; then
|
echo "$SESSION_CONTENT" | grep -q '"worktree_path"'; then
|
||||||
pass "session file has v2.2 format"
|
pass "session file has v2.2 format"
|
||||||
@@ -367,8 +369,8 @@ echo ""
|
|||||||
|
|
||||||
# Test 22: status when base missing
|
# Test 22: status when base missing
|
||||||
echo "--- Test: status (base missing) ---"
|
echo "--- Test: status (base missing) ---"
|
||||||
mkdir -p ~/.kugetsu/sessions
|
mkdir -p $TEST_KUGETSU_DIR/sessions
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > $TEST_KUGETSU_DIR/index.json << EOF
|
||||||
{
|
{
|
||||||
"base": null,
|
"base": null,
|
||||||
"pm_agent": "$TEST_PM_AGENT_SESSION_ID",
|
"pm_agent": "$TEST_PM_AGENT_SESSION_ID",
|
||||||
@@ -385,7 +387,7 @@ echo ""
|
|||||||
|
|
||||||
# Test 23: status when pm-agent missing
|
# Test 23: status when pm-agent missing
|
||||||
echo "--- Test: status (pm-agent missing) ---"
|
echo "--- Test: status (pm-agent missing) ---"
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > $TEST_KUGETSU_DIR/index.json << EOF
|
||||||
{
|
{
|
||||||
"base": "$TEST_BASE_SESSION_ID",
|
"base": "$TEST_BASE_SESSION_ID",
|
||||||
"pm_agent": null,
|
"pm_agent": null,
|
||||||
@@ -402,7 +404,7 @@ echo ""
|
|||||||
|
|
||||||
# Test 24: status when pm-agent is "None" (Python None output)
|
# Test 24: status when pm-agent is "None" (Python None output)
|
||||||
echo "--- Test: status (pm-agent is Python None) ---"
|
echo "--- Test: status (pm-agent is Python None) ---"
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > $TEST_KUGETSU_DIR/index.json << EOF
|
||||||
{
|
{
|
||||||
"base": "$TEST_BASE_SESSION_ID",
|
"base": "$TEST_BASE_SESSION_ID",
|
||||||
"pm_agent": "None",
|
"pm_agent": "None",
|
||||||
@@ -445,8 +447,8 @@ echo ""
|
|||||||
# Test 27: delegate when pm-agent missing
|
# Test 27: delegate when pm-agent missing
|
||||||
echo "--- Test: delegate (pm-agent missing) ---"
|
echo "--- Test: delegate (pm-agent missing) ---"
|
||||||
cleanup
|
cleanup
|
||||||
mkdir -p ~/.kugetsu/sessions ~/.kugetsu/worktrees
|
mkdir -p $TEST_KUGETSU_DIR/sessions $TEST_KUGETSU_DIR/worktrees
|
||||||
cat > ~/.kugetsu/index.json << EOF
|
cat > $TEST_KUGETSU_DIR/index.json << EOF
|
||||||
{
|
{
|
||||||
"base": "$TEST_BASE_SESSION_ID",
|
"base": "$TEST_BASE_SESSION_ID",
|
||||||
"pm_agent": null,
|
"pm_agent": null,
|
||||||
@@ -508,7 +510,7 @@ echo ""
|
|||||||
# Test 32: delegate is fire-and-forget (returns immediately)
|
# Test 32: delegate is fire-and-forget (returns immediately)
|
||||||
echo "--- Test: delegate is fire-and-forget ---"
|
echo "--- Test: delegate is fire-and-forget ---"
|
||||||
setup_mock_base
|
setup_mock_base
|
||||||
mkdir -p ~/.kugetsu/logs
|
mkdir -p $TEST_KUGETSU_DIR/logs
|
||||||
START=$(date +%s)
|
START=$(date +%s)
|
||||||
OUTPUT=$($KUGETSU delegate "test fire-and-forget" 2>&1 || true)
|
OUTPUT=$($KUGETSU delegate "test fire-and-forget" 2>&1 || true)
|
||||||
END=$(date +%s)
|
END=$(date +%s)
|
||||||
@@ -527,10 +529,10 @@ echo ""
|
|||||||
# Test 33: delegate creates log file
|
# Test 33: delegate creates log file
|
||||||
echo "--- Test: delegate creates log file ---"
|
echo "--- Test: delegate creates log file ---"
|
||||||
setup_mock_base
|
setup_mock_base
|
||||||
LOG_COUNT_BEFORE=$(ls ~/.kugetsu/logs/*.log 2>/dev/null | wc -l)
|
LOG_COUNT_BEFORE=$(ls $TEST_KUGETSU_DIR/logs/*.log 2>/dev/null | wc -l)
|
||||||
$KUGETSU delegate "test log file" 2>&1 || true
|
$KUGETSU delegate "test log file" 2>&1 || true
|
||||||
sleep 1
|
sleep 1
|
||||||
LOG_COUNT_AFTER=$(ls ~/.kugetsu/logs/*.log 2>/dev/null | wc -l)
|
LOG_COUNT_AFTER=$(ls $TEST_KUGETSU_DIR/logs/*.log 2>/dev/null | wc -l)
|
||||||
if [ $LOG_COUNT_AFTER -gt $LOG_COUNT_BEFORE ]; then
|
if [ $LOG_COUNT_AFTER -gt $LOG_COUNT_BEFORE ]; then
|
||||||
pass "delegate creates log file"
|
pass "delegate creates log file"
|
||||||
else
|
else
|
||||||
@@ -558,10 +560,10 @@ echo ""
|
|||||||
|
|
||||||
# Test E2: env set creates file
|
# Test E2: env set creates file
|
||||||
echo "--- Test: env set creates env file ---"
|
echo "--- Test: env set creates env file ---"
|
||||||
mkdir -p ~/.kugetsu/env
|
mkdir -p $TEST_KUGETSU_DIR/env
|
||||||
rm -f ~/.kugetsu/env/pm-agent.env
|
rm -f $TEST_KUGETSU_DIR/env/pm-agent.env
|
||||||
$KUGETSU env set TEST_VAR "test_value" pm-agent 2>&1 || true
|
$KUGETSU env set TEST_VAR "test_value" pm-agent 2>&1 || true
|
||||||
if [ -f ~/.kugetsu/env/pm-agent.env ]; then
|
if [ -f $TEST_KUGETSU_DIR/env/pm-agent.env ]; then
|
||||||
pass "env set creates pm-agent.env file"
|
pass "env set creates pm-agent.env file"
|
||||||
else
|
else
|
||||||
fail "env set did not create pm-agent.env"
|
fail "env set did not create pm-agent.env"
|
||||||
@@ -570,7 +572,7 @@ echo ""
|
|||||||
|
|
||||||
# Test E3: env show masks sensitive values
|
# Test E3: env show masks sensitive values
|
||||||
echo "--- Test: env show masks sensitive values ---"
|
echo "--- Test: env show masks sensitive values ---"
|
||||||
cat > ~/.kugetsu/env/pm-agent.env << 'ENVEOF'
|
cat > $TEST_KUGETSU_DIR/env/pm-agent.env << 'ENVEOF'
|
||||||
export GITEA_TOKEN="secret_token_123"
|
export GITEA_TOKEN="secret_token_123"
|
||||||
export MY_VAR="visible_value"
|
export MY_VAR="visible_value"
|
||||||
ENVEOF
|
ENVEOF
|
||||||
@@ -584,14 +586,14 @@ echo ""
|
|||||||
|
|
||||||
# Test E4: Variables exported to child processes via set -a
|
# Test E4: Variables exported to child processes via set -a
|
||||||
echo "--- Test: set -a exports variables to children ---"
|
echo "--- Test: set -a exports variables to children ---"
|
||||||
mkdir -p ~/.kugetsu/env
|
mkdir -p $TEST_KUGETSU_DIR/env
|
||||||
cat > ~/.kugetsu/env/test.env << 'ENVEOF'
|
cat > $TEST_KUGETSU_DIR/env/test.env << 'ENVEOF'
|
||||||
export EXPORT_TEST="exported_value"
|
export EXPORT_TEST="exported_value"
|
||||||
SIMPLE_TEST="not_exported"
|
SIMPLE_TEST="not_exported"
|
||||||
ENVEOF
|
ENVEOF
|
||||||
|
|
||||||
# Simulate what cmd_delegate does
|
# Simulate what cmd_delegate does
|
||||||
ENV_FILE="~/.kugetsu/env/test.env"
|
ENV_FILE="$TEST_KUGETSU_DIR/env/test.env"
|
||||||
env_sh="set -a; source '$ENV_FILE'; set +a; "
|
env_sh="set -a; source '$ENV_FILE'; set +a; "
|
||||||
result=$(bash -c "${env_sh}bash -c 'echo \$EXPORT_TEST'")
|
result=$(bash -c "${env_sh}bash -c 'echo \$EXPORT_TEST'")
|
||||||
|
|
||||||
@@ -604,11 +606,11 @@ echo ""
|
|||||||
|
|
||||||
# Test E5: pm-agent.env takes precedence
|
# Test E5: pm-agent.env takes precedence
|
||||||
echo "--- Test: pm-agent.env takes precedence over default ---"
|
echo "--- Test: pm-agent.env takes precedence over default ---"
|
||||||
mkdir -p ~/.kugetsu/env
|
mkdir -p $TEST_KUGETSU_DIR/env
|
||||||
cat > ~/.kugetsu/env/default.env << 'ENVEOF'
|
cat > $TEST_KUGETSU_DIR/env/default.env << 'ENVEOF'
|
||||||
export GITEA_TOKEN="default_token"
|
export GITEA_TOKEN="default_token"
|
||||||
ENVEOF
|
ENVEOF
|
||||||
cat > ~/.kugetsu/env/pm-agent.env << 'ENVEOF'
|
cat > $TEST_KUGETSU_DIR/env/pm-agent.env << 'ENVEOF'
|
||||||
export GITEA_TOKEN="pm_agent_token"
|
export GITEA_TOKEN="pm_agent_token"
|
||||||
ENVEOF
|
ENVEOF
|
||||||
|
|
||||||
@@ -644,7 +646,7 @@ fi
|
|||||||
echo ""
|
echo ""
|
||||||
|
|
||||||
# Cleanup env files
|
# Cleanup env files
|
||||||
rm -rf ~/.kugetsu/env 2>/dev/null || true
|
rm -rf $TEST_KUGETSU_DIR/env 2>/dev/null || true
|
||||||
|
|
||||||
# Test E7: fix_session_permissions function exists
|
# Test E7: fix_session_permissions function exists
|
||||||
echo "--- Test: fix_session_permissions function exists ---"
|
echo "--- Test: fix_session_permissions function exists ---"
|
||||||
@@ -736,7 +738,7 @@ PASS=0
|
|||||||
FAIL=0
|
FAIL=0
|
||||||
|
|
||||||
test_cleanup() {
|
test_cleanup() {
|
||||||
rm -rf ~/.kugetsu/sessions/* ~/.kugetsu/worktrees/* ~/.kugetsu/index.json ~/.kugetsu/logs/* ~/.kugetsu/.agent_count ~/.kugetsu/.agent_lock 2>/dev/null || true
|
rm -rf $TEST_KUGETSU_DIR/sessions/* $TEST_KUGETSU_DIR/worktrees/* $TEST_KUGETSU_DIR/index.json $TEST_KUGETSU_DIR/logs/* $TEST_KUGETSU_DIR/.agent_count $TEST_KUGETSU_DIR/.agent_lock 2>/dev/null || true
|
||||||
}
|
}
|
||||||
|
|
||||||
pass() {
|
pass() {
|
||||||
@@ -750,25 +752,25 @@ fail() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
setup_mock_sessions() {
|
setup_mock_sessions() {
|
||||||
mkdir -p ~/.kugetsu/sessions ~/.kugetsu/worktrees ~/.kugetsu/logs
|
mkdir -p $TEST_KUGETSU_DIR/sessions $TEST_KUGETSU_DIR/worktrees $TEST_KUGETSU_DIR/logs
|
||||||
cat > ~/.kugetsu/index.json << INDEX
|
cat > $TEST_KUGETSU_DIR/index.json << INDEX
|
||||||
{
|
{
|
||||||
"base": "ses_test_base_123",
|
"base": "ses_test_base_123",
|
||||||
"pm_agent": "ses_test_pm_456",
|
"pm_agent": "ses_test_pm_456",
|
||||||
"issues": {}
|
"issues": {}
|
||||||
}
|
}
|
||||||
INDEX
|
INDEX
|
||||||
echo '{"type": "base", "opencode_session_id": "ses_test_base_123", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}' > ~/.kugetsu/sessions/base.json
|
echo '{"type": "base", "opencode_session_id": "ses_test_base_123", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}' > $TEST_KUGETSU_DIR/sessions/base.json
|
||||||
echo '{"type": "pm_agent", "opencode_session_id": "ses_test_pm_456", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}' > ~/.kugetsu/sessions/pm-agent.json
|
echo '{"type": "pm_agent", "opencode_session_id": "ses_test_pm_456", "created_at": "2026-03-29T18:00:00+02:00", "state": "idle"}' > $TEST_KUGETSU_DIR/sessions/pm-agent.json
|
||||||
}
|
}
|
||||||
|
|
||||||
# Test C1: Agent count file is initialized to 0
|
# Test C1: Agent count file is initialized to 0
|
||||||
echo "--- Test: agent count file initialized ---"
|
echo "--- Test: agent count file initialized ---"
|
||||||
test_cleanup
|
test_cleanup
|
||||||
mkdir -p ~/.kugetsu/sessions ~/.kugetsu/worktrees
|
mkdir -p $TEST_KUGETSU_DIR/sessions $TEST_KUGETSU_DIR/worktrees
|
||||||
$KUGETSU list > /dev/null 2>&1 || true
|
$KUGETSU list > /dev/null 2>&1 || true
|
||||||
if [ -f ~/.kugetsu/.agent_count ]; then
|
if [ -f $TEST_KUGETSU_DIR/.agent_count ]; then
|
||||||
COUNT=$(cat ~/.kugetsu/.agent_count)
|
COUNT=$(cat $TEST_KUGETSU_DIR/.agent_count)
|
||||||
if [ "$COUNT" = "0" ]; then
|
if [ "$COUNT" = "0" ]; then
|
||||||
pass "agent count file initialized to 0"
|
pass "agent count file initialized to 0"
|
||||||
else
|
else
|
||||||
@@ -795,10 +797,10 @@ test_cleanup
|
|||||||
setup_mock_sessions
|
setup_mock_sessions
|
||||||
|
|
||||||
# Initialize count to 0
|
# Initialize count to 0
|
||||||
echo 0 > ~/.kugetsu/.agent_count
|
echo 0 > $TEST_KUGETSU_DIR/.agent_count
|
||||||
|
|
||||||
# Verify initial state
|
# Verify initial state
|
||||||
INITIAL=$(cat ~/.kugetsu/.agent_count)
|
INITIAL=$(cat $TEST_KUGETSU_DIR/.agent_count)
|
||||||
if [ "$INITIAL" = "0" ]; then
|
if [ "$INITIAL" = "0" ]; then
|
||||||
pass "agent count starts at 0"
|
pass "agent count starts at 0"
|
||||||
else
|
else
|
||||||
@@ -809,7 +811,7 @@ fi
|
|||||||
$KUGETSU list > /dev/null 2>&1
|
$KUGETSU list > /dev/null 2>&1
|
||||||
|
|
||||||
# Verify count is still 0 (no slot leak)
|
# Verify count is still 0 (no slot leak)
|
||||||
AFTER=$(cat ~/.kugetsu/.agent_count)
|
AFTER=$(cat $TEST_KUGETSU_DIR/.agent_count)
|
||||||
if [ "$AFTER" = "0" ]; then
|
if [ "$AFTER" = "0" ]; then
|
||||||
pass "agent count stays 0 after list (no leak)"
|
pass "agent count stays 0 after list (no leak)"
|
||||||
else
|
else
|
||||||
|
|||||||
Reference in New Issue
Block a user