mirror of
https://github.com/multica-ai/multica.git
synced 2026-06-17 03:38:32 +02:00
Re-introduces the `involves_user_id` filter on the issues list / open-list / count / grouped paths, but with the semantics nailed down for the second time around: tab 3 surfaces issues whose assignee is an *indirect* extension of the user (owned agent, or a squad they're a human member of / lead via owned agent / have an owned agent inside) — and explicitly NOT direct member assignment, which is tab 1's meaning. - server/pkg/db/queries/issue.sql: 4-branch filter on ListIssues / ListOpenIssues / CountIssues. Each subquery clamps workspace_id because issue.assignee_id is polymorphic with no FK. Leader resolution reads squad.leader_id directly, not the squad_member copy row (squad.go ignores errors when seeding that copy, so it can be missing). FindActiveDuplicateIssue switched from positional $2/$3/$4 to named sqlc.arg() — pure hygiene so the generated struct field names don't drift when new nargs are added. - server/internal/handler/issue.go: parse involves_user_id and plumb it into the three sqlc params; ListGroupedIssues (hand-written dynamic SQL) gets a mirrored 4-branch fragment, no shortcut. - packages/core: ListIssuesParams / ListGroupedIssuesParams / MyIssuesFilter / api.listIssues / api.listGroupedIssues all carry the new param through. - packages/views/my-issues: tab 3 switches from client-side agent-fanout to involves_user_id=user.id. agentListOptions import and the myAgentIds memo go away. - server/internal/handler/issue_involves_test.go: 13 integration tests cover every branch (positive + cross-workspace negatives) plus the critical ExcludesDirectMemberAssignee negative on BOTH the sqlc and the grouped paths, locking tab 3 ∩ tab 1 = ∅. Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com> Co-authored-by: multica-agent <github@multica.ai>
468 lines
20 KiB
Go
468 lines
20 KiB
Go
package handler
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
// involvesFixture seeds, for a single test, the data needed to exercise every
|
|
// branch of the `involves_user_id` 4-branch filter — owned agent, squad human
|
|
// member, squad canonical leader (via squad.leader_id, NOT a squad_member copy
|
|
// row), and squad agent member — plus a parallel set in a second workspace so
|
|
// the cross-workspace negative tests can prove subquery-level isolation.
|
|
type involvesFixture struct {
|
|
// All IDs are in the primary handler-test workspace unless noted otherwise.
|
|
userID string // the "me" user the filter is keyed on (== testUserID)
|
|
otherID string // a different user in the same workspace
|
|
|
|
ownedAgentID string // agent.owner_id = userID — branch (1) seed
|
|
otherAgentID string // agent.owner_id = otherID — must NOT match
|
|
squadMemberID string // squad with userID as human member — branch (2)
|
|
squadLeaderID string // squad whose leader_id is an agent owned by userID — branch (3)
|
|
squadAgentMemID string // squad with an owned-agent as squad_member row — branch (4)
|
|
|
|
// Other workspace, mirror objects — used by ExcludesOtherWorkspace* tests
|
|
otherWsID string
|
|
otherWsAgent string // owned by userID but in other workspace
|
|
otherWsSquadMember string // squad with userID as human member, in other ws
|
|
otherWsSquadLeader string // squad whose leader is userID's agent (in other ws)
|
|
otherWsSquadAgentMem string // squad with userID's agent as member (in other ws)
|
|
}
|
|
|
|
func setupInvolvesFixture(t *testing.T) *involvesFixture {
|
|
t.Helper()
|
|
ctx := context.Background()
|
|
suffix := time.Now().UnixNano()
|
|
|
|
fx := &involvesFixture{userID: testUserID}
|
|
|
|
// --- second user inside the primary workspace ---
|
|
var otherUserID string
|
|
if err := testPool.QueryRow(ctx, `
|
|
INSERT INTO "user" (name, email) VALUES ($1, $2) RETURNING id
|
|
`, "Involves Other User", fmt.Sprintf("involves-other-%d@multica.ai", suffix)).Scan(&otherUserID); err != nil {
|
|
t.Fatalf("create other user: %v", err)
|
|
}
|
|
t.Cleanup(func() { testPool.Exec(context.Background(), `DELETE FROM "user" WHERE id = $1`, otherUserID) })
|
|
fx.otherID = otherUserID
|
|
if _, err := testPool.Exec(ctx, `
|
|
INSERT INTO member (workspace_id, user_id, role) VALUES ($1, $2, 'member')
|
|
`, testWorkspaceID, otherUserID); err != nil {
|
|
t.Fatalf("create other member: %v", err)
|
|
}
|
|
|
|
runtimeID := handlerTestRuntimeID(t)
|
|
|
|
// --- agents in primary workspace ---
|
|
fx.ownedAgentID = insertAgent(t, ctx, testWorkspaceID, runtimeID, fx.userID,
|
|
fmt.Sprintf("Involves Owned Agent %d", suffix))
|
|
fx.otherAgentID = insertAgent(t, ctx, testWorkspaceID, runtimeID, fx.otherID,
|
|
fmt.Sprintf("Involves Other Agent %d", suffix))
|
|
|
|
// --- a squad we already have to satisfy NOT NULL leader_id ---
|
|
leaderForMemberSquad := insertAgent(t, ctx, testWorkspaceID, runtimeID, fx.otherID,
|
|
fmt.Sprintf("Involves Leader-for-MemberSquad %d", suffix))
|
|
fx.squadMemberID = insertSquad(t, ctx, testWorkspaceID, leaderForMemberSquad,
|
|
fmt.Sprintf("InvolvesSquadMember-%d", suffix))
|
|
// Add the test user as a human member.
|
|
if _, err := testPool.Exec(ctx, `
|
|
INSERT INTO squad_member (squad_id, member_type, member_id) VALUES ($1, 'member', $2)
|
|
`, fx.squadMemberID, fx.userID); err != nil {
|
|
t.Fatalf("add squad human member: %v", err)
|
|
}
|
|
|
|
// --- squad with leader = our owned agent — branch (3). Critically, we do
|
|
// NOT insert a squad_member row for the leader, so the test exercises the
|
|
// canonical squad.leader_id path. ---
|
|
fx.squadLeaderID = insertSquad(t, ctx, testWorkspaceID, fx.ownedAgentID,
|
|
fmt.Sprintf("InvolvesSquadLeader-%d", suffix))
|
|
|
|
// --- squad whose agent member is our owned agent — branch (4) ---
|
|
leaderForAgentMemSquad := insertAgent(t, ctx, testWorkspaceID, runtimeID, fx.otherID,
|
|
fmt.Sprintf("Involves Leader-for-AgentMemSquad %d", suffix))
|
|
fx.squadAgentMemID = insertSquad(t, ctx, testWorkspaceID, leaderForAgentMemSquad,
|
|
fmt.Sprintf("InvolvesSquadAgentMem-%d", suffix))
|
|
// Use a fresh owned agent so the squad_member row is the only signal —
|
|
// keeps branch (4) independent from branch (1)/(3).
|
|
branch4Agent := insertAgent(t, ctx, testWorkspaceID, runtimeID, fx.userID,
|
|
fmt.Sprintf("Involves Branch4 Agent %d", suffix))
|
|
if _, err := testPool.Exec(ctx, `
|
|
INSERT INTO squad_member (squad_id, member_type, member_id) VALUES ($1, 'agent', $2)
|
|
`, fx.squadAgentMemID, branch4Agent); err != nil {
|
|
t.Fatalf("add squad agent member: %v", err)
|
|
}
|
|
|
|
// --- second workspace, mirroring all four shapes for cross-ws negatives ---
|
|
var otherWsID string
|
|
if err := testPool.QueryRow(ctx, `
|
|
INSERT INTO workspace (name, slug, description, issue_prefix)
|
|
VALUES ($1, $2, '', 'OTH')
|
|
RETURNING id
|
|
`, fmt.Sprintf("InvolvesOtherWs-%d", suffix), fmt.Sprintf("involves-other-ws-%d", suffix)).Scan(&otherWsID); err != nil {
|
|
t.Fatalf("create other workspace: %v", err)
|
|
}
|
|
t.Cleanup(func() { testPool.Exec(context.Background(), `DELETE FROM workspace WHERE id = $1`, otherWsID) })
|
|
fx.otherWsID = otherWsID
|
|
|
|
// Membership in other workspace (so the user could legitimately be assigned
|
|
// there too — exercises whether subquery workspace_id clause filters it out).
|
|
if _, err := testPool.Exec(ctx, `
|
|
INSERT INTO member (workspace_id, user_id, role) VALUES ($1, $2, 'owner')
|
|
`, otherWsID, fx.userID); err != nil {
|
|
t.Fatalf("create other-ws member: %v", err)
|
|
}
|
|
|
|
var otherRuntimeID string
|
|
if err := testPool.QueryRow(ctx, `
|
|
INSERT INTO agent_runtime (
|
|
workspace_id, daemon_id, name, runtime_mode, provider, status, device_info, metadata, last_seen_at
|
|
) VALUES ($1, NULL, $2, 'cloud', 'other_ws_runtime', 'online', $3, '{}'::jsonb, now())
|
|
RETURNING id
|
|
`, otherWsID, fmt.Sprintf("OtherWs Runtime %d", suffix), "other-ws-runtime").Scan(&otherRuntimeID); err != nil {
|
|
t.Fatalf("create other-ws runtime: %v", err)
|
|
}
|
|
|
|
fx.otherWsAgent = insertAgent(t, ctx, otherWsID, otherRuntimeID, fx.userID,
|
|
fmt.Sprintf("OtherWs Owned Agent %d", suffix))
|
|
|
|
leaderForOtherWsMemberSquad := insertAgent(t, ctx, otherWsID, otherRuntimeID, fx.otherID,
|
|
fmt.Sprintf("OtherWs Leader-for-MemberSquad %d", suffix))
|
|
fx.otherWsSquadMember = insertSquad(t, ctx, otherWsID, leaderForOtherWsMemberSquad,
|
|
fmt.Sprintf("OtherWsSquadMember-%d", suffix))
|
|
if _, err := testPool.Exec(ctx, `
|
|
INSERT INTO squad_member (squad_id, member_type, member_id) VALUES ($1, 'member', $2)
|
|
`, fx.otherWsSquadMember, fx.userID); err != nil {
|
|
t.Fatalf("add other-ws squad member: %v", err)
|
|
}
|
|
|
|
fx.otherWsSquadLeader = insertSquad(t, ctx, otherWsID, fx.otherWsAgent,
|
|
fmt.Sprintf("OtherWsSquadLeader-%d", suffix))
|
|
|
|
leaderForOtherWsAgentMemSquad := insertAgent(t, ctx, otherWsID, otherRuntimeID, fx.otherID,
|
|
fmt.Sprintf("OtherWs Leader-for-AgentMemSquad %d", suffix))
|
|
fx.otherWsSquadAgentMem = insertSquad(t, ctx, otherWsID, leaderForOtherWsAgentMemSquad,
|
|
fmt.Sprintf("OtherWsSquadAgentMem-%d", suffix))
|
|
otherWsBranch4Agent := insertAgent(t, ctx, otherWsID, otherRuntimeID, fx.userID,
|
|
fmt.Sprintf("OtherWs Branch4 Agent %d", suffix))
|
|
if _, err := testPool.Exec(ctx, `
|
|
INSERT INTO squad_member (squad_id, member_type, member_id) VALUES ($1, 'agent', $2)
|
|
`, fx.otherWsSquadAgentMem, otherWsBranch4Agent); err != nil {
|
|
t.Fatalf("add other-ws squad agent member: %v", err)
|
|
}
|
|
|
|
return fx
|
|
}
|
|
|
|
func insertAgent(t *testing.T, ctx context.Context, workspaceID, runtimeID, ownerID, name string) string {
|
|
t.Helper()
|
|
var id string
|
|
if err := testPool.QueryRow(ctx, `
|
|
INSERT INTO agent (
|
|
workspace_id, name, description, runtime_mode, runtime_config,
|
|
runtime_id, visibility, max_concurrent_tasks, owner_id
|
|
)
|
|
VALUES ($1, $2, '', 'cloud', '{}'::jsonb, $3, 'workspace', 1, $4)
|
|
RETURNING id
|
|
`, workspaceID, name, runtimeID, ownerID).Scan(&id); err != nil {
|
|
t.Fatalf("create agent %q: %v", name, err)
|
|
}
|
|
t.Cleanup(func() { testPool.Exec(context.Background(), `DELETE FROM agent WHERE id = $1`, id) })
|
|
return id
|
|
}
|
|
|
|
func insertSquad(t *testing.T, ctx context.Context, workspaceID, leaderAgentID, name string) string {
|
|
t.Helper()
|
|
var id string
|
|
// creator_id is loose (no FK) — reuse testUserID to keep the row valid.
|
|
if err := testPool.QueryRow(ctx, `
|
|
INSERT INTO squad (workspace_id, name, leader_id, creator_id)
|
|
VALUES ($1, $2, $3, $4)
|
|
RETURNING id
|
|
`, workspaceID, name, leaderAgentID, testUserID).Scan(&id); err != nil {
|
|
t.Fatalf("create squad %q: %v", name, err)
|
|
}
|
|
t.Cleanup(func() { testPool.Exec(context.Background(), `DELETE FROM squad WHERE id = $1`, id) })
|
|
return id
|
|
}
|
|
|
|
// insertIssueTo creates an issue in the given workspace assigned to the given
|
|
// (assigneeType, assigneeID) pair and returns its UUID. Issue rows are
|
|
// best-effort-cleaned up by the test.
|
|
func insertIssueTo(t *testing.T, ctx context.Context, workspaceID, title, assigneeType, assigneeID string) string {
|
|
t.Helper()
|
|
var number int32
|
|
if err := testPool.QueryRow(ctx, `
|
|
UPDATE workspace
|
|
SET issue_counter = GREATEST(
|
|
issue_counter,
|
|
(SELECT COALESCE(MAX(number), 0) FROM issue WHERE workspace_id = $1)
|
|
) + 1
|
|
WHERE id = $1
|
|
RETURNING issue_counter
|
|
`, workspaceID).Scan(&number); err != nil {
|
|
t.Fatalf("next issue number: %v", err)
|
|
}
|
|
var id string
|
|
if err := testPool.QueryRow(ctx, `
|
|
INSERT INTO issue (
|
|
workspace_id, title, description, status, priority,
|
|
assignee_type, assignee_id, creator_type, creator_id,
|
|
position, number
|
|
)
|
|
VALUES ($1, $2, NULL, 'todo', 'none', $3, $4, 'member', $5, 0, $6)
|
|
RETURNING id
|
|
`, workspaceID, title, assigneeType, assigneeID, testUserID, number).Scan(&id); err != nil {
|
|
t.Fatalf("create issue %q: %v", title, err)
|
|
}
|
|
t.Cleanup(func() { testPool.Exec(context.Background(), `DELETE FROM issue WHERE id = $1`, id) })
|
|
return id
|
|
}
|
|
|
|
// listIssuesInvolves runs ListIssues with `involves_user_id` set to userID
|
|
// (against testWorkspaceID) and returns the resulting issue IDs.
|
|
func listIssuesInvolves(t *testing.T, userID string) []string {
|
|
t.Helper()
|
|
path := fmt.Sprintf("/api/issues?workspace_id=%s&involves_user_id=%s&limit=500",
|
|
testWorkspaceID, userID)
|
|
w := httptest.NewRecorder()
|
|
testHandler.ListIssues(w, newRequest("GET", path, nil))
|
|
if w.Code != http.StatusOK {
|
|
t.Fatalf("ListIssues: expected 200, got %d: %s", w.Code, w.Body.String())
|
|
}
|
|
var resp struct {
|
|
Issues []IssueResponse `json:"issues"`
|
|
}
|
|
if err := json.NewDecoder(w.Body).Decode(&resp); err != nil {
|
|
t.Fatalf("decode list response: %v", err)
|
|
}
|
|
ids := make([]string, 0, len(resp.Issues))
|
|
for _, iss := range resp.Issues {
|
|
ids = append(ids, iss.ID)
|
|
}
|
|
return ids
|
|
}
|
|
|
|
// listGroupedIssuesInvolves runs ListGroupedIssues with `involves_user_id`
|
|
// set to userID and returns the flattened set of issue IDs across all groups.
|
|
func listGroupedIssuesInvolves(t *testing.T, userID string) []string {
|
|
t.Helper()
|
|
path := fmt.Sprintf(
|
|
"/api/issues/grouped?workspace_id=%s&group_by=assignee&statuses=todo&involves_user_id=%s&limit=100",
|
|
testWorkspaceID, userID)
|
|
w := httptest.NewRecorder()
|
|
testHandler.ListGroupedIssues(w, newRequest("GET", path, nil))
|
|
if w.Code != http.StatusOK {
|
|
t.Fatalf("ListGroupedIssues: expected 200, got %d: %s", w.Code, w.Body.String())
|
|
}
|
|
var resp GroupedIssuesResponse
|
|
if err := json.NewDecoder(w.Body).Decode(&resp); err != nil {
|
|
t.Fatalf("decode grouped response: %v", err)
|
|
}
|
|
ids := []string{}
|
|
for _, g := range resp.Groups {
|
|
for _, iss := range g.Issues {
|
|
ids = append(ids, iss.ID)
|
|
}
|
|
}
|
|
return ids
|
|
}
|
|
|
|
func containsIssueID(ids []string, target string) bool {
|
|
for _, id := range ids {
|
|
if id == target {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// ---- positive branches ----
|
|
|
|
func TestListIssues_InvolvesUserID_MatchesOwnedAgentAssignee(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
wantID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"issue assigned to my owned agent", "agent", fx.ownedAgentID)
|
|
if got := listIssuesInvolves(t, fx.userID); !containsIssueID(got, wantID) {
|
|
t.Fatalf("branch (1) miss: owned-agent assignee not surfaced (want %s, got %v)", wantID, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_MatchesSquadMember(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
wantID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"issue assigned to a squad I'm a member of", "squad", fx.squadMemberID)
|
|
if got := listIssuesInvolves(t, fx.userID); !containsIssueID(got, wantID) {
|
|
t.Fatalf("branch (2) miss: human-member squad assignee not surfaced (want %s, got %v)", wantID, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_MatchesLeaderViaCanonicalRelation(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
// Fixture deliberately omits the squad_member leader-copy row, so this
|
|
// can only match if the SQL reads squad.leader_id directly (branch 3).
|
|
wantID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"issue assigned to a squad my agent leads", "squad", fx.squadLeaderID)
|
|
if got := listIssuesInvolves(t, fx.userID); !containsIssueID(got, wantID) {
|
|
t.Fatalf("branch (3) miss: squad-leader-via-canonical assignee not surfaced (want %s, got %v)", wantID, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_MatchesSquadAgentMember(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
wantID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"issue assigned to a squad my agent is a member of", "squad", fx.squadAgentMemID)
|
|
if got := listIssuesInvolves(t, fx.userID); !containsIssueID(got, wantID) {
|
|
t.Fatalf("branch (4) miss: squad agent-member assignee not surfaced (want %s, got %v)", wantID, got)
|
|
}
|
|
}
|
|
|
|
// ---- the critical negative: tab 3 must be disjoint from tab 1 ----
|
|
|
|
// Nails the semantics: `involves_user_id` MUST NOT surface issues whose
|
|
// assignee is the user themself (member type). Direct member assignment is
|
|
// the meaning of `assignee_id` (tab 1 "Assigned to me"); the two tabs must
|
|
// produce disjoint result sets. If anyone adds a fifth UNION branch
|
|
// `(assignee_type='member' AND assignee_id=involves_user_id)` back in, this
|
|
// test fails.
|
|
func TestListIssues_InvolvesUserID_ExcludesDirectMemberAssignee(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
issueID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"tab 3 must NOT surface member-direct assignment", "member", fx.userID)
|
|
if got := listIssuesInvolves(t, fx.userID); containsIssueID(got, issueID) {
|
|
t.Fatalf("tab 3 semantics violated: involves_user_id surfaced a member-direct assignee issue (id=%s); that belongs to tab 1. Full result: %v",
|
|
issueID, got)
|
|
}
|
|
}
|
|
|
|
// Same negative on the grouped (dynamic SQL) path — the dynamic builder is a
|
|
// separate code path from sqlc, so it gets its own regression.
|
|
func TestListGroupedIssues_InvolvesUserID_ExcludesDirectMemberAssignee(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
issueID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"grouped tab 3 must NOT surface member-direct assignment", "member", fx.userID)
|
|
if got := listGroupedIssuesInvolves(t, fx.userID); containsIssueID(got, issueID) {
|
|
t.Fatalf("grouped tab 3 semantics violated: involves_user_id surfaced a member-direct assignee issue (id=%s); full result: %v",
|
|
issueID, got)
|
|
}
|
|
}
|
|
|
|
// ---- workspace isolation negatives — each subquery must clamp workspace_id ----
|
|
|
|
func TestListIssues_InvolvesUserID_ExcludesOtherWorkspaceAgent(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
// Issue lives in the *primary* workspace but is assigned to an agent UUID
|
|
// that only exists in the OTHER workspace and is owned by our user. If the
|
|
// agent subquery is missing `a.workspace_id = $1`, this match would leak.
|
|
issueID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"cross-ws agent assignee must not leak", "agent", fx.otherWsAgent)
|
|
if got := listIssuesInvolves(t, fx.userID); containsIssueID(got, issueID) {
|
|
t.Fatalf("workspace isolation violated: cross-workspace agent surfaced (id=%s); full result: %v",
|
|
issueID, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_ExcludesOtherWorkspaceLeader(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
issueID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"cross-ws squad-leader assignee must not leak", "squad", fx.otherWsSquadLeader)
|
|
if got := listIssuesInvolves(t, fx.userID); containsIssueID(got, issueID) {
|
|
t.Fatalf("workspace isolation violated: cross-workspace squad-leader surfaced (id=%s); full result: %v",
|
|
issueID, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_ExcludesOtherWorkspaceSquadMember(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
issueID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"cross-ws squad-human-member assignee must not leak", "squad", fx.otherWsSquadMember)
|
|
if got := listIssuesInvolves(t, fx.userID); containsIssueID(got, issueID) {
|
|
t.Fatalf("workspace isolation violated: cross-workspace squad-human-member surfaced (id=%s); full result: %v",
|
|
issueID, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_ExcludesOtherWorkspaceSquadAgentMember(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
issueID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"cross-ws squad-agent-member assignee must not leak", "squad", fx.otherWsSquadAgentMem)
|
|
if got := listIssuesInvolves(t, fx.userID); containsIssueID(got, issueID) {
|
|
t.Fatalf("workspace isolation violated: cross-workspace squad-agent-member surfaced (id=%s); full result: %v",
|
|
issueID, got)
|
|
}
|
|
}
|
|
|
|
// ---- combo + boundary ----
|
|
|
|
// involves_user_id and creator_id must AND together — combining narrowing
|
|
// filters should never widen the result.
|
|
func TestListIssues_InvolvesUserID_CombinesWithCreatorID(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
// Issue with creator = otherID: involves matches (branch 1) but creator
|
|
// filter must exclude it.
|
|
exclude := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"involves matches but creator does not", "agent", fx.ownedAgentID)
|
|
// Patch the creator to otherID directly.
|
|
if _, err := testPool.Exec(ctx, `UPDATE issue SET creator_id = $1 WHERE id = $2`, fx.otherID, exclude); err != nil {
|
|
t.Fatalf("patch creator: %v", err)
|
|
}
|
|
|
|
path := fmt.Sprintf("/api/issues?workspace_id=%s&involves_user_id=%s&creator_id=%s&limit=500",
|
|
testWorkspaceID, fx.userID, fx.userID)
|
|
w := httptest.NewRecorder()
|
|
testHandler.ListIssues(w, newRequest("GET", path, nil))
|
|
if w.Code != http.StatusOK {
|
|
t.Fatalf("ListIssues: expected 200, got %d: %s", w.Code, w.Body.String())
|
|
}
|
|
var resp struct {
|
|
Issues []IssueResponse `json:"issues"`
|
|
}
|
|
if err := json.NewDecoder(w.Body).Decode(&resp); err != nil {
|
|
t.Fatalf("decode list response: %v", err)
|
|
}
|
|
got := make([]string, 0, len(resp.Issues))
|
|
for _, iss := range resp.Issues {
|
|
got = append(got, iss.ID)
|
|
}
|
|
if containsIssueID(got, exclude) {
|
|
t.Fatalf("combined filter widened result: issue %s with non-matching creator surfaced; full result: %v",
|
|
exclude, got)
|
|
}
|
|
}
|
|
|
|
func TestListIssues_InvolvesUserID_InvalidUUIDReturns400(t *testing.T) {
|
|
path := fmt.Sprintf("/api/issues?workspace_id=%s&involves_user_id=not-a-uuid", testWorkspaceID)
|
|
w := httptest.NewRecorder()
|
|
testHandler.ListIssues(w, newRequest("GET", path, nil))
|
|
if w.Code != http.StatusBadRequest {
|
|
t.Fatalf("expected 400 on invalid UUID, got %d: %s", w.Code, w.Body.String())
|
|
}
|
|
}
|
|
|
|
// Grouped path also exercises canonical-leader resolution, so a single positive
|
|
// guards the dynamic SQL builder against accidentally dropping branch (3).
|
|
func TestListGroupedIssues_InvolvesUserID_MatchesLeaderViaCanonicalRelation(t *testing.T) {
|
|
ctx := context.Background()
|
|
fx := setupInvolvesFixture(t)
|
|
wantID := insertIssueTo(t, ctx, testWorkspaceID,
|
|
"grouped: squad my agent leads", "squad", fx.squadLeaderID)
|
|
if got := listGroupedIssuesInvolves(t, fx.userID); !containsIssueID(got, wantID) {
|
|
t.Fatalf("grouped branch (3) miss: squad-leader-via-canonical not surfaced (want %s, got %v)", wantID, got)
|
|
}
|
|
}
|