Releases: temporalio/sdk-typescript
1.10.0
Important updates
-
π₯ Support for Node.js 14 and older GLIBC releases has been dropped. As announced previously, TS SDK v1.9 was the last minor release to support Node.js 14.
Going forward, the TypeScript SDK requires:
- Node.js 16.15 or more recent;
- GLIBC 2.35 or more recent (update: we restored compatibility with GLIBC 2.18 in v1.10.1)
-
π₯ Support for Node.js 16 will soon be dropped. TS SDK v1.10 will be the last minor release to support Node.js 16. TS SDK v1.11 will require Node 18 or higher.
Node.js 14 officially reached end-of-life on April 30th, 2023, and Node.js 16 reached end-of-life on September 11th, 2023 (see Node.js release schedule). We strongly encourage everyone to take immediate action in updating their deployments to supported versions of Node.js to ensure continued compatibility and receive the latest features and security updates.
Features
-
[
client
] BothConnection
andNativeConnection
now accept anapiKey
option, making it easier to connect to Temporal Cloud and other Temporal servers that use bearer token authentication (#1385), as well as HTTP CONNECT Proxy (#1411, thanks to @brendan-myers for his help on getting this started π). -
[
client
] Thestartdelay
Workflow option is no longer experimental (#1379). Thanks @tlafano π. -
[
core
] Core can now be configured to emit duration metrics as either seconds or milliseconds, as well as to include unit suffixes in metric names (#1383). -
[
worker
] AddtaskQueue
andlogSource
metadata attributes on all log messages emitted inside the context of a worker. These attributes are there make is easier to diagnose some low level issues, as well as allow implementing fine-grained filtering of messages (eg. show DEBUG level messages for user-code, but only WARN for messages from the worker) (#1391).
Bug Fixes
-
[
client
] Bring gRPC retry options in line with other SDKs (#1368). Thanks to @chronos-tachyon π. -
[
worker
] Unreference timer created inparseWorkflowCode
(#1370). Thanks @jhecking π. -
On POSIX systems,
stdout
andstderr
are now properly inherited by ephemeral servers (e.g.TestWorkflowEnvironment.createLocal()
andTestWorkflowEnvironment.createTimeSkipping()
), making it easier to diagnose issues while creating a test environment (#1394). -
@grpc/grpc-js
has been updated to 1.10.x (#1388). It had previously been pinned to 1.7.3 to avoid multiple bugs and incompatibilities introduced ingrpc-js
1.8.0. We have since reviewed these changes and confirmed that there is no longer any reason to refrain form updating that dependency. -
bundler
: Webpack output is now recorded as a single log entry, and color are disabled if output is not being sent to the console (#1414). Thanks @ikonst π. -
MockActivityEnvironment
no longer causes initialization of theRuntime
(#1401). -
[
common
] Improve error messages forfailureConverters
(#1373). -
[
doc
] Fix comment on defaultmaxConcurrentActivityTaskPolls
value (#1374) Thanks @taonic π.
v1.9.3
Bug fixes
- Fix deprecated patch removal when no subsequent command (temporalio/sdk-core#671).
- Fix incorrect ordering of a query execution that could happen if new workflow tasks came in while there were buffered queries (temporalio/sdk-core#673).
- Fix a possibility of sending local activity marker after the Workflow Task Complete command (temporalio/sdk-core#678).
- Wait for activity completes to reach server before shutdown (temporalio/sdk-core#681).
- Update protobuf definitions to work with the new
EVENT_TYPE_WORKFLOW_EXECUTION_UPDATE_REQUESTED
event (temporalio/sdk-core#677). - Fix a regression regarding gRPC incoming message limit due to a behavior change in tonic 0.9.0.
Documentation
- Avoid doctype reexports canonicalization issue (#1354)
Miscellaneous Tasks
- Add test for module property mutation isolation (#1356)
v1.9.1
Bug Fixes
- [
worker
] Fix prometheus and otel regressions (#1345) - Fix
devDependencies
in some of the SDK packages where file URLs were used instead of version numbers.
Documentation
- Add note on custom conveters and bundleWorkflowCode (#1338)
- Fix sort order + snipsync examples issue (#1341)
- [
worker
] Update webpack error doc link in worker/bundler.ts (#1344) - Explain the purpose of
updateId
(#1318)
Features
- [
worker
] Add a workflow metadata query (#1319)
v1.9.0
Important updates
-
The
reuseV8Context
worker option is now enabled by default. This major optimization of the Workflow sandboxing runtime significantly reduces RAM and CPU usage, without compromising the safety of the deterministic sandbox. You may still revert to the legacy execution model by addingreuseV8Context: false
to your worker options.Note that support for the legacy, non-
reuseV8Context
execution model may be removed at some point in the future. Please reach out if you find any issue that requires disablingreuseV8Context
. -
Support for Node.js 14 and Node.js 16 will soon be dropped. Node.js 14 officially reached end-of-life on April 30th, 2023, and Node.js 16 reached end-of-life on September 11th, 2023 (see Node.js release schedule).
This has since then caused a rippling effect, as more and more libraries in the NPM ecosystem are progressively dropping support for both of those older targets. Given that challenges to maintaining compatibility with these outdated Node.js releases will only continue to grow, it is evident that we will soon need to follow suit in discontinuing support for Node.js 14 and 16 in the Temporal TypeScript SDK.
For those reasons:
- TS SDK v1.9 will be the last minor release to support Node.js 14.
- TS SDK v1.10 could possibly be the last minor release to support Node.js 16. Note that this also implies that GLIBC 2.27 will no longer be supported from that point on, since Node.js 18+ wonβt compile on GLIBC older than 2.28.
We strongly encourage everyone to take immediate action in updating their deployments to supported versions of Node.js to ensure continued compatibility and receive the latest features and security updates.
New & Noteworthy
Support for Workflow Update (experimental)
-
Introduced Workflow Update APIs, both on Workflow side and Client side (#1277, #1312 and #1320). Kudos to @dandavison π.
β οΈ The Workflow Update feature is currently in pre-release stage. Related APIs are considered experimental and may change in the near future. This feature requires Temporal Server version 1.21 or later, and must be enabled through configuration.Here are some examples to get started:
// An Update is defined similarly to Query and Signal. The second type parameter (input) is optional const myUpdate = wf.defineUpdate<string, [string]>('myUpdate'); // A handler is a sync or async function; the optional validator is always sync. wf.setHandler(myUpdate, handler, { validator }); // `await wfHandle.executeUpdate` to start an update and block until it completes or fails. // The signature resembles `WorkflowClient.start` in that the arguments are supplied // in the optional options object // (as opposed to the variadic signatures of wfHandle.query and wfHandle.signal) // An error is thrown if the RPC times out (we do not poll for this call) // or if the validator rejects, or the handler throws ApplicationFailure. const updateResult = await handle.executeUpdate(myUpdate, { args: ['a'] }); // `startUpdate` to block until Accepted and obtain a handle to the Update. Signature the same as `executeUpdate`. const updateHandle = await handle.startUpdate(myUpdate, { args: ['a'] }); // Blocking call to fetch result, with polling. // Validation failures or ApplicationFailure in the handler would throw here. const updateResult = await updateHandle.result();
Major overhaul of logging support
This release features a major overhaul of Workflow logging, Activity logging support, and Core log forwarding support.
These changes add up to better DX, notably by promoting clear usage patterns that work out of the box, removing the need to use insufficiently documented features, and removing some requirements that were error-prone and a recurring source of support questions.
Workflow Logging
-
Calling
appendDefaultInterceptors()
when registering custom interceptors is no longer required in order for Workflow logger support to work properly.appendDefaultInterceptors()
is now deprecated and behave as a no-op in most cases (i.e. except when a custom logger is supplied in order to redirect log emitted by activities) (#1290). -
Similarly, calling
defaultSink()
when registering custom sinks is no longer required for Workflow logging to work properly (#1283). SDK's logger sink is now forcibly injected. Registering a custom logger sink nameddefaultWorkerLogger
is still supported, but discouraged and deprecated. -
Attributes from the current Workflow are now automatically included as metadata on every log entries emitted using the Workflow context logger, and some key events of the Workflow's lifecycle are now automatically logged (at
DEBUG
level for most messages;WARN
for failures) (#1290). -
WorkflowLogInterceptor
has been deprecated (#1290).
Activity Logging
-
Attributes from the current Activity context are now automatically included as metadata on every log entries emitted using the Activity context logger, and some key events of the Activity's lifecycle are now automatically logged (at
DEBUG
level for most messages;WARN
for failures) (#1284). -
Logged Activity attributes can be customized by registering an
ActivityOutboundCallsInterceptor
that intercepts thegetLogAttributes()
method (#1284). -
ActivityInboundLogInterceptor
has been deprecated; registering such an interceptor with a custom logger still works, but will prevent the use of the newly introducedgetLogAttributes()
interceptor. The same is true about modifying the context logger directly (e.g.context.log = myCustomLogger
) (#1284).
Core Logging
-
Log entries forwarded from Core to TS now retain metadata (#1225).
-
Reduce the default Core log level to
WARN
and non Core Rust logs toERROR
(#1270).
Improvements to Activity-related APIs
-
π₯ BREAKING: Non-local activities may now receive cancellation for schedule-to-close and start-to-close timeouts, without heartbeating. Heartbeating is still required to receive server sent cancellations.
-
All APIs provided through
Context.current()
can now be accessed directly from the@temporalio/activity
import (#1252). Also, thanks to @SergeiMeza for fixing a related issue that would occur whenheartbeat
andsleep
functions were destructured out ofContext.current()
(#1249) π.Instead of this:
import { Context } from `@temporalio/activity`; // ... Context.current().sleep('5 minutes');
you can now do:
import { sleep } from `@temporalio/activity`; // ... sleep('5 minutes');
-
ActivityInboundCallsInterceptorFactory
has been deprecated. Activity Interceptors should now be instantiated using anActivityInterceptorsFactory
(#1284). -
MockActivityEnvironment
now accept a list of activity interceptors, and more closely match an actual activity execution (#1284).
Improvements to Client-related APIs
-
Allow specifying gRPC
CallCredentials
on Client connection. Also, fixed the fact thatConnectionOptions.credentials
was completely ignored (#1261). -
Client
andConnection
classes now expose awithAbortSignal()
method. This can be used to cancel all ongoing connections started within a function's scope (#1272).Example:
const ctrl = new AbortController(); setTimeout(() => ctrl.abort(), 10_000); // π throws if incomplete by the timeout. await conn.withAbortSignal(ctrl.signal, () => { /* make client or direct grpc calls */);
-
Add
startDelay
option to workflowstart
,execute
, andsignalWithStart
(#1300).
Miscellaneous
-
WorkerOptions
now exposes anonStickyToStickyPollRatio
property, which controls the proportion of Workflow Task Pollers assigned to polling from Regular Task Queue vs to polling from Sticky Tasks Queues (#1254). -
Fixed a bug that could happen if the
@temporalio/proto
was minimized (#1279). -
Prevents a Workflow-only Worker from requesting Eager Activity Execution for activities schedules on the same task queue name (#1326).
1.9.0-rc0
Support for Workflow Update (experimental)
-
Introduced Workflow Update APIs, both on Workflow side and Client side (#1277 and #1312). Kudos to @dandavison π.
β οΈ The Workflow Update feature is currently in Public Preview. Related APIs are considered experimental and may change in the near future. This feature requires Temporal Server version 1.21 or later, and must be enabled through configuration.Here are some examples to get started:
// An Update is defined similarly to Query and Signal. The second type parameter (input) is optional const myUpdate = wf.defineUpdate<string, [string]>('myUpdate'); // A handler is a sync or async function; the optional validator is always sync. wf.setHandler(myUpdate, handler, { validator }); // `await wfHandle.executeUpdate` to start an update and block until it completes or fails. // The signature resembles `WorkflowClient.start` in that the arguments are supplied // in the optional options object // (as opposed to the variadic signatures of wfHandle.query and wfHandle.signal) // An error is thrown if the RPC times out (we do not poll for this call) // or if the validator rejects, or the handler throws ApplicationFailure. const updateResult = await handle.executeUpdate(myUpdate, { args: ['a'] }); // `startUpdate` to block until Accepted and obtain a handle to the Update. // Signature the same as `executeUpdate`. const updateHandle = await handle.startUpdate(myUpdate, { args: ['a'] }); // Blocking call to fetch result, with polling. // Validation failures or ApplicationFailure in the handler would throw here. const updateResult = await updateHandle.result();
Major overhaul of logging support
This release feature a major overhaul of Workflow logging, Activity logging support, and Core log forwarding support.
These changes add up to better DX, notably by promoting clear usage patterns that work out of the box, remove the need to use insufficiently documented features, and remove some requirements that were error-prone and recurring source of support questions.
Workflow Logging
-
Calling
appendDefaultInterceptors()
when registering custom interceptors is no longer required in order for Workflow logger support to work properly.appendDefaultInterceptors()
is now deprecated and behave as a no-op in most cases (i.e. except when a custom logger is supplied in order to redirect log emitted by activities) (#1290). -
Similarly, calling
defaultSink()
when registering custom sinks is no longer required for Workflow logging to work properly (#1283). SDK's logger sink is now forcibly injected. Registering a custom logger sink nameddefaultWorkerLogger
is still supported, but discouraged and deprecated. -
Attributes from the current Workflow are now automatically included as metadata on every log entries emited using the Workflow context logger, and some key events of the Workflow's lifecycle are now automatically logged (at
DEBUG
level for most messages;WARN
for failures) (#1290). -
WorkflowLogInterceptor
has been deprecated (#1290).
Activity Logging
-
Attributes from the current Activity context are now automatically included as metadata on every log entries emited using the Activity context logger, and some key events of the Activity's lifecycle are now automatically logged (at
DEBUG
level for most messages;WARN
for failures) (#1284). -
Logged Activity attributes can be customized by registering an
ActivityOutboundCallsInterceptor
that intercepts thegetLogAttributes()
method (#1284). -
ActivityInboundLogInterceptor
has been deprecated; registering such an interceptor with a custom logger still works, but will prevent the use of the newly introducedgetLogAttributes()
interceptor. The same is true about modifying the context logger directly (eg.context.log = myCustomLogger
) (#1284).
Core Logging
-
Log entries forwarded from Core to TS now retain metadata (#1225).
-
Reduce the default Core log level to
WARN
and non Core Rust logs toERROR
(#1270).
Improvements to Activity-related APIs
-
π₯ Remote activities may now receive cancellation for schedule-to-close and start-to-close timeouts, without heartbeating. Heartbeating is still required to receive server sent cancellations.
-
All APIs provided through
Context.current()
can now be accessed directly from the@temporalio/activity
import (#1252). Also, thanks to @SergeiMeza for fixing a related issue that would occur whenheartbeat
andsleep
functions were destructured out ofContext.current()
(#1249) π.Instead of this:
import { Context } from `@temporalio/activity`; // ... Context.current().sleep('5 minutes');
you can now do:
import { sleep } from `@temporalio/activity`; // ... sleep('5 minutes');
-
ActivityInboundCallsInterceptorFactory
has been deprecated. Activity Interceptors should now be instantiated using anActivityInterceptorsFactory
(#1284). -
MockActivityEnvironment
now accept a list of activity interceptors, and more closely match an actual activity execution (#1284).
Improvements to Client-related APIs
-
Allow specifying gRPC
CallCredentials
on Client connection. Also, fixed the fact thatConnectionOptions.credentials
was completely ignored (#1261). -
Client
andConnection
classes now expose awithAbortSignal()
method. This can be used to cancel all ongoing connections started within a function's scope. (#1272)Example:
const ctrl = new AbortController(); setTimeout(() => ctrl.abort(), 10_000); // π throws if incomplete by the timeout. await conn.withAbortSignal(ctrl.signal, () => { /* make client or direct grpc calls */);
-
Add
startDelay
to workflow (signal) start options (#1300)
Miscellaneous
-
WorkerOptions
now expose anonStickyToStickyPollRatio
property, which control the proportion of Workflow Task Pollers assigned to polling from Regular Task Queue vs to polling from Sticky Tasks Queues. (#1254) -
Enable
reuseV8Context
by default (#1310) -
Fixed a bug that could happen if the
@temporalio/proto
was minimized (#1279)
v1.8.6
v1.8.5
Features
-
[
workflow
] AddhistorySizeInBytes
andcontinueAsNewSuggested
toWorkflowInfo
(#1223). -
Remove experimental flags on the following APIs (#1235):
- Schedules
- Failure Converters
- Local Activities
- Replay Histories
Bug Fixes
-
[
workflow
] Fix a particular case whereisReplaying
could be incorrectlyfalse
if a query came in right after a cache eviction and there had been a signal in the last workflow task before the cache eviction (#1234). This could notably cause situations where calls to sink functions configured withcalledDuringReplay = false
would be called more than once for a same code location, as well as situations wherepatched(...)
would return true even though that patch had not been set on the first execution. -
[
workflow
] MakeExternalWorkflowHandle.signal
signature match that ofBaseWorkflowHandle.signal
(#1237). Thanks to@gabrielsantosblanchet
and@josiah-roberts
π. -
[
worker
] Avoid rxjs error while flushing logs on worker shutdown (#1238). -
Upgrade
protobufjs
dependencies (#1219). Thanks to@alex-dixon
π.
1.8.4
Bug Fixes
-
[
workflow
] Fix a particular case whereisReplaying
would be incorrectlytrue
if a query came in right after a cache eviction (#1211). This could notably cause situations where calls to sink functions not configured withcalledDuringReplay = true
would not be relayed, as well as situations wherepatched(...)
would return false even though this part of the code was being executed for the first time. -
[
workflow
] Avoid errors in Search Attributes Payload Converter if enumerable properties had been added onArray.prototype
(#1209). -
[
worker
] ValidatebuildId
is set whenuseVersioning
is true (#1207).
1.8.3
Bug Fixes
- [
workflow
] Remove accidental import of large protos into workflow library (#1203). This had been causing significant memory usage increases on workflow workers since 1.8.1. - [
core
] Fix removal of deprecated patch call adjacent to other patch (#587) - [
client
] ThrowWorkflowExecutionAlreadyStartedError
onsignalWithStart
(#1199 thanks to@satazor
π)
1.8.2
Features
-
[
workflow
] Add support for URL/URLSearchParams inside the workflow sandbox (#1173) -
[
worker
] Export theWorkerStatus
interface (#1184).
Bug Fixes
-
Use custom symbol-based implementation of
instanceof
for all of our custom error classes (#1166). This is a complete rework of the fixes introduced in 1.8.0 and 1.8.1, both of which turned out to be insufficient.instanceof
now works correctly both across execution contexts and when running tests with Jest.π₯ The static
is
function introduced previously on some of our error classes is no longer required, as theinstanceof
operator itself now behave correctly; theseis
functions have therefore been removed. -
[
client
] Makeaction.workflowId
optional on Schedule update (#1176) -
[
activity
]heartbeatTimeoutMs
is now correctly set on Activity'sContext.current().info
. ThecurrentAttemptScheduledTimestampMs
property has also been added to that data structure (#1187) -
[
workflow
] The Workflow Bundler is now smarter regarding requiring files with incorrect or missing file extensions (#1186). Thanks to@GauBen
π. -
[
workflow
] Fix incorrect values ofworkflowInfo.historyLength
andworkflowInfo.unsafe.isReplaying
as reported in out-of-sandbox log messages, and as argument of sink function implementations (#1181). -
[
workflow
] Sink functions configured withcallDuringReplay = false
are no longer invoked from a replay-only worker (i.e.Worker.runReplayHistories()
); it was previously possible for these to get called in some cases on the very last Workflow Task (#1181). -
[
workflow
] Encoding and decoding of protobuf JSON payloads is now working in workflow context. This was previously failing, due to the fact that an underlying library depends on theBuffer
class, which is not available inside the workflow sandbox (#1170). Thanks to@antlai-temporal
π. -
[
core
] Improve warning message on error response to completion (temporalio/sdk-core#581) Thanks to@dandavison
π. -
[
core
] Fix abandoned children nondeterminism if they complete (temporalio/sdk-core#580).