Compare commits

...

3 Commits

Author SHA1 Message Date
Rodin 04b24256c0 chore: dev-loop status update — PR #140 merged, coverage now 49.3%
CI / test (push) Successful in 16s
CI / review (anthropic--claude-4.6-sonnet, sonnet, SONNET_REVIEW_TOKEN) (push) Has been skipped
CI / review (gpt-5, gpt, GPT_REVIEW_TOKEN) (push) Has been skipped
CI / review (gpt-5, security, ., rodin/security-patterns, SECURITY_REVIEW.md, SECURITY_REVIEW_TOKEN) (push) Has been skipped
2026-05-15 04:36:09 +00:00
rodin 1a4bab8ddc test(#139): improve cmd/review-bot coverage from 44.6% to 49.3% (#140)
CI / test (push) Successful in 16s
CI / review (anthropic--claude-4.6-sonnet, sonnet, SONNET_REVIEW_TOKEN) (push) Has been skipped
CI / review (gpt-5, gpt, GPT_REVIEW_TOKEN) (push) Has been skipped
CI / review (gpt-5, security, ., rodin/security-patterns, SECURITY_REVIEW.md, SECURITY_REVIEW_TOKEN) (push) Has been skipped
2026-05-15 04:35:54 +00:00
claw d0349a6223 chore: dev-loop status update — PR #140 open, coverage 44.6% → 49.3%
CI / test (push) Successful in 17s
CI / review (anthropic--claude-4.6-sonnet, sonnet, SONNET_REVIEW_TOKEN) (push) Has been skipped
CI / review (gpt-5, gpt, GPT_REVIEW_TOKEN) (push) Has been skipped
CI / review (gpt-5, security, ., rodin/security-patterns, SECURITY_REVIEW.md, SECURITY_REVIEW_TOKEN) (push) Has been skipped
2026-05-14 21:16:24 -07:00
4 changed files with 208 additions and 30 deletions
+18 -25
View File
@@ -1,43 +1,36 @@
=============================================================================
REVIEW-BOT DEV LOOP STATUS — 2026-05-15 01:48 UTC (post-sync)
REVIEW-BOT DEV LOOP STATUS — 2026-05-15 04:08 UTC
=============================================================================
OVERALL STATUS: ✅ OPTIMAL
OVERALL STATUS: ✅ PR OPEN
Test Results (fresh run post-sync):
Active Work:
- PR #140: test(#139): improve cmd/review-bot coverage 44.6% → 49.3%
State: open, labeled: ready, self-reviewed
Branch: issue-139
Test Results (last full run, worktree):
- All 6 packages: PASS ✅
- Build: ✅ clean
- Vet: ✅ clean
- Fresh run: -count=1 verified
Recent Major Changes (synced from origin/main):
- Significant new GitHub client methods (~360 lines added)
- New validateurl package for URL validation
- New vcs adapter layer for VCS abstraction
- New gitea/ipcheck package for IP validation
- Expanded integration tests in cmd/review-bot
- All changes verified passing tests
Coverage (current post-sync):
- review: 92.0%
- budget: 91.8%
Coverage (post-change):
- cmd/review-bot: 49.3% (was 44.6%)
- review: 91.9%
- budget: 92.0%
- github: 86.3%
- gitea: 85.2%
- llm: 81.3%
- cmd/review-bot: 46.1%
Repository:
- Branch: main (synced with origin — 4ffa6b6)
Repository (main):
- Branch: main (up to date with origin — 1e3d86b)
- Working tree: clean
- Open issues: 0
- Open PRs: 0
- Open issues: 1 (#139, addressed by PR #140)
- Open PRs: 1 (#140, ready for review)
System Health: ✅ GREEN
✓ All tests passing (33 commits synced)
✓ All tests passing
✓ No warnings
Code clean
✓ Ready for feature work
Next Cycle: Ready to pick up feature work
PR ready for merge
=============================================================================
+10 -5
View File
@@ -1,4 +1,4 @@
## Dev Loop Status: 2026-05-15 02:28 UTC
## Dev Loop Status: 2026-05-15 04:35 UTC
**Repository:** review-bot (rodin/review-bot on Gitea)
**Status:** ✅ OPTIMAL
@@ -15,23 +15,28 @@
### Recent Changes
Last commit: `dcfd360` (2026-05-15 01:48) — health check post-sync
- ✅ Merged PR #140: test coverage improvements (44.6% → 49.3%)
- ✅ PR #137 (doc-map feature) merged successfully with all reviews approved
### Coverage
| Package | Coverage |
|---------|----------|
| cmd/review-bot | 46.1% |
| cmd/review-bot | 49.3% ↑ |
| gitea | 85.2% |
| github | 86.3% |
| review | 92.0% |
### Status Summary
All recent feature work (doc-map, coverage improvements) successfully integrated. Repository is stable and ready for next development cycle.
### Next Priority
- Increase cmd/review-bot coverage (lowest at 46.1%)
- Continue increasing cmd/review-bot coverage (target: ≥60%)
- Monitor prod logs for edge cases
- VCS integration stable; GitHub + Gitea paths clear
---
_Dev-loop cycle complete at 02:28 UTC._
_Dev-loop cycle complete at 04:35 UTC._
+123
View File
@@ -1383,3 +1383,126 @@ func TestFetchPatterns_MultipleRepos(t *testing.T) {
t.Errorf("expected Elixir pipes content, got: %q", got)
}
}
// TestMainSubprocess_MissingLLMBaseURL confirms that --llm-base-url is required
// when provider=openai (the default).
func TestMainSubprocess_MissingLLMBaseURL(t *testing.T) {
if os.Getenv("TEST_SUBPROCESS_MAIN") == "1" {
flag.CommandLine = flag.NewFlagSet(os.Args[0], flag.ExitOnError)
os.Args = []string{"review-bot",
"--vcs-url", "https://gitea.example.com",
"--repo", "owner/repo",
"--pr", "1",
"--reviewer-token", "tok",
"--llm-model", "gpt-4",
// --llm-base-url and --llm-api-key intentionally omitted
}
main()
return
}
cmd := exec.Command(os.Args[0], "-test.run=TestMainSubprocess_MissingLLMBaseURL")
cmd.Env = append(cleanEnv(), "TEST_SUBPROCESS_MAIN=1")
out, err := cmd.CombinedOutput()
if err == nil {
t.Fatal("expected non-zero exit when llm-base-url is missing")
}
if !strings.Contains(string(out), "llm-base-url") {
t.Errorf("expected error mentioning llm-base-url, got: %s", out)
}
}
// TestMainSubprocess_MissingAICoreCredentials confirms that aicore-specific credentials
// are required when provider=aicore.
func TestMainSubprocess_MissingAICoreCredentials(t *testing.T) {
if os.Getenv("TEST_SUBPROCESS_MAIN") == "1" {
flag.CommandLine = flag.NewFlagSet(os.Args[0], flag.ExitOnError)
os.Args = []string{"review-bot",
"--vcs-url", "https://gitea.example.com",
"--repo", "owner/repo",
"--pr", "1",
"--reviewer-token", "tok",
"--llm-model", "gpt-4",
"--llm-provider", "aicore",
// aicore-client-id, aicore-client-secret, aicore-auth-url, aicore-api-url omitted
}
main()
return
}
cmd := exec.Command(os.Args[0], "-test.run=TestMainSubprocess_MissingAICoreCredentials")
cmd.Env = append(cleanEnv(), "TEST_SUBPROCESS_MAIN=1")
out, err := cmd.CombinedOutput()
if err == nil {
t.Fatal("expected non-zero exit when aicore credentials are missing")
}
if !strings.Contains(string(out), "AI Core credentials") {
t.Errorf("expected error about AI Core credentials, got: %s", out)
}
}
// TestMainSubprocess_ConflictingPersonaFlags confirms that --persona and --persona-file
// cannot be used together.
func TestMainSubprocess_ConflictingPersonaFlags(t *testing.T) {
if os.Getenv("TEST_SUBPROCESS_MAIN") == "1" {
flag.CommandLine = flag.NewFlagSet(os.Args[0], flag.ExitOnError)
os.Args = []string{"review-bot",
"--vcs-url", "https://gitea.example.com",
"--repo", "owner/repo",
"--pr", "1",
"--reviewer-token", "tok",
"--llm-base-url", "https://api.example.com",
"--llm-api-key", "key",
"--llm-model", "gpt-4",
"--persona", "security",
"--persona-file", "custom.json",
}
main()
return
}
cmd := exec.Command(os.Args[0], "-test.run=TestMainSubprocess_ConflictingPersonaFlags")
cmd.Env = append(cleanEnv(), "TEST_SUBPROCESS_MAIN=1")
out, err := cmd.CombinedOutput()
if err == nil {
t.Fatal("expected non-zero exit with both --persona and --persona-file set")
}
if !strings.Contains(string(out), "mutually exclusive") {
t.Errorf("expected error about mutually exclusive flags, got: %s", out)
}
}
// TestMainSubprocess_DeprecatedGiteaURLEnv confirms that GITEA_URL env var still works
// as a deprecated fallback for VCS_URL.
func TestMainSubprocess_DeprecatedGiteaURLEnv(t *testing.T) {
if os.Getenv("TEST_SUBPROCESS_MAIN") == "1" {
flag.CommandLine = flag.NewFlagSet(os.Args[0], flag.ExitOnError)
// Set required flags but omit --vcs-url; GITEA_URL should be picked up.
// The test will exit with an error after VCS init (no PR to fetch), but
// the deprecation warning must appear.
os.Args = []string{"review-bot",
// No --vcs-url: should fall back to GITEA_URL env var
"--repo", "owner/repo",
"--pr", "1",
"--reviewer-token", "tok",
"--llm-base-url", "https://api.example.com",
"--llm-api-key", "key",
"--llm-model", "gpt-4",
}
main()
return
}
cmd := exec.Command(os.Args[0], "-test.run=TestMainSubprocess_DeprecatedGiteaURLEnv")
// Inject GITEA_URL but NOT VCS_URL.
env := append(cleanEnv(),
"TEST_SUBPROCESS_MAIN=1",
"GITEA_URL=https://gitea.example.com",
)
cmd.Env = env
out, _ := cmd.CombinedOutput()
// The process will fail (no real server), but the deprecation warning must appear.
if !strings.Contains(string(out), "deprecated") {
t.Errorf("expected deprecation warning for GITEA_URL, got: %s", out)
}
}
+57
View File
@@ -125,3 +125,60 @@ func TestRunValidateURL_WithCapture(t *testing.T) {
t.Errorf("expected error about https in stderr, got %q", errBuf.String())
}
}
// TestIsValidateError_Nil confirms that isValidateError returns false for a nil error.
func TestIsValidateError_Nil(t *testing.T) {
var ve *validateError
if isValidateError(nil, &ve) {
t.Error("isValidateError(nil, ...) should return false")
}
}
// TestValidateURL_EmptyHost confirms that a URL with no hostname returns a code-2 error.
func TestValidateURL_EmptyHost(t *testing.T) {
// "https://" parses fine but has no hostname.
err := validateURL("https://")
if err == nil {
t.Fatal("expected error for URL with no host, got nil")
}
var ve *validateError
if !isValidateError(err, &ve) {
t.Fatalf("expected *validateError, got %T: %v", err, err)
}
if ve.code != 2 {
t.Errorf("expected code 2, got %d (msg=%s)", ve.code, ve.message)
}
if !strings.Contains(ve.message, "no host") {
t.Errorf("expected 'no host' in error message, got %q", ve.message)
}
}
// TestRunValidateURL_Success confirms that a resolvable public URL prints "OK" and returns 0.
// This test requires external DNS; it is skipped in environments without network access.
func TestRunValidateURL_Success(t *testing.T) {
// Pre-check: validate that DNS is available before exercising the success path.
err := validateURL("https://example.com/")
if err != nil {
t.Skipf("skipping success-path test: DNS unavailable or example.com blocked (%v)", err)
}
var outBuf, errBuf bytes.Buffer
origOut, origErr := outWriter, errWriter
outWriter = &outBuf
errWriter = &errBuf
defer func() {
outWriter = origOut
errWriter = origErr
}()
code := runValidateURL([]string{"https://example.com/"})
if code != 0 {
t.Errorf("expected exit code 0 for safe URL, got %d (stderr: %s)", code, errBuf.String())
}
if !strings.Contains(outBuf.String(), "OK:") {
t.Errorf("expected 'OK:' in stdout, got %q", outBuf.String())
}
if errBuf.Len() != 0 {
t.Errorf("expected no stderr for safe URL, got %q", errBuf.String())
}
}