fix: harden JSONL path handling
- bound fresh-clone JSONL discovery to the .beads dir (abs path, traversal guard) before reading counts - add safeWorkspacePath/isWithinWorkspace helpers and use in doctor fixes (database_config, untracked) to reject absolute/traversal inputs and confine .gitattributes edits - normalize git status paths and path-guard tests for cross-OS (Windows) compatibility - add regression tests for the new guards
This commit is contained in:
55
cmd/bd/doctor/fix/common_test.go
Normal file
55
cmd/bd/doctor/fix/common_test.go
Normal file
@@ -0,0 +1,55 @@
|
||||
package fix
|
||||
|
||||
import (
|
||||
"path/filepath"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestSafeWorkspacePath(t *testing.T) {
|
||||
root := t.TempDir()
|
||||
absEscape, _ := filepath.Abs(filepath.Join(root, "..", "escape"))
|
||||
|
||||
tests := []struct {
|
||||
name string
|
||||
relPath string
|
||||
wantErr bool
|
||||
}{
|
||||
{
|
||||
name: "normal relative path",
|
||||
relPath: ".beads/issues.jsonl",
|
||||
wantErr: false,
|
||||
},
|
||||
{
|
||||
name: "nested relative path",
|
||||
relPath: filepath.Join(".beads", "nested", "file.txt"),
|
||||
wantErr: false,
|
||||
},
|
||||
{
|
||||
name: "absolute path rejected",
|
||||
relPath: absEscape,
|
||||
wantErr: true,
|
||||
},
|
||||
{
|
||||
name: "path traversal rejected",
|
||||
relPath: filepath.Join("..", "escape"),
|
||||
wantErr: true,
|
||||
},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got, err := safeWorkspacePath(root, tt.relPath)
|
||||
if (err != nil) != tt.wantErr {
|
||||
t.Fatalf("safeWorkspacePath() error = %v, wantErr %v", err, tt.wantErr)
|
||||
}
|
||||
if err == nil {
|
||||
if !isWithinWorkspace(root, got) {
|
||||
t.Fatalf("resolved path %q not within workspace %q", got, root)
|
||||
}
|
||||
if !filepath.IsAbs(got) {
|
||||
t.Fatalf("resolved path is not absolute: %q", got)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user