mirror of
https://github.com/paperclipai/paperclip
synced 2026-04-25 17:25:15 +02:00
## Thinking Path > - Paperclip orchestrates AI agents inside execution and project workspaces > - Workspace runtime services can be controlled manually by operators and reused by agent runs > - Manual start/stop state was not preserved consistently across workspace policies and routine launches > - Routine launches also needed branch/workspace variables to default from the selected workspace context > - This pull request makes runtime policy state explicit, preserves manual control, and auto-fills routine branch variables from workspace data > - The benefit is less surprising workspace service behavior and fewer manual inputs when running workspace-scoped routines ## What Changed - Added runtime-state handling for manual workspace control across execution and project workspace validators, routes, and services. - Updated heartbeat/runtime startup behavior so manually stopped services are respected. - Auto-filled routine workspace branch variables from available workspace context. - Added focused server and UI tests for workspace runtime and routine variable behavior. - Removed muted gray background styling from workspace pages and cards for a cleaner workspace UI. ## Verification - `pnpm install --frozen-lockfile --ignore-scripts` - `pnpm exec vitest run server/src/__tests__/routines-service.test.ts server/src/__tests__/workspace-runtime.test.ts ui/src/components/RoutineRunVariablesDialog.test.tsx` - Result: 55 tests passed, 21 skipped. The embedded Postgres routines tests skipped on this host with the existing PGlite/Postgres init warning; workspace-runtime and UI tests passed. ## Risks - Medium risk: this touches runtime service start/stop policy and heartbeat launch behavior. - The focused tests cover manual runtime state, routine variables, and workspace runtime reuse paths. > For core feature work, check [`ROADMAP.md`](ROADMAP.md) first and discuss it in `#dev` before opening the PR. Feature PRs that overlap with planned core work may need to be redirected — check the roadmap first. See `CONTRIBUTING.md`. ## Model Used - OpenAI Codex coding agent based on GPT-5, tool-enabled local shell and GitHub workflow, exact runtime context window not exposed in this session. ## Checklist - [x] I have included a thinking path that traces from project context to this change - [x] I have specified the model used (with version and capability details) - [x] I have checked ROADMAP.md and confirmed this PR does not duplicate planned core work - [x] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [x] If this change affects the UI, I have included before/after screenshots, or documented why targeted component/service verification is sufficient here - [x] I have updated relevant documentation to reflect my changes - [x] I have considered and documented any risks above - [x] I will address all Greptile and reviewer comments before requesting merge --------- Co-authored-by: Paperclip <noreply@paperclip.ing>
519 lines
20 KiB
TypeScript
519 lines
20 KiB
TypeScript
import { useCallback, useEffect, useMemo, useState } from "react";
|
|
import {
|
|
WORKSPACE_BRANCH_ROUTINE_VARIABLE,
|
|
type Agent,
|
|
type IssueExecutionWorkspaceSettings,
|
|
type Project,
|
|
type RoutineVariable,
|
|
} from "@paperclipai/shared";
|
|
import { useQuery } from "@tanstack/react-query";
|
|
import { instanceSettingsApi } from "../api/instanceSettings";
|
|
import { queryKeys } from "../lib/queryKeys";
|
|
import { IssueWorkspaceCard } from "./IssueWorkspaceCard";
|
|
import { AgentIcon } from "./AgentIconPicker";
|
|
import { InlineEntitySelector, type InlineEntityOption } from "./InlineEntitySelector";
|
|
import { getRecentAssigneeIds, sortAgentsByRecency, trackRecentAssignee } from "../lib/recent-assignees";
|
|
import { getRecentProjectIds, trackRecentProject } from "../lib/recent-projects";
|
|
import { Button } from "@/components/ui/button";
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogDescription,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
} from "@/components/ui/dialog";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "@/components/ui/select";
|
|
import { Textarea } from "@/components/ui/textarea";
|
|
|
|
function buildInitialValues(variables: RoutineVariable[]) {
|
|
return Object.fromEntries(variables.map((variable) => [variable.name, variable.defaultValue ?? ""]));
|
|
}
|
|
|
|
function buildInitialRunSelection(input: {
|
|
defaultAssigneeAgentId?: string | null;
|
|
defaultProjectId?: string | null;
|
|
}) {
|
|
return {
|
|
assigneeAgentId: input.defaultAssigneeAgentId ?? "",
|
|
projectId: input.defaultProjectId ?? "",
|
|
};
|
|
}
|
|
|
|
function defaultProjectWorkspaceIdForProject(project: Project | null | undefined) {
|
|
if (!project) return null;
|
|
return project.executionWorkspacePolicy?.defaultProjectWorkspaceId
|
|
?? project.workspaces?.find((workspace) => workspace.isPrimary)?.id
|
|
?? project.workspaces?.[0]?.id
|
|
?? null;
|
|
}
|
|
|
|
function defaultExecutionWorkspaceModeForProject(project: Project | null | undefined) {
|
|
const defaultMode = project?.executionWorkspacePolicy?.enabled ? project.executionWorkspacePolicy.defaultMode : null;
|
|
if (
|
|
defaultMode === "isolated_workspace" ||
|
|
defaultMode === "operator_branch" ||
|
|
defaultMode === "adapter_default"
|
|
) {
|
|
return defaultMode === "adapter_default" ? "agent_default" : defaultMode;
|
|
}
|
|
return "shared_workspace";
|
|
}
|
|
|
|
function buildInitialWorkspaceConfig(project: Project | null | undefined) {
|
|
const defaultMode = defaultExecutionWorkspaceModeForProject(project);
|
|
return {
|
|
executionWorkspaceId: null as string | null,
|
|
executionWorkspacePreference: defaultMode,
|
|
executionWorkspaceSettings: { mode: defaultMode } as IssueExecutionWorkspaceSettings,
|
|
projectWorkspaceId: defaultProjectWorkspaceIdForProject(project),
|
|
};
|
|
}
|
|
|
|
function workspaceConfigEquals(
|
|
a: ReturnType<typeof buildInitialWorkspaceConfig>,
|
|
b: ReturnType<typeof buildInitialWorkspaceConfig>,
|
|
) {
|
|
return a.executionWorkspaceId === b.executionWorkspaceId
|
|
&& a.executionWorkspacePreference === b.executionWorkspacePreference
|
|
&& a.projectWorkspaceId === b.projectWorkspaceId
|
|
&& JSON.stringify(a.executionWorkspaceSettings ?? null) === JSON.stringify(b.executionWorkspaceSettings ?? null);
|
|
}
|
|
|
|
function applyWorkspaceDraft(
|
|
current: ReturnType<typeof buildInitialWorkspaceConfig>,
|
|
data: Record<string, unknown>,
|
|
) {
|
|
const next = {
|
|
...current,
|
|
executionWorkspaceId: (data.executionWorkspaceId as string | null | undefined) ?? null,
|
|
executionWorkspacePreference:
|
|
(data.executionWorkspacePreference as string | null | undefined)
|
|
?? current.executionWorkspacePreference,
|
|
executionWorkspaceSettings:
|
|
(data.executionWorkspaceSettings as IssueExecutionWorkspaceSettings | null | undefined)
|
|
?? current.executionWorkspaceSettings,
|
|
};
|
|
return workspaceConfigEquals(current, next) ? current : next;
|
|
}
|
|
|
|
function isMissingRequiredValue(value: unknown) {
|
|
return value == null || (typeof value === "string" && value.trim().length === 0);
|
|
}
|
|
|
|
function supportsRoutineRunWorkspaceSelection(
|
|
project: Project | null | undefined,
|
|
isolatedWorkspacesEnabled: boolean,
|
|
) {
|
|
return isolatedWorkspacesEnabled && Boolean(project?.executionWorkspacePolicy?.enabled);
|
|
}
|
|
|
|
export function routineRunNeedsConfiguration(input: {
|
|
variables: RoutineVariable[];
|
|
project: Project | null | undefined;
|
|
isolatedWorkspacesEnabled: boolean;
|
|
}) {
|
|
return input.variables.length > 0
|
|
|| supportsRoutineRunWorkspaceSelection(input.project, input.isolatedWorkspacesEnabled);
|
|
}
|
|
|
|
export interface RoutineRunDialogSubmitData {
|
|
variables?: Record<string, string | number | boolean>;
|
|
assigneeAgentId?: string | null;
|
|
projectId?: string | null;
|
|
executionWorkspaceId?: string | null;
|
|
executionWorkspacePreference?: string | null;
|
|
executionWorkspaceSettings?: IssueExecutionWorkspaceSettings | null;
|
|
}
|
|
|
|
export function RoutineRunVariablesDialog({
|
|
open,
|
|
onOpenChange,
|
|
companyId,
|
|
routineName,
|
|
projects,
|
|
agents,
|
|
defaultProjectId,
|
|
defaultAssigneeAgentId,
|
|
variables,
|
|
isPending,
|
|
onSubmit,
|
|
}: {
|
|
open: boolean;
|
|
onOpenChange: (open: boolean) => void;
|
|
companyId: string | null | undefined;
|
|
routineName?: string | null;
|
|
projects: Project[];
|
|
agents: Agent[];
|
|
defaultProjectId?: string | null;
|
|
defaultAssigneeAgentId?: string | null;
|
|
variables: RoutineVariable[];
|
|
isPending: boolean;
|
|
onSubmit: (data: RoutineRunDialogSubmitData) => void;
|
|
}) {
|
|
const [values, setValues] = useState<Record<string, unknown>>({});
|
|
const [selection, setSelection] = useState(() => buildInitialRunSelection({
|
|
defaultAssigneeAgentId,
|
|
defaultProjectId,
|
|
}));
|
|
const selectedProject = useMemo(
|
|
() => projects.find((project) => project.id === selection.projectId) ?? null,
|
|
[projects, selection.projectId],
|
|
);
|
|
const recentAssigneeIds = useMemo(() => getRecentAssigneeIds(), [open]);
|
|
const recentProjectIds = useMemo(() => getRecentProjectIds(), [open]);
|
|
const assigneeOptions = useMemo<InlineEntityOption[]>(
|
|
() =>
|
|
sortAgentsByRecency(
|
|
agents.filter((agent) => agent.status !== "terminated"),
|
|
recentAssigneeIds,
|
|
).map((agent) => ({
|
|
id: agent.id,
|
|
label: agent.name,
|
|
searchText: `${agent.name} ${agent.role} ${agent.title ?? ""}`,
|
|
})),
|
|
[agents, recentAssigneeIds],
|
|
);
|
|
const projectOptions = useMemo<InlineEntityOption[]>(
|
|
() => projects.map((project) => ({
|
|
id: project.id,
|
|
label: project.name,
|
|
searchText: project.description ?? "",
|
|
})),
|
|
[projects],
|
|
);
|
|
const currentAssignee = selection.assigneeAgentId
|
|
? agents.find((agent) => agent.id === selection.assigneeAgentId) ?? null
|
|
: null;
|
|
const [workspaceConfig, setWorkspaceConfig] = useState(() => buildInitialWorkspaceConfig(selectedProject));
|
|
const [workspaceConfigValid, setWorkspaceConfigValid] = useState(true);
|
|
const [workspaceBranchName, setWorkspaceBranchName] = useState<string | null>(null);
|
|
|
|
const { data: experimentalSettings } = useQuery({
|
|
queryKey: queryKeys.instance.experimentalSettings,
|
|
queryFn: () => instanceSettingsApi.getExperimental(),
|
|
retry: false,
|
|
});
|
|
|
|
const workspaceSelectionEnabled = supportsRoutineRunWorkspaceSelection(
|
|
selectedProject,
|
|
experimentalSettings?.enableIsolatedWorkspaces === true,
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (!open) return;
|
|
setValues(buildInitialValues(variables));
|
|
const nextSelection = buildInitialRunSelection({ defaultAssigneeAgentId, defaultProjectId });
|
|
setSelection(nextSelection);
|
|
setWorkspaceConfig(buildInitialWorkspaceConfig(projects.find((project) => project.id === nextSelection.projectId) ?? null));
|
|
setWorkspaceConfigValid(true);
|
|
setWorkspaceBranchName(null);
|
|
}, [defaultAssigneeAgentId, defaultProjectId, open, projects, variables]);
|
|
|
|
const workspaceBranchAutoValue = workspaceSelectionEnabled && workspaceBranchName
|
|
? workspaceBranchName
|
|
: null;
|
|
|
|
const isAutoWorkspaceBranchVariable = useCallback(
|
|
(variable: RoutineVariable) =>
|
|
variable.name === WORKSPACE_BRANCH_ROUTINE_VARIABLE && Boolean(workspaceBranchAutoValue),
|
|
[workspaceBranchAutoValue],
|
|
);
|
|
|
|
const missingRequired = useMemo(
|
|
() =>
|
|
variables
|
|
.filter((variable) => variable.required)
|
|
.filter((variable) => !isAutoWorkspaceBranchVariable(variable))
|
|
.filter((variable) => isMissingRequiredValue(values[variable.name]))
|
|
.map((variable) => variable.label || variable.name),
|
|
[isAutoWorkspaceBranchVariable, values, variables],
|
|
);
|
|
|
|
const workspaceIssue = useMemo(() => ({
|
|
companyId: companyId ?? null,
|
|
projectId: selectedProject?.id ?? null,
|
|
projectWorkspaceId: workspaceConfig.projectWorkspaceId,
|
|
executionWorkspaceId: workspaceConfig.executionWorkspaceId,
|
|
executionWorkspacePreference: workspaceConfig.executionWorkspacePreference,
|
|
executionWorkspaceSettings: workspaceConfig.executionWorkspaceSettings,
|
|
currentExecutionWorkspace: null,
|
|
}), [
|
|
companyId,
|
|
selectedProject?.id,
|
|
workspaceConfig.executionWorkspaceId,
|
|
workspaceConfig.executionWorkspacePreference,
|
|
workspaceConfig.executionWorkspaceSettings,
|
|
workspaceConfig.projectWorkspaceId,
|
|
]);
|
|
|
|
const canSubmit =
|
|
selection.assigneeAgentId.trim().length > 0 &&
|
|
missingRequired.length === 0 &&
|
|
(!workspaceSelectionEnabled || workspaceConfigValid);
|
|
|
|
const handleWorkspaceUpdate = useCallback((data: Record<string, unknown>) => {
|
|
setWorkspaceConfig((current) => applyWorkspaceDraft(current, data));
|
|
}, []);
|
|
|
|
const handleWorkspaceDraftChange = useCallback((
|
|
data: Record<string, unknown>,
|
|
meta: { canSave: boolean; workspaceBranchName?: string | null },
|
|
) => {
|
|
setWorkspaceConfig((current) => applyWorkspaceDraft(current, data));
|
|
setWorkspaceConfigValid((current) => (current === meta.canSave ? current : meta.canSave));
|
|
setWorkspaceBranchName((current) => {
|
|
const next = meta.workspaceBranchName ?? null;
|
|
return current === next ? current : next;
|
|
});
|
|
}, []);
|
|
|
|
return (
|
|
<Dialog open={open} onOpenChange={(next) => !isPending && onOpenChange(next)}>
|
|
<DialogContent className="max-w-xl">
|
|
<DialogHeader>
|
|
{routineName && (
|
|
<p className="text-muted-foreground text-sm">{routineName}</p>
|
|
)}
|
|
<DialogTitle>Run routine</DialogTitle>
|
|
<DialogDescription>
|
|
Choose the agent and optional project for this one run. Routine defaults are prefilled and won't be changed.
|
|
</DialogDescription>
|
|
</DialogHeader>
|
|
|
|
<div className="space-y-4">
|
|
<div className="grid gap-4 md:grid-cols-2">
|
|
<div className="space-y-1.5">
|
|
<Label className="text-xs">Agent *</Label>
|
|
<InlineEntitySelector
|
|
value={selection.assigneeAgentId}
|
|
options={assigneeOptions}
|
|
recentOptionIds={recentAssigneeIds}
|
|
placeholder="Agent"
|
|
noneLabel="Select an agent"
|
|
searchPlaceholder="Search agents..."
|
|
emptyMessage="No agents found."
|
|
disablePortal
|
|
openOnFocus={false}
|
|
onChange={(assigneeAgentId) => {
|
|
if (assigneeAgentId) trackRecentAssignee(assigneeAgentId);
|
|
setSelection((current) => ({ ...current, assigneeAgentId }));
|
|
}}
|
|
renderTriggerValue={(option) =>
|
|
option ? (
|
|
currentAssignee ? (
|
|
<>
|
|
<AgentIcon icon={currentAssignee.icon} className="h-3.5 w-3.5 shrink-0 text-muted-foreground" />
|
|
<span className="truncate">{option.label}</span>
|
|
</>
|
|
) : (
|
|
<span className="truncate">{option.label}</span>
|
|
)
|
|
) : (
|
|
<span className="text-muted-foreground">Select an agent</span>
|
|
)
|
|
}
|
|
renderOption={(option) => {
|
|
if (!option.id) return <span className="truncate">{option.label}</span>;
|
|
const assignee = agents.find((agent) => agent.id === option.id);
|
|
return (
|
|
<>
|
|
{assignee ? <AgentIcon icon={assignee.icon} className="h-3.5 w-3.5 shrink-0 text-muted-foreground" /> : null}
|
|
<span className="truncate">{option.label}</span>
|
|
</>
|
|
);
|
|
}}
|
|
/>
|
|
</div>
|
|
<div className="space-y-1.5">
|
|
<Label className="text-xs">Project</Label>
|
|
<InlineEntitySelector
|
|
value={selection.projectId}
|
|
options={projectOptions}
|
|
recentOptionIds={recentProjectIds}
|
|
placeholder="Project"
|
|
noneLabel="No project"
|
|
searchPlaceholder="Search projects..."
|
|
emptyMessage="No projects found."
|
|
disablePortal
|
|
openOnFocus={false}
|
|
onChange={(projectId) => {
|
|
const project = projects.find((entry) => entry.id === projectId) ?? null;
|
|
if (projectId) trackRecentProject(projectId);
|
|
setSelection((current) => ({ ...current, projectId }));
|
|
setWorkspaceConfig(buildInitialWorkspaceConfig(project));
|
|
setWorkspaceConfigValid(true);
|
|
setWorkspaceBranchName(null);
|
|
}}
|
|
renderTriggerValue={(option) =>
|
|
option && selectedProject ? (
|
|
<>
|
|
<span
|
|
className="h-3.5 w-3.5 shrink-0 rounded-sm"
|
|
style={{ backgroundColor: selectedProject.color ?? "#64748b" }}
|
|
/>
|
|
<span className="truncate">{option.label}</span>
|
|
</>
|
|
) : (
|
|
<span className="text-muted-foreground">No project</span>
|
|
)
|
|
}
|
|
renderOption={(option) => {
|
|
if (!option.id) return <span className="truncate">{option.label}</span>;
|
|
const project = projects.find((entry) => entry.id === option.id);
|
|
return (
|
|
<>
|
|
<span
|
|
className="h-3.5 w-3.5 shrink-0 rounded-sm"
|
|
style={{ backgroundColor: project?.color ?? "#64748b" }}
|
|
/>
|
|
<span className="truncate">{option.label}</span>
|
|
</>
|
|
);
|
|
}}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
{variables.map((variable) => (
|
|
<div key={variable.name} className="space-y-1.5">
|
|
<Label className="text-xs">
|
|
{variable.label || variable.name}
|
|
{variable.required ? " *" : ""}
|
|
</Label>
|
|
{isAutoWorkspaceBranchVariable(variable) ? (
|
|
<Input
|
|
readOnly
|
|
disabled
|
|
value={workspaceBranchAutoValue ?? ""}
|
|
/>
|
|
) : variable.type === "textarea" ? (
|
|
<Textarea
|
|
rows={4}
|
|
value={typeof values[variable.name] === "string" ? values[variable.name] as string : ""}
|
|
onChange={(event) => setValues((current) => ({ ...current, [variable.name]: event.target.value }))}
|
|
/>
|
|
) : variable.type === "boolean" ? (
|
|
<Select
|
|
value={values[variable.name] === true ? "true" : values[variable.name] === false ? "false" : "__unset__"}
|
|
onValueChange={(next) => setValues((current) => ({
|
|
...current,
|
|
[variable.name]: next === "__unset__" ? "" : next === "true",
|
|
}))}
|
|
>
|
|
<SelectTrigger>
|
|
<SelectValue />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectItem value="__unset__">No value</SelectItem>
|
|
<SelectItem value="true">True</SelectItem>
|
|
<SelectItem value="false">False</SelectItem>
|
|
</SelectContent>
|
|
</Select>
|
|
) : variable.type === "select" ? (
|
|
<Select
|
|
value={typeof values[variable.name] === "string" && values[variable.name] ? values[variable.name] as string : "__unset__"}
|
|
onValueChange={(next) => setValues((current) => ({
|
|
...current,
|
|
[variable.name]: next === "__unset__" ? "" : next,
|
|
}))}
|
|
>
|
|
<SelectTrigger>
|
|
<SelectValue placeholder="Choose a value" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectItem value="__unset__">No value</SelectItem>
|
|
{variable.options.map((option) => (
|
|
<SelectItem key={option} value={option}>{option}</SelectItem>
|
|
))}
|
|
</SelectContent>
|
|
</Select>
|
|
) : (
|
|
<Input
|
|
type={variable.type === "number" ? "number" : "text"}
|
|
value={values[variable.name] == null ? "" : String(values[variable.name])}
|
|
onChange={(event) => setValues((current) => ({ ...current, [variable.name]: event.target.value }))}
|
|
/>
|
|
)}
|
|
</div>
|
|
))}
|
|
|
|
{workspaceSelectionEnabled && selectedProject && companyId ? (
|
|
<IssueWorkspaceCard
|
|
key={`${open ? "open" : "closed"}:${selectedProject.id}`}
|
|
issue={workspaceIssue}
|
|
project={selectedProject}
|
|
initialEditing
|
|
livePreview
|
|
onUpdate={handleWorkspaceUpdate}
|
|
onDraftChange={handleWorkspaceDraftChange}
|
|
/>
|
|
) : null}
|
|
</div>
|
|
|
|
<DialogFooter showCloseButton={false}>
|
|
{!selection.assigneeAgentId ? (
|
|
<p className="mr-auto text-xs text-amber-600">Default agent required for this run.</p>
|
|
) : missingRequired.length > 0 ? (
|
|
<p className="mr-auto text-xs text-amber-600">
|
|
Missing: {missingRequired.join(", ")}
|
|
</p>
|
|
) : workspaceSelectionEnabled && !workspaceConfigValid ? (
|
|
<p className="mr-auto text-xs text-amber-600">
|
|
Choose an existing workspace before running.
|
|
</p>
|
|
) : (
|
|
<span className="mr-auto" />
|
|
)}
|
|
<Button variant="ghost" onClick={() => onOpenChange(false)} disabled={isPending}>
|
|
Cancel
|
|
</Button>
|
|
<Button
|
|
onClick={() => {
|
|
const nextVariables: Record<string, string | number | boolean> = {};
|
|
for (const variable of variables) {
|
|
if (isAutoWorkspaceBranchVariable(variable)) {
|
|
nextVariables[variable.name] = workspaceBranchAutoValue!;
|
|
continue;
|
|
}
|
|
const rawValue = values[variable.name];
|
|
if (isMissingRequiredValue(rawValue)) continue;
|
|
if (variable.type === "number") {
|
|
nextVariables[variable.name] = Number(rawValue);
|
|
} else if (variable.type === "boolean") {
|
|
nextVariables[variable.name] = rawValue === true;
|
|
} else {
|
|
nextVariables[variable.name] = String(rawValue);
|
|
}
|
|
}
|
|
onSubmit({
|
|
variables: nextVariables,
|
|
assigneeAgentId: selection.assigneeAgentId,
|
|
projectId: selection.projectId || null,
|
|
...(workspaceSelectionEnabled
|
|
? {
|
|
executionWorkspaceId: workspaceConfig.executionWorkspaceId,
|
|
executionWorkspacePreference: workspaceConfig.executionWorkspacePreference,
|
|
executionWorkspaceSettings: workspaceConfig.executionWorkspaceSettings,
|
|
}
|
|
: {}),
|
|
});
|
|
}}
|
|
disabled={isPending || !canSubmit}
|
|
>
|
|
{isPending ? "Running..." : "Run routine"}
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
);
|
|
}
|