Skip to content

DT-4001 - standalone nexus operations#3496

Draft
rossedfort wants to merge 10 commits into
mainfrom
DT-4001-SANO
Draft

DT-4001 - standalone nexus operations#3496
rossedfort wants to merge 10 commits into
mainfrom
DT-4001-SANO

Conversation

@rossedfort

Copy link
Copy Markdown
Contributor

Description & motivation 💭

Screenshots (if applicable) 📸

Design Considerations 🎨

Testing 🧪

How was this tested 👻

  • Manual testing
  • E2E tests added
  • Unit tests added

Steps for others to test: 🚶🏽‍♂️🚶🏽‍♀️

Checklists

Draft Checklist

Merge Checklist

Issue(s) closed

Docs

Any docs updates needed?

rossedfort and others added 4 commits May 27, 2026 14:57
…rations (#3444)

* feat(nexus-operations): add types and API routes for standalone nexus operations

Adds NexusOperationExecution types, extends APIRouteParameters with operationId,
and registers 6 new API routes (list, describe/start, poll, cancel, terminate, count).
Adds route-for helpers mirroring the standalone activities pattern.

DT-4005

* feat(nexus-operations): add services for standalone nexus operations

Adds fetchPaginatedNexusOperations, startStandaloneNexusOperation,
getNexusOperationExecution, pollNexusOperationExecution, cancel, and terminate.
Adds count and count-by-status service functions.

DT-4005

* feat(nexus-operations): add stores, poller, and filters for standalone nexus operations

Adds nexus operation writable stores (refresh, loading, updating, count, error, query).
Adds StandaloneNexusOperationPoller with long-poll loop mirroring the activity poller.
Adds nexusOperationFilters store to filters.ts.

DT-4005

* test(nexus-operations): add route-for-base-path test cases for standalone nexus operations
…figurable table columns (#3466)

Adds Phase 2 (DT-4006) infrastructure for standalone nexus operations:

- NamespaceCapabilities type with standaloneNexusOperations field in src/lib/types/index.ts
- Guard component that checks both capability flag and minimum server version (1.31.0)
- Disabled state component with Dynamic Config instructions
- i18n namespace for standalone-nexus-operations
- TABLE_TYPE.NEXUS_OPERATIONS, NexusOperationHeaderLabels, default/available column arrays,
  persistedNexusOperationsTableColumns store, configurableTableColumns derived store update,
  availableNexusOperationColumns export, and exhaustive switch cases in configurable-table-columns.ts
…3] (#3470)

* feat(nexus-operations): add standalone nexus operations list page

- Add list page route at namespaces/[namespace]/nexus-operations
- Add page component with status count filters and configurable table
- Add filter bar with search attribute support
- Add configurable table with header/body cells and empty state
- Add get-nexus-operation-status-and-count utility and tests
- Add nexus operation search attributes to search-attributes store
- Wire nav layout item for standalone nexus operations
- Add i18n strings for nexus operations page

* feat(nexus-operations): add saved views panel, CTA button, and rich empty state

Match Figma design for standalone nexus operations list page:
- Add saved views panel with system/user views, expand/collapse, save/edit/delete/share
- Add 'Start a Standalone Nexus Operation' CTA button via headerActions snippet
- Add rich empty state with value proposition, docs links, and GitHub code samples
- Add i18n strings for all new UI copy
- Add savedNexusQueryNavOpen persist store and savedNexusQueries/systemNexusViews stores

* Update api version to v1.62.12 for SANO APIs and capability

* fix capability name
@vercel

vercel Bot commented Jun 3, 2026

Copy link
Copy Markdown

The latest updates on your projects. Learn more about Vercel for GitHub.

Project Deployment Actions Updated (UTC)
holocene Ready Ready Preview, Comment Jun 8, 2026 8:35pm

Request Review

const handleCopy = (e: Event) => {
const sharableViewUrl =
new URL(page.url.href) +
'&savedQuery=' +

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Argument of type 'string | undefined' is not assignable to parameter of type 'string | number | boolean'.

export class StandaloneNexusOperationPoller {
private abortController: AbortController;
private namespace: string;
private operationId: string;

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Property 'token' has no initializer and is not definitely assigned in the constructor.

this.namespace,
this.operationId,
);
} catch (error) {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Argument of type 'unknown' is not assignable to parameter of type 'Error'.

@temporal-cicd

temporal-cicd Bot commented Jun 3, 2026

Copy link
Copy Markdown
Contributor
Warnings
⚠️

📊 Strict Mode: 26 errors in 10 files (2.8% of 919 total)

src/lib/services/standalone-nexus-operations.ts (3)
  • L63:6: Type 'ErrorCallback' is not assignable to type '(error: unknown, toasts?: Toaster, errors?: Writable<NetworkError | null>, isBrowser?: boolean) => void'.
  • L155:6: Type '{ indexedFields?: ({ [k: string]: temporal.api.common.v1.IPayload; } | null); }' is not assignable to type '{ [k: string]: IPayload; }'.
  • L156:31: Argument of type 'Record<string, unknown>[]' is not assignable to parameter of type '{ type: "Unspecified" | "Keyword" | "Text" | "Int" | "Double" | "Bool" | "KeywordList" | "Datetime"; label: string; value?: any; }[]'.
src/lib/stores/configurable-table-columns.ts (1)
  • L263:51: Property 'name' does not exist on type 'INamespaceInfo | null | undefined'.
src/lib/utilities/standalone-activity-poller.svelte.ts (5)
  • L18:10: Property 'token' has no initializer and is not definitely assigned in the constructor.
  • L47:19: Argument of type 'unknown' is not assignable to parameter of type 'Error'.
  • L54:6: Type 'string | undefined' is not assignable to type 'string'.
  • L70:12: Type 'string | undefined' is not assignable to type 'string'.
  • L78:23: Argument of type 'unknown' is not assignable to parameter of type 'Error'.
src/lib/utilities/standalone-nexus-operation-poller.svelte.ts (5)
  • L18:10: Property 'token' has no initializer and is not definitely assigned in the constructor.
  • L45:19: Argument of type 'unknown' is not assignable to parameter of type 'Error'.
  • L55:6: Type 'string | undefined' is not assignable to type 'string'.
  • L71:12: Type 'string | undefined' is not assignable to type 'string'.
  • L79:23: Argument of type 'unknown' is not assignable to parameter of type 'Error'.
src/lib/components/standalone-nexus-operations/cancel-confirmation-modal.svelte (1)
  • L44:6: Type 'string | undefined' is not assignable to type 'string'.
src/lib/components/standalone-nexus-operations/terminate-confirmation-modal.svelte (1)
  • L46:6: Type 'string | undefined' is not assignable to type 'string'.
src/lib/components/standalone-nexus-operations/saved-views.svelte (1)
  • L152:25: Argument of type 'string | undefined' is not assignable to parameter of type 'string | number | boolean'.
src/lib/components/standalone-nexus-operations/start-standalone-nexus-operation-form/form.svelte (7)
  • L144:6: Type 'ClientValidationAdapter<{ encoding: "json/plain" | "json/protobuf"; namespace: string; endpoint: string; service: string; operation: string; identity: string; operationId: string; nexusHeader: { key: string; value: string; }[]; ... 8 more ...; idConflictPolicy?: string | undefined; }, { ...; }>' is not assignable to type 'ClientValidationAdapter<Partial<{ input: string; messageType: string; scheduleToStartTimeout: string; summary: string; details: string; idReusePolicy: string; idConflictPolicy: string; nexusHeader: never[]; encoding: "json/plain" | "json/protobuf"; ... 7 more ...; scheduleToCloseTimeout: string | undefined; }>, Reco...'.
  • L244:25: Type '{ key: string; value: string; }' is not assignable to type 'never'.
  • L244:47: Type '{ key: string; value: string; }' is not assignable to type 'never'.
  • L322:11: Type 'string | undefined' is not assignable to type 'string'.
  • L331:11: Type 'string | undefined' is not assignable to type 'string'.
  • L363:49: Property 'key' does not exist on type 'never'.
  • L372:49: Property 'value' does not exist on type 'never'.
src/lib/pages/standalone-nexus-operation-metadata.svelte (1)
  • L40:23: Argument of type 'IUserMetadata | undefined' is not assignable to parameter of type 'IUserMetadata'.
src/lib/pages/standalone-nexus-operation-search-attributes.svelte (1)
  • L12:22: Type 'Record<string, IPayload> | undefined' is not assignable to type 'object | IPayloads | IPayload'.

Generated by 🚫 dangerJS against c58facb

…002] (#3495)

* [DT-4039] Workflow query builder doesn't work with numeric search attributes (#3435)

* fix(DT-4039): Workflow query builder for numeric search attributes

The query builder wrapped Int/Double search attribute values in
double quotes (e.g. `Foo="1"`), which the Visibility backend rejects
for numeric types. Stop quoting numeric values in the serializer.

The tokenizer also merged unquoted values with their conditional
operator (e.g. `Foo=1` tokenized as `Foo`, `=1`), which previously
only mattered for booleans and was patched in the parser by stripping
`=`. With numerics this breaks down for 2-char operators (`>=`, `<=`,
`!=`). Fix the tokenizer to emit the conditional as its own token and
to accept operator-style 2-char conditionals without a trailing space.
Update the boolean parser to read the value from `tokenTwoAhead` to
match the new shape.

* fix(DT-4039): Update integration tests for unquoted numeric search attributes

The desktop and mobile workflow-filter specs asserted the previous
buggy serialization (`HistoryLength`="10"). Update them to match the
new correct shape (`HistoryLength`=10).

* Make Common Errors dismissable (#3471)

* Make common errors dismissable

* Remove common errors

* Capitalize Common Errors

* Add description

* Update description

* [DT-4048] Add accessibility PR triage and notification helpers (#3465)

* Implement helpers for accessibility PRs

* fix formatting

* fix linting

* fix title gating regex

* improve deduplication detection on notify

* Move CLAUDE.md a11y conventions to DT-4049

* feat(DT-4017): Schedules List UI Update (#3467)

* fix(markdown): allow nested lists to render as block (DT-4047) (#3463)

* Add a prop to hide select/deselect controls (#3474)

* DT-4051: pre-populate input when starting standalone activity like this one (#3469)

* fix(a11y): reveal Copyable's CopyButton on focus-within (WCAG 2.1.1) (#3452)

The Copyable primitive hides its CopyButton behind
'invisible group-hover:visible'. Tailwind's 'invisible' compiles
to visibility: hidden, which removes the element from the focus
order -- Tab skips it entirely. Mouse users can hover to reveal
and click; keyboard users in default mode have no way to invoke
copy at all.

Add 'group-focus-within:visible' alongside the existing
'group-hover:visible' so the button becomes visible (and
therefore focus-eligible) whenever focus moves anywhere inside
the wrapping group -- typically when the user Tabs onto the
slot's content (workflow IDs, run IDs, etc.). The next Tab now
lands on the CopyButton, where Enter or Space triggers copy.

Applied to both branches (clickAllToCopy and default mode).

Affects detail-list value cells, event-summary rows, and every
other Copyable consumer in default mode.

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>

* fix: accept autocomplete prop on NumberInput, ChipInput, and Combobox (#3425)

Allows consumers to pass WCAG-defined autocomplete tokens (e.g.
"cc-number", "email") to these input primitives. All three previously
hardcoded autocomplete="off", preventing purpose identification per
WCAG 2.2 SC 1.3.5. Default remains "off" so existing behavior is
unchanged.

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>

* fix: cap ZoomSvg container height to viewport (#3428)

Uses CSS min() to limit the container to the lesser of containerHeight
(default 600px) or viewport height minus 8rem. On narrow landscape
viewports (e.g. 320x500) the container now fits within the screen
instead of overflowing. Desktop behavior is unchanged.

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>

* fix(a11y): accommodate text-spacing overrides on Badge and Chip (WCAG 1.4.12) (#3433)

* fix(a11y): accommodate text-spacing overrides (WCAG 1.4.12)

Three same-family fixes for SC 1.4.12 Text Spacing. WCAG requires
that text-spacing user overrides (line-height 1.5, letter-spacing
0.12em, etc.) not cause content loss.

- Badge: leading-4 (16px) -> leading-[1.5]. text-sm at 1.5 line-
  height is 21px which previously clipped.
- Chip: h-7 -> min-h-7, add leading-[1.5]. Preserves the 28px
  visual minimum but allows growth.
- Table row: h-8 -> min-h-8. The densest text-bearing surface in
  the product. Preserves the 32px default but allows growth under
  user override.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* revert(a11y): undo h-8 -> min-h-8 on table row

Per reviewer feedback: <tr> uses table layout rules, where the
height property already acts as a minimum (the row grows to fit
cell content). min-height on <tr> is either ignored or behaves
differently per browser. The change broke the skeleton table
without providing the intended SC 1.4.12 benefit -- under
text-spacing override, the row would have grown naturally.

Badge and chip portions of the PR remain in place (those are flex
elements where min-h-* works as expected).

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

---------

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>

* fix(a11y): non-color signal for Label required indicator (WCAG 1.4.1) (#3439)

* fix(a11y): non-color signals for required state and timeline graph nodes (WCAG 1.4.1)

Two SC 1.4.1 Use of Color fixes:

- Label required-field indicator: replace the 6x6 red dot with a
  red asterisk (aria-hidden) so users with color-perception
  differences see a shape signal, not just a color signal. The
  programmatic required state continues to come from native HTML
  `required` on the input (which the 1.3.1 forwarders ensure).
- Timeline graph nodes (workflow-row, history-graph-row-visual):
  add aria-label to the <g role="button"> wrapper so AT users hear
  the workflow id + status (or event type + classification)
  instead of bare "button". Removes the previous dependency on
  reaching the legend tooltip to decode color.

New i18n keys: workflows.row-accessible-name,
events.row-accessible-name.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* revert: remove timeline graph aria-label changes from this PR

Per the PR scope decision, keep this PR focused on the Label
required-indicator fix only. The timeline graph status accessible-
name fix (workflow-row.svelte, history-graph-row-visual.svelte,
and the supporting i18n keys) will ship as its own separate PR
so the two SC 1.4.1 defects can be reviewed independently.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* fix(a11y): use text-danger so the asterisk actually renders red

The previous text-interactive-error class was a no-op for text
color (interactive-error is only registered under backgroundColor
in the theme plugin, not textColor), so the asterisk inherited
the default text color and rendered black/off-white instead of
red. text-danger maps to --color-text-danger (red.700 light /
red.400 dark) which is the proper red text token.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* refactor: simplify required-asterisk span

The parent <label> already provides text-sm and font-medium, so
the span inherits both. Only text-danger is doing actual work;
the size/weight/leading classes were redundant.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* refactor(label): refine required-asterisk visual alignment

- font-mono: mono asterisks are larger and more geometrically
  centered than proportional ones, making them feel like a proper
  visual indicator rather than a tiny text-style superscript.
- leading-none: tightens the span's line-box to the character.
- translate-y-0.5: small downward nudge so the asterisk aligns
  with the label baseline rather than sitting above it.
- -ml-1: reduces the gap to the preceding label text (8px -> 4px).

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

---------

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>

* fix(a11y): tabindex on <main> so the skip link moves focus reliably (WCAG 2.4.1) (#3451)

The SkipNavigation primitive renders <a href="#content">, which
targets <main id="content"> in main-content-container.svelte. By
HTML spec, <main> without tabindex is not programmatically
focusable -- so activating the skip link only reliably moves
focus on Chromium-based browsers (Chrome, Edge). Safari and
older Firefox scroll the target into view but leave focus on the
skip link itself, defeating the bypass for keyboard and screen-
reader users.

Adding tabindex="-1" makes <main> programmatically focusable
(so the hash-link navigation lands focus there) while keeping it
out of the natural Tab order. Closes WCAG 2.4.1 sufficient-
technique G1's "programmatically moves focus" requirement across
all browsers.

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>

* fix(a11y): info-and-relationships compliance (WCAG 1.3.1) (#3432)

Four related fixes under SC 1.3.1, grouped per the audit's
correction of the matrix pathway:

- NumberInput / Textarea: forward `required` to the underlying
  native element. Previously the visual red-dot appeared via
  <Label> but the field was not programmatically required, so AT
  users heard an optional-field announcement despite the visible
  indicator.
- RadioGroup: add role="radiogroup" and link the optional
  description via aria-labelledby so AT users hear the group as a
  whole rather than each radio in isolation.
- Tables: add scope="col" to every <th> in product data tables
  (workflows, schedules, workers, batch-operations, child/parent
  workflow relationships). Fixes WCAG technique H63.

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
Co-authored-by: Ardie Wen <ardie.wen@temporal.io>

* fix(query): quote ExecutionDuration Go duration strings in visibility SQL (#3482)

ExecutionDuration is typed as INT in the search attributes store, so
formatValue returns its values unquoted. PR #3435 (DT-4039) added an
explicit unquoted INT path which broke ExecutionDuration because it uses
Go duration strings (30s, 1m30s) that vitess rejects as unquoted tokens.

Add an ExecutionDuration guard before the INT/duration paths: quote
non-integer values as strings, pass plain integers (nanoseconds) through
unquoted.

* Passthrough goto params to link component (#3483)

* feat: add centerButton, menuButton, and linksContent snippets to BottomNavigation (#3485)

* feat: add centerButton snippet to BottomNavigation for custom center content

* feat: add centerButton, menuButton, and linksContent snippets to BottomNavigation

* feat: add bars icon to holocene icon set

* fix: add "for" to validate connection modal title (#3488)

The worker deployment version validate connection modal title now reads
"Validate Connection for <version>" instead of "Validate Connection <version>".

Co-authored-by: Claude Opus 4.8 (1M context) <noreply@anthropic.com>

* feat(DT-4069): Update modal backdrop to 50% opaque (#3486)

* refactor: Input & DatePicker - Svelte 5 & afterLabel snippet (#3479)

* Fix decoded object payload summaries (#3491)

* Fix decoded object payload summaries

* Convert event details row to Svelte 5

* fix(DT-4044): only use browser codec endpoint when override option is selected (#3490)

When the user selects "Use Namespace-level setting, where available" and
no namespace codec is configured, the browser endpoint was incorrectly
used as a fallback. Now returns empty string so no codec is invoked.

* fix(a11y): improve 1.1.1 non-text content compliance (#3431)

- nexus-empty-state.svelte: mark Andromeda illustration decorative
  with alt="" + aria-hidden on wrapper. Previously screen readers
  announced "Andromeda, image" (the filename), conveying nothing.
- toast.svelte: add runtime fallback to translate('common.close')
  when closeButtonLabel is missing or empty. TypeScript-required
  prop still flags missing-at-compile-time, but an empty string or
  undefined at runtime now produces a labeled icon button rather
  than aria-label="".

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
Co-authored-by: Ardie Wen <ardie.wen@temporal.io>

* fix(a11y): make Tooltip keyboard-accessible, dismissible, and hoverable (#3429)

* fix: make Tooltip keyboard-accessible, dismissible, and hoverable

The Tooltip primitive was mouse-only: no focus handlers, no Escape
dismiss, and portal tooltips disappeared when moving the pointer to
the popover content. This failed all three WCAG 2.2 SC 1.4.13
criteria (Content on Hover or Focus).

Changes:
- Show tooltip on keyboard focus via focusin/focusout on wrapper
- Dismiss on Escape (resets when interaction ends)
- Track pointer hover on the popover itself (diagonal hover bridge)
- Unify both portal and inline variants on a single isOpen derived
- Add role="tooltip" and aria-describedby linkage for screen readers

No API changes — all existing consumers work identically, with the
added benefit that tooltips now appear on keyboard focus.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* refactor(tooltip): unify hover-zone handling and match repo patterns

- Use <svelte:window on:keydown> instead of reactive document
  listener (mirrors maximizable.svelte)
- Fix hover state lingering when mouse leaves popover to empty
  space by unifying wrapper + popover into a single hover zone
- Extract HOVER_HIDE_DELAY_MS constant
- Drop redundant isPopoverHovered state

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* refactor(tooltip): use crypto.randomUUID() for tooltip id

Align tooltip id generation with the codebase convention used across
Holocene (accordion.svelte, accordion-light.svelte) and the rest of the
app, replacing the ad-hoc Math.random().toString(36) approach.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>

* refactor(tooltip): drop unused group/tooltip class

The group/tooltip marker only existed to drive the group-hover/tooltip:
utilities that previously controlled visibility. Tooltip open/close is
now fully JS-driven via isOpen, leaving this class with no consumers.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>

---------

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
Co-authored-by: Ardie Wen <ardie.wen@temporal.io>

* fix(a11y): name-role-value compliance, partial (WCAG 4.1.2) (#3434)

* fix(a11y): name-role-value compliance (WCAG 4.1.2)

Five fixes under SC 4.1.2 Name, Role, Value:

- Accordion content panel: remove role="textbox". The disclosure
  panel is static content, not an editable text input; AT
  announced "edit text" on every Accordion consumer.
- Accordion + AccordionLight: move slot="action" out of the
  disclosure <button> into a sibling element. Previously
  consumers placing focusable content in the action slot
  produced <button><button>... nested-interactive HTML.
- CopyButton: ship translated default labels via i18n. Empty
  string defaults in CodeBlock previously propagated to an Icon
  that hid itself, producing unlabeled icon-only buttons across
  ~11 CodeBlock instances.
- nav-section: filter out hidden nav items. The NavLinkItem.hidden
  flag was declared on the type and set at source but never
  consulted at render time, so Standalone Activities / Nexus
  links rendered as focusable for AT users on servers that
  cannot serve them.
- Workflow family tree: add aria-label to the two SVG
  <g role="button"> widgets so screen readers announce node
  identity instead of "button".

Deferred to follow-up PRs (audit blockers):
- 4.1.2-button-anchor-empty: audit recommends a two-PR sequenced
  rollout (primitive change + consumer migration).
- 4.1.2-codemirror-no-name: ~5 CodeBlock consumer surfaces need
  design input on label content.
- relationships nested-interactive restructure: site B (button
  wraps Link) and site A's nested <a> extraction are substantial
  SVG/HTML restructures separate from the labeling fix above.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* fix(a11y): localize workflow family-tree node aria-labels

The family-tree node aria-labels hardcoded the English word "Workflow",
inconsistent with the codebase convention of localizing aria-labels via
translate(). Add a parameterized workflows.family-node-label i18n key and
use it for both the root and child node widgets.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>

---------

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
Co-authored-by: Ardie Wen <ardie.wen@temporal.io>

* feat(nexus-operations): standalone nexus operation details page [DT-4002]

Add tab-based detail view for individual nexus operations with live long-polling.
Mirrors standalone activity detail page pattern with nexus-specific fields:
identity, operation, status, timing, attempt, timeout config, cancellation info,
nexus header, and input/outcome payload display.

* feat(nexus-operations): redesign details page to match Figma layout

Restructure the details page around "Operation Event History" with three
subsections: Run Details, Operation Details, Timeout Configuration.

- Add filterable links for endpoint, service, and operation fields
- Add handler namespace and handler operation link from nexus operation links
- Show nexus header code block inline in Operation Details section
- Move last attempt failure into the Attempt section (conditional)
- Add handler namespace note when handler workflow link is present

* fix(nexus-operations): pass namespace to filter links on details page

* fix nexus operation id

---------

Co-authored-by: Andrew Zamojc <andrew.zamojc@temporal.io>
Co-authored-by: Alex Tideman <alex.tideman@temporal.io>
Co-authored-by: Ardie Wen <39966887+ardiewen@users.noreply.github.com>
Co-authored-by: Tegan Churchill <tegan.churchill@temporal.io>
Co-authored-by: Ross Nelson <ross.nelson@temporal.io>
Co-authored-by: Alex Thomsen <alex.scott.thomsen@gmail.com>
Co-authored-by: Bilal Karim <4129613+bilal-karim@users.noreply.github.com>
Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
Co-authored-by: Ardie Wen <ardie.wen@temporal.io>
return metadata;
};
</script>

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Argument of type 'IUserMetadata | undefined' is not assignable to parameter of type 'IUserMetadata'.

</script>

<div class="flex flex-col gap-2">
{#if searchAttributes}

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'Record<string, IPayload> | undefined' is not assignable to type 'object | IPayloads | IPayload'.

@@ -46,6 +46,7 @@ export class StandaloneActivityPoller {
);
} catch (error) {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Argument of type 'unknown' is not assignable to parameter of type 'Error'.

if (
nexusOperationExecution.info.status ===
'NEXUS_OPERATION_EXECUTION_STATUS_RUNNING'
) {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'string | undefined' is not assignable to type 'string'.

if (
polledNexusOperationExecution &&
!isEmptyObject(polledNexusOperationExecution)
) {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'string | undefined' is not assignable to type 'string'.

if (error instanceof Error && error.name === 'AbortError') {
return;
}

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Argument of type 'unknown' is not assignable to parameter of type 'Error'.

…type with StatusCounts prop

Return Promise<Required<CountWorkflowExecutionsResponse>> and default groups to [] to match the
pattern used by fetchWorkflowCountByExecutionStatus.
rossedfort and others added 2 commits June 5, 2026 13:32
…ncel/terminate) [DT-4008] (#3504)

* feat(nexus-operations): add cancel/terminate modals and actions to nexus operation header [DT-4008]

Add cancel confirmation modal, terminate confirmation modal, and nexus
operation actions component. Wire actions into the header with
Request Cancellation button and More Actions menu (Terminate + Start Like
This One). Add route-for helper and i18n strings for the start form.

* feat(nexus-operations): add start standalone nexus operation form and route [DT-4008]

Adds the start nexus operation form with fields for operation ID (with random UUID generation), endpoint, service, operation name, payload input, timeouts, and advanced options (nexus header, search attributes, user metadata, ID policies).

* feat(nexus-operations): pre-fill input payload when starting nexus operation like this one [DT-4008]

Adds fetchInitialValuesForStartNexusOperation to decode the source operation's input payload, encoding, message type, user metadata, and search attributes. The start form pre-populates these fields on mount when operationId and runId are present in the URL, and auto-expands advanced options when there is pre-filled data.
message: translate('standalone-nexus-operations.cancel-success'),
});
onConfirm();
} catch (err: unknown) {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'string | undefined' is not assignable to type 'string'.

SPA: true,
dataType: 'json',
resetForm: false,
invalidateAll: false,

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'ClientValidationAdapter<{ encoding: "json/plain" | "json/protobuf"; namespace: string; endpoint: string; service: string; operation: string; identity: string; operationId: string; nexusHeader: { key: string; value: string; }[]; ... 8 more ...; idConflictPolicy?: string | undefined; }, { ...; }>' is not assignable to type 'ClientValidationAdapter<Partial<{ input: string; messageType: string; scheduleToStartTimeout: string; summary: string; details: string; idReusePolicy: string; idConflictPolicy: string; nexusHeader: never[]; encoding: "json/plain" | "json/protobuf"; ... 7 more ...; scheduleToCloseTimeout: string | undefined; }>, Reco...'.

$form.operationId = crypto.randomUUID();
};

const addNexusHeader = () => {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type '{ key: string; value: string; }' is not assignable to type 'never'.
  • ⚠️ Type '{ key: string; value: string; }' is not assignable to type 'never'.

label={translate(
'standalone-nexus-operations.form-start-to-close-timeout-label',
)}
required={!$form.scheduleToCloseTimeout}

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'string | undefined' is not assignable to type 'string'.

label={translate(
'standalone-nexus-operations.form-schedule-to-close-timeout-label',
)}
required={!$form.startToCloseTimeout}

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'string | undefined' is not assignable to type 'string'.

labelHidden
placeholder={translate(
'standalone-nexus-operations.form-nexus-header-value-placeholder',
)}

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Property 'value' does not exist on type 'never'.

message: translate('standalone-nexus-operations.terminate-success'),
});
onConfirm();
} catch (err: unknown) {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'string | undefined' is not assignable to type 'string'.

...(query ? { query } : {}),
},
request,
onError,

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type 'ErrorCallback' is not assignable to type '(error: unknown, toasts?: Toaster, errors?: Writable<NetworkError | null>, isBrowser?: boolean) => void'.

}

if (formData.searchAttributes) {
searchAttributes = {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Type '{ indexedFields?: ({ [k: string]: temporal.api.common.v1.IPayload; } | null); }' is not assignable to type '{ [k: string]: IPayload; }'.


if (formData.searchAttributes) {
searchAttributes = {
indexedFields: {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • ⚠️ Argument of type 'Record<string, unknown>[]' is not assignable to parameter of type '{ type: "Unspecified" | "Keyword" | "Text" | "Int" | "Double" | "Bool" | "KeywordList" | "Datetime"; label: string; value?: any; }[]'.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

1 participant