mirror of
https://github.com/alkimake/paperclip.git
synced 2026-06-16 19:00:38 +09:00
Merge pull request #1708 from paperclipai/pr/pap-817-onboarding-goal-context
Seed onboarding project and issue goal context
This commit is contained in:
commit
03f44d0089
8 changed files with 556 additions and 34 deletions
131
ui/src/lib/onboarding-launch.test.ts
Normal file
131
ui/src/lib/onboarding-launch.test.ts
Normal file
|
|
@ -0,0 +1,131 @@
|
|||
import { describe, expect, it } from "vitest";
|
||||
import {
|
||||
buildOnboardingIssuePayload,
|
||||
buildOnboardingProjectPayload,
|
||||
selectDefaultCompanyGoalId,
|
||||
} from "./onboarding-launch";
|
||||
|
||||
describe("selectDefaultCompanyGoalId", () => {
|
||||
it("prefers the earliest active root company goal", () => {
|
||||
expect(
|
||||
selectDefaultCompanyGoalId([
|
||||
{
|
||||
id: "team-goal",
|
||||
companyId: "company-1",
|
||||
title: "Nested",
|
||||
description: null,
|
||||
level: "team",
|
||||
status: "active",
|
||||
parentId: null,
|
||||
ownerAgentId: null,
|
||||
createdAt: new Date("2026-03-04T00:00:00Z"),
|
||||
updatedAt: new Date("2026-03-04T00:00:00Z"),
|
||||
},
|
||||
{
|
||||
id: "goal-2",
|
||||
companyId: "company-1",
|
||||
title: "Later active root",
|
||||
description: null,
|
||||
level: "company",
|
||||
status: "active",
|
||||
parentId: null,
|
||||
ownerAgentId: null,
|
||||
createdAt: new Date("2026-03-03T00:00:00Z"),
|
||||
updatedAt: new Date("2026-03-03T00:00:00Z"),
|
||||
},
|
||||
{
|
||||
id: "goal-1",
|
||||
companyId: "company-1",
|
||||
title: "Earliest active root",
|
||||
description: null,
|
||||
level: "company",
|
||||
status: "active",
|
||||
parentId: null,
|
||||
ownerAgentId: null,
|
||||
createdAt: new Date("2026-03-02T00:00:00Z"),
|
||||
updatedAt: new Date("2026-03-02T00:00:00Z"),
|
||||
},
|
||||
]),
|
||||
).toBe("goal-1");
|
||||
});
|
||||
|
||||
it("falls back to the earliest root company goal when none are active", () => {
|
||||
expect(
|
||||
selectDefaultCompanyGoalId([
|
||||
{
|
||||
id: "goal-2",
|
||||
companyId: "company-1",
|
||||
title: "Cancelled root",
|
||||
description: null,
|
||||
level: "company",
|
||||
status: "cancelled",
|
||||
parentId: null,
|
||||
ownerAgentId: null,
|
||||
createdAt: new Date("2026-03-03T00:00:00Z"),
|
||||
updatedAt: new Date("2026-03-03T00:00:00Z"),
|
||||
},
|
||||
{
|
||||
id: "goal-1",
|
||||
companyId: "company-1",
|
||||
title: "Earliest root",
|
||||
description: null,
|
||||
level: "company",
|
||||
status: "planned",
|
||||
parentId: null,
|
||||
ownerAgentId: null,
|
||||
createdAt: new Date("2026-03-02T00:00:00Z"),
|
||||
updatedAt: new Date("2026-03-02T00:00:00Z"),
|
||||
},
|
||||
]),
|
||||
).toBe("goal-1");
|
||||
});
|
||||
});
|
||||
|
||||
describe("onboarding launch payloads", () => {
|
||||
it("links the onboarding project and first issue to the selected goal", () => {
|
||||
expect(buildOnboardingProjectPayload("goal-1")).toEqual({
|
||||
name: "Onboarding",
|
||||
status: "in_progress",
|
||||
goalIds: ["goal-1"],
|
||||
});
|
||||
|
||||
expect(
|
||||
buildOnboardingIssuePayload({
|
||||
title: " Hire your first engineer ",
|
||||
description: " Kick off the hiring plan ",
|
||||
assigneeAgentId: "agent-1",
|
||||
projectId: "project-1",
|
||||
goalId: "goal-1",
|
||||
}),
|
||||
).toEqual({
|
||||
title: "Hire your first engineer",
|
||||
description: "Kick off the hiring plan",
|
||||
assigneeAgentId: "agent-1",
|
||||
projectId: "project-1",
|
||||
goalId: "goal-1",
|
||||
status: "todo",
|
||||
});
|
||||
});
|
||||
|
||||
it("omits goal links when no default company goal exists", () => {
|
||||
expect(buildOnboardingProjectPayload(null)).toEqual({
|
||||
name: "Onboarding",
|
||||
status: "in_progress",
|
||||
});
|
||||
|
||||
expect(
|
||||
buildOnboardingIssuePayload({
|
||||
title: "Task",
|
||||
description: "",
|
||||
assigneeAgentId: "agent-1",
|
||||
projectId: "project-1",
|
||||
goalId: null,
|
||||
}),
|
||||
).toEqual({
|
||||
title: "Task",
|
||||
assigneeAgentId: "agent-1",
|
||||
projectId: "project-1",
|
||||
status: "todo",
|
||||
});
|
||||
});
|
||||
});
|
||||
53
ui/src/lib/onboarding-launch.ts
Normal file
53
ui/src/lib/onboarding-launch.ts
Normal file
|
|
@ -0,0 +1,53 @@
|
|||
import type { Goal } from "@paperclipai/shared";
|
||||
|
||||
export const ONBOARDING_PROJECT_NAME = "Onboarding";
|
||||
|
||||
function goalCreatedAt(goal: Goal) {
|
||||
const createdAt = goal.createdAt instanceof Date ? goal.createdAt : new Date(goal.createdAt);
|
||||
return Number.isNaN(createdAt.getTime()) ? 0 : createdAt.getTime();
|
||||
}
|
||||
|
||||
function pickEarliestGoal(goals: Goal[]) {
|
||||
return [...goals].sort((a, b) => goalCreatedAt(a) - goalCreatedAt(b))[0] ?? null;
|
||||
}
|
||||
|
||||
export function selectDefaultCompanyGoalId(goals: Goal[]): string | null {
|
||||
const companyGoals = goals.filter((goal) => goal.level === "company");
|
||||
const rootGoals = companyGoals.filter((goal) => !goal.parentId);
|
||||
const activeRootGoals = rootGoals.filter((goal) => goal.status === "active");
|
||||
|
||||
return (
|
||||
pickEarliestGoal(activeRootGoals)?.id ??
|
||||
pickEarliestGoal(rootGoals)?.id ??
|
||||
pickEarliestGoal(companyGoals)?.id ??
|
||||
null
|
||||
);
|
||||
}
|
||||
|
||||
export function buildOnboardingProjectPayload(goalId: string | null) {
|
||||
return {
|
||||
name: ONBOARDING_PROJECT_NAME,
|
||||
status: "in_progress" as const,
|
||||
...(goalId ? { goalIds: [goalId] } : {}),
|
||||
};
|
||||
}
|
||||
|
||||
export function buildOnboardingIssuePayload(input: {
|
||||
title: string;
|
||||
description: string;
|
||||
assigneeAgentId: string;
|
||||
projectId: string;
|
||||
goalId: string | null;
|
||||
}) {
|
||||
const title = input.title.trim();
|
||||
const description = input.description.trim();
|
||||
|
||||
return {
|
||||
title,
|
||||
...(description ? { description } : {}),
|
||||
assigneeAgentId: input.assigneeAgentId,
|
||||
projectId: input.projectId,
|
||||
...(input.goalId ? { goalId: input.goalId } : {}),
|
||||
status: "todo" as const,
|
||||
};
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue