# 🤓 TypeScript
## Available types
The following types can be imported: [`ResultPromise`](api.md#return-value), [`Subprocess`](api.md#subprocess), [`Result`](api.md#result), [`ExecaError`](api.md#execaerror), [`Options`](api.md#options-1), [`StdinOption`](api.md#optionsstdin), [`StdoutStderrOption`](api.md#optionsstdout), [`TemplateExpression`](api.md#execacommand), [`Message`](api.md#subprocesssendmessagemessage-sendmessageoptions), [`VerboseObject`](api.md#verbose-object), [`ExecaMethod`](api.md#execaoptions), [`ExecaNodeMethod`](api.md#execanodeoptions) and [`ExecaScriptMethod`](api.md#options).
```ts
import {
execa as execa_,
ExecaError,
type ResultPromise,
type Result,
type Options,
type StdinOption,
type StdoutStderrOption,
type TemplateExpression,
type Message,
type VerboseObject,
type ExecaMethod,
} from 'execa';
const execa: ExecaMethod = execa_({preferLocal: true});
const options: Options = {
stdin: 'inherit' satisfies StdinOption,
stdout: 'pipe' satisfies StdoutStderrOption,
stderr: 'pipe' satisfies StdoutStderrOption,
timeout: 1000,
ipc: true,
verbose(verboseLine: string, verboseObject: VerboseObject) {
return verboseObject.type === 'duration' ? verboseLine : undefined;
},
};
const task: TemplateExpression = 'build';
const message: Message = 'hello world';
try {
const subprocess: ResultPromise = execa(options)`npm run ${task}`;
await subprocess.sendMessage?.(message);
const result: Result = await subprocess;
console.log(result.stdout);
} catch (error) {
if (error instanceof ExecaError) {
console.error(error);
}
}
```
## Synchronous execution
Their [synchronous](#synchronous-execution) counterparts are [`SyncResult`](api.md#result), [`ExecaSyncError`](api.md#execasyncerror), [`SyncOptions`](api.md#options-1), [`StdinSyncOption`](api.md#optionsstdin), [`StdoutStderrSyncOption`](api.md#optionsstdout), [`TemplateExpression`](api.md#execacommand), [`SyncVerboseObject`](api.md#verbose-object), [`ExecaSyncMethod`](api.md#execasyncoptions) and [`ExecaScriptSyncMethod`](api.md#syncoptions).
```ts
import {
execaSync as execaSync_,
ExecaSyncError,
type SyncResult,
type SyncOptions,
type StdinSyncOption,
type StdoutStderrSyncOption,
type TemplateExpression,
type SyncVerboseObject,
type ExecaSyncMethod,
} from 'execa';
const execaSync: ExecaSyncMethod = execaSync_({preferLocal: true});
const options: SyncOptions = {
stdin: 'inherit' satisfies StdinSyncOption,
stdout: 'pipe' satisfies StdoutStderrSyncOption,
stderr: 'pipe' satisfies StdoutStderrSyncOption,
timeout: 1000,
verbose(verboseLine: string, verboseObject: SyncVerboseObject) {
return verboseObject.type === 'duration' ? verboseLine : undefined;
},
};
const task: TemplateExpression = 'build';
try {
const result: SyncResult = execaSync(options)`npm run ${task}`;
console.log(result.stdout);
} catch (error) {
if (error instanceof ExecaSyncError) {
console.error(error);
}
}
```
## Type inference
The above examples demonstrate those types. However, types are automatically inferred. Therefore, explicit types are only needed when defining functions that take those values as parameters.
```ts
import {
execa as execa_,
ExecaError,
type Result,
type VerboseObject,
} from 'execa';
const execa = execa_({preferLocal: true});
const printResultStdout = (result: Result) => {
console.log('Stdout', result.stdout);
};
const options = {
stdin: 'inherit',
stdout: 'pipe',
stderr: 'pipe',
timeout: 1000,
ipc: true,
verbose(verboseLine: string, verboseObject: VerboseObject) {
return verboseObject.type === 'duration' ? verboseLine : undefined;
},
} as const;
const task = 'build';
const message = 'hello world';
try {
const subprocess = execa(options)`npm run ${task}`;
await subprocess.sendMessage(message);
const result = await subprocess;
printResultStdout(result);
} catch (error) {
if (error instanceof ExecaError) {
console.error(error);
}
}
```
## Troubleshooting
### Supported version
The minimum supported TypeScript version is [`5.1.6`](https://github.com/microsoft/TypeScript/releases/tag/v5.1.6).
### ES modules
This package uses pure ES modules. Therefore the TypeScript's `--module` compiler option must be set to [`nodenext`](https://www.typescriptlang.org/docs/handbook/modules/reference.html#node16-nodenext) or [`preserve`](https://www.typescriptlang.org/docs/handbook/modules/reference.html#preserve). [More info.](https://gist.github.com/sindresorhus/a39789f98801d908bbc7ff3ecc99d99c)
Otherwise, transpilation will work, but running the transpiled file will throw the following runtime error:
```
Error [ERR_REQUIRE_ESM]: require() of ES Module ... not supported.
```
Or:
```
ReferenceError: exports is not defined in ES module scope
```
### Strict unions
Several options are typed as unions of strings: [`stdin`](api.md#optionsstdin), [`stdout`](api.md#optionsstdout), [`stderr`](api.md#optionsstderr), [`encoding`](api.md#optionsencoding), [`serialization`](api.md#optionsserialization), [`verbose`](api.md#optionsverbose), [`killSignal`](api.md#optionskillsignal), [`from`](api.md#pipeoptionsfrom) and [`to`](api.md#pipeoptionsto). For example, the `serialization` option's type is `'advanced' | 'json'`, not `string`. Therefore the following example fails:
```ts
import {execa} from 'execa';
// Type error: "No overload matches this call"
const spawnSubprocess = (serialization: string) => execa({serialization})`npm run build`;
// Without `as const`, `options.serialization` is typed as `string`, not `'json'`
const options = {serialization: 'json'};
// Type error: "No overload matches this call"
await execa(options)`npm run build`;
```
But this works:
```ts
import {execa, type Options} from 'execa';
const spawnSubprocess = (serialization: Options['serialization']) => execa({serialization})`npm run build`;
const options = {serialization: 'json'} as const;
await execa(options)`npm run build`;
```
[**Next**: 📔 API reference](api.md)\
[**Previous**: 🐭 Small packages](small.md)\
[**Top**: Table of contents](../readme.md#documentation)