d9f91576a0
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies, so planning approvals and child-issue fan-out are part of the core control-plane loop. > - Accepted plans are supposed to be a safe bridge from planning into execution, especially when agents wake from review decisions and reuse isolated workspaces. > - The duplicate-subtask incident showed that an accepted plan revision could be interpreted more than once across overlapping runs, which broke the single-source-of-truth model for issue decomposition. > - Fixing that required tightening the backend contract first: accepted-plan decomposition needs an exact-once fingerprint, durable claim state, and retry-safe child creation. > - Once that backend behavior existed, the board still needed visibility into what happened, so the issue detail view needed a dedicated decomposition section instead of forcing operators to reconstruct child creation from raw activity. > - This pull request adds the exact-once decomposition primitive, hardens wake routing and regressions around the incident, and surfaces decomposition state in the UI so future incidents are both prevented and easier to inspect. ## What Changed - Added accepted-plan decomposition semantics to `doc/execution-semantics.md`, including the exact-once fingerprint, durable claim/result expectations, and retry/resume behavior. - Added persistent accepted-plan decomposition claims in the backend, including schema, shared types/validators, service logic, and issue routes for creating and listing decomposition state. - Hardened heartbeat routing so an accepted-plan continuation stays scoped to the relevant planning issue instead of opportunistically re-decomposing another accepted issue on the same assignee. - Added regression coverage for the original failure modes: concurrent same-parent retries, cross-issue accepted-plan isolation, and partial child recreation under the same fingerprint. - Added the `Plan decomposition` issue-detail section plus supporting API/query-key/activity formatting updates so operators can see revision status, owner, child counts, and the linked child issues directly in the UI. - Included the small follow-up UI fix so the decomposition section still renders when the issue work mode is no longer `planning`. ## Verification - `pnpm --filter @paperclipai/server typecheck` - `pnpm --filter @paperclipai/ui typecheck` - `pnpm --filter @paperclipai/db typecheck` - `pnpm exec vitest run server/src/__tests__/issues-service.test.ts` - `pnpm exec vitest run server/src/__tests__/issues-service.test.ts -t "lists persisted decompositions with child issue summaries"` - `pnpm exec vitest run server/src/__tests__/issues-service.test.ts -t "accepted plan decomposition" server/src/__tests__/heartbeat-accepted-plan-workspace-refresh.test.ts server/src/__tests__/heartbeat-context-summary.test.ts` - Manual UI path: create a planning issue without an isolated execution workspace, add a `plan` document, accept the `request_confirmation`, let Paperclip create child issues, then reopen the parent issue detail page and confirm the `Plan decomposition` section shows the accepted revision, status, idempotent-claim badge, and child links. - Separate follow-up bug noted during manual UI validation: accepting a plan on an issue whose run never records `workspace_finalize` is tracked in `PAPA-445` and is not part of this PR’s fix scope. ## Risks - This adds a new migration and a large Drizzle snapshot update; reviewers should confirm the schema shape and generated metadata match the intended decomposition table. - The exact-once claim changes sit on the accepted-plan fan-out path, so regressions there could block legitimate child creation or mis-handle retries if the claim state machine is wrong. - The new UI only appears when decomposition records exist; reviewers should use the manual verification path above rather than expecting existing issues on a stale local instance to show the section automatically. - `PAPA-445` remains an open follow-up for the `workspace_finalize` accept gate when a planning handoff never records finalize; that bug can interfere with reproducing the UI flow on isolated workspaces but does not change the correctness of the exact-once decomposition feature itself. > Checked `ROADMAP.md`: this PR is a bug fix / control-plane hardening change for accepted-plan decomposition, not a new uncoordinated roadmap feature. ## Model Used - OpenAI Codex via Paperclip `codex_local` (GPT-5-based coding agent; exact backend model ID/context window not exposed in the run context), with repository tool use, shell execution, and code-editing capabilities. <img width="806" height="1069" alt="Screenshot 2026-05-27 at 11 05 48 PM" src="https://github.com/user-attachments/assets/5b00b670-96cd-4470-b0a3-581743bcae28" /> ## 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 - [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>
479 lines
14 KiB
TypeScript
479 lines
14 KiB
TypeScript
export {
|
|
instanceGeneralSettingsSchema,
|
|
patchInstanceGeneralSettingsSchema,
|
|
type InstanceGeneralSettings,
|
|
type PatchInstanceGeneralSettings,
|
|
instanceExperimentalSettingsSchema,
|
|
patchInstanceExperimentalSettingsSchema,
|
|
issueGraphLivenessAutoRecoveryRequestSchema,
|
|
type InstanceExperimentalSettings,
|
|
type PatchInstanceExperimentalSettings,
|
|
type IssueGraphLivenessAutoRecoveryRequest,
|
|
} from "./instance.js";
|
|
|
|
export {
|
|
upsertBudgetPolicySchema,
|
|
resolveBudgetIncidentSchema,
|
|
type UpsertBudgetPolicy,
|
|
type ResolveBudgetIncident,
|
|
} from "./budget.js";
|
|
|
|
export {
|
|
createCompanySchema,
|
|
updateCompanySchema,
|
|
updateCompanyBrandingSchema,
|
|
type CreateCompany,
|
|
type UpdateCompany,
|
|
type UpdateCompanyBranding,
|
|
} from "./company.js";
|
|
export {
|
|
environmentDriverSchema,
|
|
environmentStatusSchema,
|
|
environmentLeaseStatusSchema,
|
|
environmentLeaseCleanupStatusSchema,
|
|
createEnvironmentSchema,
|
|
updateEnvironmentSchema,
|
|
probeEnvironmentConfigSchema,
|
|
type CreateEnvironment,
|
|
type UpdateEnvironment,
|
|
type ProbeEnvironmentConfig,
|
|
} from "./environment.js";
|
|
export {
|
|
feedbackDataSharingPreferenceSchema,
|
|
feedbackTargetTypeSchema,
|
|
feedbackTraceStatusSchema,
|
|
feedbackVoteValueSchema,
|
|
upsertIssueFeedbackVoteSchema,
|
|
type UpsertIssueFeedbackVote,
|
|
} from "./feedback.js";
|
|
export {
|
|
sidebarOrderPreferenceSchema,
|
|
upsertSidebarOrderPreferenceSchema,
|
|
type UpsertSidebarOrderPreference,
|
|
} from "./sidebar-preferences.js";
|
|
export {
|
|
resourceMembershipStateSchema,
|
|
updateResourceMembershipSchema,
|
|
type UpdateResourceMembership,
|
|
} from "./resource-memberships.js";
|
|
export {
|
|
companySkillSourceTypeSchema,
|
|
companySkillTrustLevelSchema,
|
|
companySkillCompatibilitySchema,
|
|
companySkillSourceBadgeSchema,
|
|
companySkillFileInventoryEntrySchema,
|
|
companySkillSchema,
|
|
companySkillListItemSchema,
|
|
companySkillUsageAgentSchema,
|
|
companySkillDetailSchema,
|
|
companySkillUpdateStatusSchema,
|
|
companySkillAuditFindingSchema,
|
|
companySkillAuditResultSchema,
|
|
companySkillImportSchema,
|
|
companySkillProjectScanRequestSchema,
|
|
companySkillProjectScanSkippedSchema,
|
|
companySkillProjectScanConflictSchema,
|
|
companySkillProjectScanResultSchema,
|
|
companySkillCreateSchema,
|
|
companySkillFileDetailSchema,
|
|
companySkillFileUpdateSchema,
|
|
catalogSkillKindSchema,
|
|
catalogSkillFileSchema,
|
|
catalogSkillSchema,
|
|
catalogSkillListQuerySchema,
|
|
catalogSkillFileDetailSchema,
|
|
companySkillInstallCatalogSchema,
|
|
companySkillInstallCatalogResultSchema,
|
|
companySkillInstallUpdateSchema,
|
|
companySkillResetSchema,
|
|
type CompanySkillImport,
|
|
type CompanySkillProjectScan,
|
|
type CompanySkillCreate,
|
|
type CompanySkillFileUpdate,
|
|
type CatalogSkillListQuery,
|
|
type CompanySkillInstallCatalog,
|
|
type CompanySkillInstallUpdate,
|
|
type CompanySkillReset,
|
|
} from "./company-skill.js";
|
|
export {
|
|
agentSkillStateSchema,
|
|
agentSkillSyncModeSchema,
|
|
agentSkillEntrySchema,
|
|
agentSkillSnapshotSchema,
|
|
agentSkillSyncSchema,
|
|
type AgentSkillSync,
|
|
} from "./adapter-skills.js";
|
|
export {
|
|
portabilityIncludeSchema,
|
|
portabilityEnvInputSchema,
|
|
portabilityCompanyManifestEntrySchema,
|
|
portabilitySidebarOrderSchema,
|
|
portabilityAgentManifestEntrySchema,
|
|
portabilitySkillManifestEntrySchema,
|
|
portabilityManifestSchema,
|
|
portabilitySourceSchema,
|
|
portabilityTargetSchema,
|
|
portabilityAgentSelectionSchema,
|
|
portabilityCollisionStrategySchema,
|
|
companyPortabilityExportSchema,
|
|
companyPortabilityPreviewSchema,
|
|
companyPortabilityImportSchema,
|
|
type CompanyPortabilityExport,
|
|
type CompanyPortabilityPreview,
|
|
type CompanyPortabilityImport,
|
|
} from "./company-portability.js";
|
|
|
|
export {
|
|
createAgentSchema,
|
|
createAgentHireSchema,
|
|
updateAgentSchema,
|
|
agentRuntimeConfigSchema,
|
|
agentInstructionsBundleModeSchema,
|
|
updateAgentInstructionsBundleSchema,
|
|
upsertAgentInstructionsFileSchema,
|
|
updateAgentInstructionsPathSchema,
|
|
createAgentKeySchema,
|
|
agentMineInboxQuerySchema,
|
|
wakeAgentSchema,
|
|
resetAgentSessionSchema,
|
|
testAdapterEnvironmentSchema,
|
|
agentPermissionsSchema,
|
|
updateAgentPermissionsSchema,
|
|
type CreateAgent,
|
|
type CreateAgentHire,
|
|
type UpdateAgent,
|
|
type UpdateAgentInstructionsBundle,
|
|
type UpsertAgentInstructionsFile,
|
|
type UpdateAgentInstructionsPath,
|
|
type CreateAgentKey,
|
|
type AgentMineInboxQuery,
|
|
type WakeAgent,
|
|
type ResetAgentSession,
|
|
type TestAdapterEnvironment,
|
|
type UpdateAgentPermissions,
|
|
} from "./agent.js";
|
|
|
|
export {
|
|
createProjectSchema,
|
|
updateProjectSchema,
|
|
createProjectWorkspaceSchema,
|
|
updateProjectWorkspaceSchema,
|
|
projectExecutionWorkspacePolicySchema,
|
|
projectWorkspaceRuntimeConfigSchema,
|
|
type CreateProject,
|
|
type UpdateProject,
|
|
type CreateProjectWorkspace,
|
|
type UpdateProjectWorkspace,
|
|
type ProjectExecutionWorkspacePolicy,
|
|
} from "./project.js";
|
|
|
|
export {
|
|
createDocumentAnnotationCommentSchema,
|
|
createDocumentAnnotationThreadSchema,
|
|
documentAnnotationAnchorConfidenceSchema,
|
|
documentAnnotationAnchorSelectorSchema,
|
|
documentAnnotationAnchorStateSchema,
|
|
documentAnnotationTextPositionSelectorSchema,
|
|
documentAnnotationTextQuoteSelectorSchema,
|
|
documentAnnotationThreadStatusSchema,
|
|
updateDocumentAnnotationThreadSchema,
|
|
type CreateDocumentAnnotationComment,
|
|
type CreateDocumentAnnotationThread,
|
|
type UpdateDocumentAnnotationThread,
|
|
} from "./document-annotation.js";
|
|
|
|
export {
|
|
createIssueSchema,
|
|
createIssueInputSchema,
|
|
createChildIssueSchema,
|
|
createAcceptedPlanDecompositionSchema,
|
|
resolveCreateIssueStatusDefault,
|
|
createIssueLabelSchema,
|
|
issueBlockedInboxAttentionSchema,
|
|
issueBlockedInboxIssueRefSchema,
|
|
issueBlockedInboxReasonSchema,
|
|
issueBlockedInboxSeveritySchema,
|
|
issueBlockedInboxStateSchema,
|
|
updateIssueSchema,
|
|
issueExecutionPolicySchema,
|
|
issueExecutionStateSchema,
|
|
issueRecoveryActionReadModelSchema,
|
|
resolveIssueRecoveryActionSchema,
|
|
issueReviewRequestSchema,
|
|
issueExecutionWorkspaceSettingsSchema,
|
|
checkoutIssueSchema,
|
|
issueCommentAuthorTypeSchema,
|
|
issueCommentPresentationSchema,
|
|
issueCommentMetadataRowSchema,
|
|
issueCommentMetadataSectionSchema,
|
|
issueCommentMetadataSchema,
|
|
addIssueCommentSchema,
|
|
issueThreadInteractionStatusSchema,
|
|
issueThreadInteractionKindSchema,
|
|
issueThreadInteractionContinuationPolicySchema,
|
|
suggestedTaskDraftSchema,
|
|
suggestTasksPayloadSchema,
|
|
suggestTasksResultCreatedTaskSchema,
|
|
suggestTasksResultSchema,
|
|
askUserQuestionsQuestionOptionSchema,
|
|
askUserQuestionsQuestionSchema,
|
|
askUserQuestionsPayloadSchema,
|
|
askUserQuestionsAnswerSchema,
|
|
askUserQuestionsResultSchema,
|
|
requestConfirmationIssueDocumentTargetSchema,
|
|
requestConfirmationCustomTargetSchema,
|
|
requestConfirmationTargetSchema,
|
|
requestConfirmationPayloadSchema,
|
|
requestConfirmationResultSchema,
|
|
createIssueThreadInteractionSchema,
|
|
acceptIssueThreadInteractionSchema,
|
|
rejectIssueThreadInteractionSchema,
|
|
cancelIssueThreadInteractionSchema,
|
|
respondIssueThreadInteractionSchema,
|
|
linkIssueApprovalSchema,
|
|
createIssueAttachmentMetadataSchema,
|
|
issueDocumentFormatSchema,
|
|
issueDocumentKeySchema,
|
|
upsertIssueDocumentSchema,
|
|
restoreIssueDocumentRevisionSchema,
|
|
type CreateIssue,
|
|
type CreateChildIssue,
|
|
type CreateAcceptedPlanDecomposition,
|
|
type CreateIssueLabel,
|
|
type UpdateIssue,
|
|
type IssueExecutionWorkspaceSettings,
|
|
type IssueRecoveryActionReadModel,
|
|
type ResolveIssueRecoveryAction,
|
|
type CheckoutIssue,
|
|
type AddIssueComment,
|
|
type CreateIssueThreadInteraction,
|
|
type AcceptIssueThreadInteraction,
|
|
type RejectIssueThreadInteraction,
|
|
type CancelIssueThreadInteraction,
|
|
type RespondIssueThreadInteraction,
|
|
type LinkIssueApproval,
|
|
type CreateIssueAttachmentMetadata,
|
|
type IssueDocumentFormat,
|
|
type UpsertIssueDocument,
|
|
type RestoreIssueDocumentRevision,
|
|
} from "./issue.js";
|
|
|
|
export {
|
|
COMPANY_SEARCH_DEFAULT_LIMIT,
|
|
COMPANY_SEARCH_MAX_LIMIT,
|
|
COMPANY_SEARCH_MAX_OFFSET,
|
|
COMPANY_SEARCH_MAX_QUERY_LENGTH,
|
|
COMPANY_SEARCH_MAX_TOKENS,
|
|
companySearchQuerySchema,
|
|
type CompanySearchQuery,
|
|
} from "./search.js";
|
|
|
|
export {
|
|
createIssueTreeHoldSchema,
|
|
issueTreeControlModeSchema,
|
|
issueTreeHoldReleasePolicySchema,
|
|
previewIssueTreeControlSchema,
|
|
releaseIssueTreeHoldSchema,
|
|
type CreateIssueTreeHold,
|
|
type PreviewIssueTreeControl,
|
|
type ReleaseIssueTreeHold,
|
|
} from "./issue-tree-control.js";
|
|
|
|
export {
|
|
createIssueWorkProductSchema,
|
|
updateIssueWorkProductSchema,
|
|
issueWorkProductTypeSchema,
|
|
issueWorkProductStatusSchema,
|
|
issueWorkProductReviewStateSchema,
|
|
type CreateIssueWorkProduct,
|
|
type UpdateIssueWorkProduct,
|
|
} from "./work-product.js";
|
|
|
|
export {
|
|
executionWorkspaceConfigSchema,
|
|
updateExecutionWorkspaceSchema,
|
|
executionWorkspaceStatusSchema,
|
|
executionWorkspaceCloseActionKindSchema,
|
|
executionWorkspaceCloseActionSchema,
|
|
executionWorkspaceCloseGitReadinessSchema,
|
|
executionWorkspaceCloseLinkedIssueSchema,
|
|
executionWorkspaceCloseReadinessSchema,
|
|
executionWorkspaceCloseReadinessStateSchema,
|
|
type UpdateExecutionWorkspace,
|
|
} from "./execution-workspace.js";
|
|
|
|
export {
|
|
createGoalSchema,
|
|
updateGoalSchema,
|
|
type CreateGoal,
|
|
type UpdateGoal,
|
|
} from "./goal.js";
|
|
|
|
export {
|
|
createApprovalSchema,
|
|
resolveApprovalSchema,
|
|
requestApprovalRevisionSchema,
|
|
resubmitApprovalSchema,
|
|
addApprovalCommentSchema,
|
|
type CreateApproval,
|
|
type ResolveApproval,
|
|
type RequestApprovalRevision,
|
|
type ResubmitApproval,
|
|
type AddApprovalComment,
|
|
} from "./approval.js";
|
|
|
|
export {
|
|
envBindingPlainSchema,
|
|
envBindingSecretRefSchema,
|
|
envBindingSchema,
|
|
envConfigSchema,
|
|
createSecretSchema,
|
|
createSecretProviderConfigSchema,
|
|
updateSecretProviderConfigSchema,
|
|
secretProviderConfigDiscoveryPreviewSchema,
|
|
remoteSecretImportPreviewSchema,
|
|
remoteSecretImportSchema,
|
|
remoteSecretImportSelectionSchema,
|
|
localEncryptedProviderConfigSchema,
|
|
awsSecretsManagerProviderConfigSchema,
|
|
gcpSecretManagerProviderConfigSchema,
|
|
vaultProviderConfigSchema,
|
|
secretProviderConfigPayloadSchema,
|
|
createSecretBindingSchema,
|
|
rotateSecretSchema,
|
|
secretBindingTargetSchema,
|
|
updateSecretSchema,
|
|
type CreateSecretBinding,
|
|
type CreateSecret,
|
|
type CreateSecretProviderConfig,
|
|
type UpdateSecretProviderConfig,
|
|
type SecretProviderConfigDiscoveryPreview,
|
|
type RemoteSecretImportPreview,
|
|
type RemoteSecretImport,
|
|
type RemoteSecretImportSelection,
|
|
type RotateSecret,
|
|
type UpdateSecret,
|
|
} from "./secret.js";
|
|
|
|
export {
|
|
createRoutineSchema,
|
|
updateRoutineSchema,
|
|
createRoutineTriggerSchema,
|
|
updateRoutineTriggerSchema,
|
|
routineVariableSchema,
|
|
routineRevisionSnapshotRoutineV1Schema,
|
|
routineRevisionSnapshotTriggerV1Schema,
|
|
routineRevisionSnapshotV1Schema,
|
|
routineRevisionSnapshotSchema,
|
|
runRoutineSchema,
|
|
rotateRoutineTriggerSecretSchema,
|
|
type CreateRoutine,
|
|
type UpdateRoutine,
|
|
type CreateRoutineTrigger,
|
|
type UpdateRoutineTrigger,
|
|
type RunRoutine,
|
|
type RotateRoutineTriggerSecret,
|
|
} from "./routine.js";
|
|
|
|
export {
|
|
createCostEventSchema,
|
|
updateBudgetSchema,
|
|
type CreateCostEvent,
|
|
type UpdateBudget,
|
|
} from "./cost.js";
|
|
|
|
export {
|
|
createFinanceEventSchema,
|
|
type CreateFinanceEvent,
|
|
} from "./finance.js";
|
|
|
|
export {
|
|
createAssetImageMetadataSchema,
|
|
type CreateAssetImageMetadata,
|
|
} from "./asset.js";
|
|
|
|
export {
|
|
createCompanyInviteSchema,
|
|
createOpenClawInvitePromptSchema,
|
|
acceptInviteSchema,
|
|
listCompanyInvitesQuerySchema,
|
|
listJoinRequestsQuerySchema,
|
|
claimJoinRequestApiKeySchema,
|
|
boardCliAuthAccessLevelSchema,
|
|
createCliAuthChallengeSchema,
|
|
resolveCliAuthChallengeSchema,
|
|
currentUserProfileSchema,
|
|
authSessionSchema,
|
|
updateCurrentUserProfileSchema,
|
|
updateCompanyMemberSchema,
|
|
updateCompanyMemberWithPermissionsSchema,
|
|
archiveCompanyMemberSchema,
|
|
updateMemberPermissionsSchema,
|
|
searchAdminUsersQuerySchema,
|
|
updateUserCompanyAccessSchema,
|
|
type CreateCompanyInvite,
|
|
type CreateOpenClawInvitePrompt,
|
|
type AcceptInvite,
|
|
type ListCompanyInvitesQuery,
|
|
type ListJoinRequestsQuery,
|
|
type ClaimJoinRequestApiKey,
|
|
type BoardCliAuthAccessLevel,
|
|
type CreateCliAuthChallenge,
|
|
type ResolveCliAuthChallenge,
|
|
type CurrentUserProfile,
|
|
type AuthSession,
|
|
type UpdateCurrentUserProfile,
|
|
type UpdateCompanyMember,
|
|
type UpdateCompanyMemberWithPermissions,
|
|
type ArchiveCompanyMember,
|
|
type UpdateMemberPermissions,
|
|
type SearchAdminUsersQuery,
|
|
type UpdateUserCompanyAccess,
|
|
} from "./access.js";
|
|
|
|
export {
|
|
jsonSchemaSchema,
|
|
pluginJobDeclarationSchema,
|
|
pluginWebhookDeclarationSchema,
|
|
pluginToolDeclarationSchema,
|
|
pluginEnvironmentDriverDeclarationSchema,
|
|
pluginUiSlotDeclarationSchema,
|
|
pluginLauncherActionDeclarationSchema,
|
|
pluginLauncherRenderDeclarationSchema,
|
|
pluginLauncherDeclarationSchema,
|
|
pluginDatabaseDeclarationSchema,
|
|
pluginManagedSkillFileDeclarationSchema,
|
|
pluginManagedSkillDeclarationSchema,
|
|
pluginApiRouteDeclarationSchema,
|
|
pluginManifestV1Schema,
|
|
installPluginSchema,
|
|
upsertPluginConfigSchema,
|
|
patchPluginConfigSchema,
|
|
updatePluginStatusSchema,
|
|
uninstallPluginSchema,
|
|
pluginStateScopeKeySchema,
|
|
setPluginStateSchema,
|
|
listPluginStateSchema,
|
|
type PluginJobDeclarationInput,
|
|
type PluginWebhookDeclarationInput,
|
|
type PluginToolDeclarationInput,
|
|
type PluginEnvironmentDriverDeclarationInput,
|
|
type PluginUiSlotDeclarationInput,
|
|
type PluginLauncherActionDeclarationInput,
|
|
type PluginLauncherRenderDeclarationInput,
|
|
type PluginLauncherDeclarationInput,
|
|
type PluginDatabaseDeclarationInput,
|
|
type PluginManagedSkillFileDeclarationInput,
|
|
type PluginManagedSkillDeclarationInput,
|
|
type PluginApiRouteDeclarationInput,
|
|
type PluginManifestV1Input,
|
|
type InstallPlugin,
|
|
type UpsertPluginConfig,
|
|
type PatchPluginConfig,
|
|
type UpdatePluginStatus,
|
|
type UninstallPlugin,
|
|
type PluginStateScopeKey,
|
|
type SetPluginState,
|
|
type ListPluginState,
|
|
} from "./plugin.js";
|