Build/Submit details page URL
No response
Summary
Symptom
When eas workflow:run is invoked on a CI agent without a controlling terminal, the command emits one line of output (Using workflow file from …) and then hangs indefinitely. No workflow run is ever created on the EAS side (verified via the dashboard). The same agent successfully runs eas build with identical flags.
Reproduced on Azure DevOps ubuntu-latest agents with eas-cli 19.x.
Workaround
Redirect stdin from /dev/null:
npx eas workflow:run path/to/workflow.yml --non-interactive --json </dev/null
With this, the command completes in seconds and the workflow run is created normally.
Root cause
packages/eas-cli/src/commands/workflow/run.ts:224 unconditionally calls maybeReadStdinAsync() at the top of the command:
const stdinData = await maybeReadStdinAsync();
The implementation in packages/eas-cli/src/commandUtils/workflow/utils.ts:273 only short-circuits when process.stdin.isTTY is true:
export async function maybeReadStdinAsync(): Promise<string | null> {
if (process.stdin.isTTY) {
return null;
}
return await new Promise((resolve, reject) => {
let data = '';
process.stdin.setEncoding('utf8');
process.stdin.on('readable', () => { /* … */ });
process.stdin.on('end', () => {
resolve(data.trim() || null);
});
process.stdin.on('error', err => reject(err));
});
}
On a CI agent stdin is typically connected to an open pipe from the agent harness — not a TTY, but never emits 'end' either. The await new Promise(...) never resolves, the command hangs forever.
eas build does not have this issue because it does not call maybeReadStdinAsync; inputs come from flags only.
Suggested fixes
Any of:
- Skip stdin reading entirely when
--non-interactive is set. Easiest fix.
- Skip stdin reading when no
--input JSON is expected, i.e. when all required inputs are provided via -F flags.
- Check
process.stdin.readableEnded / destroyed before attaching the listener, and short-circuit if stdin is unreadable.
- Apply a short timeout to the stdin read (e.g. 1s) and fall back to
null if no data arrives.
(Personally I'd combine 1+3 — --non-interactive is a clear signal not to expect stdin, and the readable-state check guards the remaining edge cases.)
Managed or bare?
managed
Environment
System:
OS: macOS 26.4
Shell: 5.9 - /bin/zsh
Binaries:
Node: 25.9.0 - /opt/homebrew/bin/node
Yarn: 1.22.22 - /opt/homebrew/bin/yarn
npm: 11.12.1 - /opt/homebrew/bin/npm
Watchman: 2026.03.30.00 - /opt/homebrew/bin/watchman
Managers:
CocoaPods: 1.16.2 - /opt/homebrew/bin/pod
SDKs:
iOS SDK:
Platforms: DriverKit 25.4, iOS 26.4, macOS 26.4, tvOS 26.4, visionOS 26.4, watchOS 26.4
IDEs:
Android Studio: 2025.3 AI-253.30387.90.2532.14935130
Xcode: 26.4.1/17E202 - /usr/bin/xcodebuild
npmGlobalPackages:
eas-cli: 19.0.5
Expo Workflow: managed
Error output
No response
Reproducible demo or steps to reproduce from a blank project
See above
Build/Submit details page URL
No response
Summary
Symptom
When
eas workflow:runis invoked on a CI agent without a controlling terminal, the command emits one line of output (Using workflow file from …) and then hangs indefinitely. No workflow run is ever created on the EAS side (verified via the dashboard). The same agent successfully runseas buildwith identical flags.Reproduced on Azure DevOps
ubuntu-latestagents witheas-cli19.x.Workaround
Redirect stdin from
/dev/null:npx eas workflow:run path/to/workflow.yml --non-interactive --json </dev/nullWith this, the command completes in seconds and the workflow run is created normally.
Root cause
packages/eas-cli/src/commands/workflow/run.ts:224unconditionally callsmaybeReadStdinAsync()at the top of the command:The implementation in
packages/eas-cli/src/commandUtils/workflow/utils.ts:273only short-circuits whenprocess.stdin.isTTYis true:On a CI agent stdin is typically connected to an open pipe from the agent harness — not a TTY, but never emits
'end'either. Theawait new Promise(...)never resolves, the command hangs forever.eas builddoes not have this issue because it does not callmaybeReadStdinAsync; inputs come from flags only.Suggested fixes
Any of:
--non-interactiveis set. Easiest fix.--inputJSON is expected, i.e. when all required inputs are provided via-Fflags.process.stdin.readableEnded/destroyedbefore attaching the listener, and short-circuit if stdin is unreadable.nullif no data arrives.(Personally I'd combine 1+3 —
--non-interactiveis a clear signal not to expect stdin, and the readable-state check guards the remaining edge cases.)Managed or bare?
managed
Environment
Error output
No response
Reproducible demo or steps to reproduce from a blank project
See above