UNPKG

@temporalio/common

Version:

Common library for code that's used across the Client, Worker, and/or Workflow

172 lines 9.54 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.makeProtoEnumConverters = makeProtoEnumConverters; const errors_1 = require("../errors"); /** * Create encoding and decoding functions to convert between the numeric `enum` types produced by our * Protobuf compiler and "const object of strings" enum values that we expose in our public APIs. * * ### Usage * * Newly introduced enums should follow the following pattern: * * ```ts * type ParentClosePolicy = (typeof ParentClosePolicy)[keyof typeof ParentClosePolicy]; * const ParentClosePolicy = { * TERMINATE: 'TERMINATE', * ABANDON: 'ABANDON', * REQUEST_CANCEL: 'REQUEST_CANCEL', * } as const; * * const [encodeParentClosePolicy, decodeParentClosePolicy] = // * makeProtoEnumConverters< * coresdk.child_workflow.ParentClosePolicy, * typeof coresdk.child_workflow.ParentClosePolicy, * keyof typeof coresdk.child_workflow.ParentClosePolicy, * typeof ParentClosePolicy, * 'PARENT_CLOSE_POLICY_' // This may be an empty string if the proto enum doesn't add a repeated prefix on values * >( * { * [ParentClosePolicy.TERMINATE]: 1, // These numbers must match the ones in the proto enum * [ParentClosePolicy.ABANDON]: 2, * [ParentClosePolicy.REQUEST_CANCEL]: 3, * * UNSPECIFIED: 0, * } as const, * 'PARENT_CLOSE_POLICY_' * ); * ``` * * `makeProtoEnumConverters` supports other usage patterns, but they are only meant for * backward compatibility with former enum definitions and should not be used for new enums. * * ### Context * * Temporal's Protobuf APIs define several `enum` types; our Protobuf compiler transforms these to * traditional (i.e. non-const) [TypeScript numeric `enum`s](https://www.typescriptlang.org/docs/handbook/enums.html#numeric-enums). * * For various reasons, this is far from ideal: * * - Due to the dual nature of non-const TypeScript `enum`s (they are both a type and a value), * it is not possible to refer to an enum value from code without a "real" import of the enum type * (i.e. can't simply do `import type ...`). In Workflow code, such an import would result in * loading our entire Protobuf definitions into the workflow sandbox, adding several megabytes to * the per-workflow memory footprint, which is unacceptable; to avoid that, we need to maintain * a mirror copy of each enum types used by in-workflow APIs, and export these from either * `@temporalio/common` or `@temporalio/workflow`. * - It is not desirable for users to need an explicit dependency on `@temporalio/proto` just to * get access to these enum types; we therefore made it a common practice to reexport these enums * from our public facing packages. However, experience demontrated that these reexports effectively * resulted in poor and inconsistent documentation coverage compared to mirrored enums types. * - Our Protobuf enum types tend to follow a verbose and redundant naming convention, which feels * unatural and excessive according to most TypeScript style guides; e.g. instead of * `workflowIdReusePolicy: WorkflowIdReusePolicy.WORKFLOW_ID_REUSE_POLICY_REJECT_DUPLICATE`, * a TypeScript developer would generally expect to be able to write something similar to * `workflowIdReusePolicy: 'REJECT_DUPLICATE'`. * - Because of the way Protobuf works, many of our enum types contain an `UNSPECIFIED` value, which * is used to explicitly identify a value that is unset. In TypeScript code, the `undefined` value * already serves that purpose, and is definitely more idiomatic to TS developers, whereas these * `UNSPECIFIED` values create noise and confusion in our APIs. * - TypeScript editors generally do a very bad job at providing autocompletion that implies reaching * for values of a TypeScript enum type, forcing developers to explicitly type in at least part * of the name of the enum type before they can get autocompletion for its values. On the other * hand, all TS editors immediately provide autocompletion for string union types. * - The [TypeScript's official documentation](https://www.typescriptlang.org/docs/handbook/enums.html#objects-vs-enums) * itself suggests that, in modern TypeScript, the use of `as const` objects may generally suffice * and may be advantageous over the use of `enum` types. * * A const object of strings, combined with a union type of possible string values, provides a much * more idiomatic syntax and a better DX for TypeScript developers. This however requires a way to * convert back and forth between the `enum` values produced by the Protobuf compiler and the * equivalent string values. * * This helper dynamically creates these conversion functions for a given Protobuf enum type, * strongly building upon specific conventions that we have adopted in our Protobuf definitions. * * ### Validations * * The complex type signature of this helper is there to prevent most potential incoherencies * that could result from having to manually synchronize the const object of strings enum and the * conversion table with the proto enum, while not requiring a regular import on the Protobuf enum * itself (so it can be used safely for enums meant to be used from workflow code). * * In particular, failing any of the following invariants will result in build time errors: * * - For every key of the form `PREFIX_KEY: number` in the proto enum, excluding the `UNSPECIFIED` key: * - There MUST be a corresponding `KEY: 'KEY'` entry in the const object of strings enum; * - There MAY be a corresponding `PREFIX_KEY: 'KEY'` in the const object of strings enum * (this is meant to preserve backward compatibility with the former syntax; such aliases should * not be added for new enums and enum entries introduced going forward); * - There MUST be a corresponding `KEY: number` in the mapping table. * - If the proto enum contains a `PREFIX_UNSPECIFIED` entry, then: * - There MAY be a corresponding `PREFIX_UNSPECIFIED: undefined` and/or `UNSPECIFIED: undefined` * entries in the const object of strings enum — this is meant to preserve backward compatibility * with the former syntax; this alias should not be added for new enums introduced going forward; * - There MUST be an `UNSPECIFIED: 0` in the mapping table. * - The const object of strings enum MUST NOT contain any other keys than the ones mandated or * optionally allowed be the preceeding rules. * - The mapping table MUST NOT contain any other keys than the ones mandated above. * * These rules notably ensure that whenever a new value is added to an existing Proto enum, the code * will fail to compile until the corresponding entry is added on the const object of strings enum * and the mapping table. * * @internal */ function makeProtoEnumConverters(mapTable, prefix) { const reverseTable = Object.fromEntries(Object.entries(mapTable).map(([k, v]) => [v, k])); const hasUnspecified = mapTable['UNSPECIFIED'] === 0 || mapTable[`${prefix}UNSPECIFIED`] === 0; function isShortStringEnumKeys(x) { return typeof x === 'string' && x in mapTable; } function isNumericEnumValue(x) { return typeof x === 'number' && x in reverseTable; } function encode(input) { if (input == null) { return undefined; } else if (typeof input === 'string') { let shorten = input; if (shorten.startsWith(prefix)) { shorten = shorten.slice(prefix.length); } if (isShortStringEnumKeys(shorten)) { return mapTable[shorten]; } throw new errors_1.ValueError(`Invalid enum value: '${input}'`); } else if (typeof input === 'number') { return input; } else { throw new errors_1.ValueError(`Invalid enum value: '${input}' of type ${typeof input}`); } } function decode(input) { if (input == null) { return undefined; } else if (typeof input === 'number') { if (hasUnspecified && input === 0) { return undefined; } if (isNumericEnumValue(input)) { return reverseTable[input]; } // We got a proto enum value that we don't yet know about (i.e. it didn't exist when this code // was compiled). This is certainly a possibility, but given how our APIs evolve, this is is // unlikely to be a terribly bad thing by itself (we avoid adding new enum values in places // that would break backward compatibility with existing deployed code). Therefore, throwing // on "unexpected" values is likely to end up causing more problems than it might avoid, // especially given that the decoded value may actually never get read anwyay. // // Therefore, we instead cheat on type constraints and return a string of the form "unknown_23". // That somewhat mirrors the behavior we'd get with the pure numerical approach. return `unknown_${input}`; } throw new errors_1.ValueError(`Invalid proto enum value: '${input}' of type ${typeof input}`); } return [encode, decode]; } //# sourceMappingURL=enums-helpers.js.map