terminal is a PHP extension for basic terminal capabilities on Unix-like systems and Windows.
The first cut stays small on purpose. It exposes the pieces that are awkward to normalize in userland, especially once Windows enters the picture, without trying to become a full TUI toolkit.
Created and maintained by Pratik Bhujel.
Current release: v0.4.1.
v0.4.1 keeps the Terminal\Terminal class and enum API from v0.3.0, replaces raw-mode string handles with Terminal\ModeToken, and hardens key reads, resize handling, ANSI detection, and terminal-size fallbacks. The older v0.2.0 release used the first procedural terminal_*() API.
PHP already has useful pieces such as stream_isatty() and sapi_windows_vt100_support(), but there is still no small extension that exposes a shared terminal capability layer across Unix and Windows.
The main goal is native Windows parity for PHP CLI prompts and terminal apps. Users should not need WSL just to get arrow keys, raw mode, terminal size, and safe restore behavior that already work on macOS and Linux.
This also removes two common framework workarounds: spawning stty/mode CON helpers for terminal state, and bundling a Windows-only helper executable just to read hidden password input.
Older console-oriented extensions took different paths:
ncursesandtermboxwrap external terminal librariesphp-wcliis Windows-only
This extension stays narrower:
- no ncurses dependency
- no framework coupling
- one user-facing API on both backends
The core-oriented API is namespaced and uses enums for values with a fixed set of cases:
Terminal\Terminal::getBackend(): Terminal\BackendTerminal\Terminal::isTty(Terminal\Stream $stream = Terminal\Stream::Stdout): boolTerminal\Terminal::supportsAnsi(Terminal\Stream $stream = Terminal\Stream::Stdout): boolTerminal\Terminal::enableAnsi(Terminal\Stream $stream = Terminal\Stream::Stdout): boolTerminal\Terminal::getSize(Terminal\Stream $stream = Terminal\Stream::Stdout): array{columns:int, rows:int}|falseTerminal\Terminal::write(string $data, Terminal\Stream $stream = Terminal\Stream::Stdout): int|falseTerminal\Terminal::enableRawMode(Terminal\Stream $stream = Terminal\Stream::Stdin): Terminal\ModeToken|falseTerminal\Terminal::restoreMode(Terminal\ModeToken $mode): boolTerminal\Terminal::readKey(?float $timeout = null, ?float $sequenceTimeout = null): Terminal\Key|string|falseTerminal\Terminal::readSecret(?float $timeout = null): string|false
Enums:
Terminal\Backend:Posix,WindowsTerminal\Stream:Stdin,Stdout,StderrTerminal\Key:Up,Down,Left,Right,Enter,Backspace,Escape,Tab,Home,End,Delete,PageUp,PageDown,Resize
Terminal\Terminal::enableAnsi() enables ANSI/VT output on Windows stdout/stderr and is a no-op capability check on Unix-like terminals. A non-empty NO_COLOR disables ANSI support checks; COLORTERM=truecolor, COLORTERM=24bit, and TERM_PROGRAM are treated as positive terminal capability signals on POSIX TTYs.
Terminal\Terminal::write() accepts Terminal\Stream::Stdout and Terminal\Stream::Stderr.
Terminal\Terminal::enableRawMode() currently accepts Terminal\Stream::Stdin and returns an opaque Terminal\ModeToken that should be passed back to Terminal\Terminal::restoreMode().
Terminal\ModeToken is process-local, non-serializable, and intentionally has no public constructor or properties.
Terminal\Terminal::enableRawMode() leaves terminal output processing intact, so normal prompt output such as "\n" keeps working while input is read one key at a time.
On POSIX, raw-mode switches use TCSANOW so mode changes are immediate; callers that type ahead should not assume pending input was drained first.
Terminal\Terminal::readKey() temporarily prepares standard input for key reads, restores the previous mode before returning, returns special keys as Terminal\Key cases, returns printable input as strings including UTF-8 input, and returns false when no key is available before the timeout. If standard input is already raw, readKey() preserves that state.
On POSIX, $sequenceTimeout controls how long readKey() waits for bytes that complete an escape or UTF-8 sequence after the first byte. null uses the default 25ms sequence timeout.
POSIX SIGWINCH and Windows WINDOW_BUFFER_SIZE_EVENT during readKey() return Terminal\Key::Resize.
Printable Unicode input is returned as the next encoded code point from the terminal, not as a full grapheme cluster.
Terminal\Terminal::readSecret() reads a hidden line from standard input, restores the previous mode before returning, handles backspace and UTF-8 input, and returns false on timeout or abort.
Current key input scope:
- normalized keys: arrows, enter, backspace, escape, tab, home, end, delete, page up, page down, resize
- printable input: returned as a string containing the next encoded code point
- control bytes such as Ctrl+C: returned as single-byte strings by
readKey() - not normalized yet: F1-F12, modifier combinations, and full grapheme clusters
- unknown POSIX escape sequences: fall back to
Terminal\Key::Escape
The earlier procedural API was removed while the project is still pre-1.0 so the extension can track the PHP core discussion more closely.
Example:
use Terminal\Key;
use Terminal\ModeToken;
use Terminal\Stream;
use Terminal\Terminal;
Terminal::enableAnsi(Stream::Stdout);
$mode = Terminal::enableRawMode();
try {
$key = Terminal::readKey();
if ($key === Key::Up) {
Terminal::write("up\n");
} elseif (is_string($key)) {
Terminal::write("typed {$key}\n");
}
} finally {
if ($mode instanceof ModeToken) {
Terminal::restoreMode($mode);
}
}After you build and install it, enable it like any normal PHP extension:
extension=terminalextension=php_terminal.dllThe v0.4.1 release is available at:
https://github.com/prateekbhujel/php-terminal/releases/tag/v0.4.1
Windows builds are attached for PHP 8.2-8.5, x64, TS/NTS. These are native Windows builds for normal Windows PHP runtimes, not WSL. Pick the zip that matches your PHP version and thread-safety mode, copy php_terminal.dll into your PHP extension directory, and enable it with:
extension=php_terminal.dllBuild from source on Unix-like systems:
git clone https://github.com/prateekbhujel/php-terminal.git
cd php-terminal
git checkout v0.4.1
phpize
./configure --enable-terminal
make
make testThen try the prompt example:
php -d extension=modules/terminal.so examples/prompt.phpFor installed builds, use your normal extension=terminal configuration instead of -d extension=....
To test unreleased changes after v0.4.1:
phpize
./configure --enable-terminal
make
make testThen try the prompt example:
php -d extension=modules/terminal.so examples/prompt.phpFor installed builds, use your normal extension=terminal configuration instead of -d extension=....
XAMPP, MAMP, WAMP, Laragon, Herd, and plain downloaded PHP builds work as long as the DLL matches the PHP build your app actually runs.
Example PHP binaries:
- XAMPP:
C:\xampp\php\php.exe - MAMP:
C:\MAMP\bin\php\php8.x.x\php.exe - Laragon:
C:\laragon\bin\php\php-8.x.x\php.exe - Plain PHP zip:
C:\php\php.exe
Check the PHP version, thread-safety mode, architecture, and compiler:
set PHP_BIN=C:\xampp\php\php.exe
%PHP_BIN% -i | findstr /C:"PHP Version" /C:"Thread Safety" /C:"Architecture" /C:"Compiler"
%PHP_BIN% --iniDownload the matching zip from the release page. For example:
- PHP 8.2, thread safety disabled:
php_terminal-v0.4.1-8.2-nts-vs16-x86_64.zip - PHP 8.2, thread safety enabled:
php_terminal-v0.4.1-8.2-ts-vs16-x86_64.zip - PHP 8.4, thread safety disabled:
php_terminal-v0.4.1-8.4-nts-vs17-x86_64.zip
Copy php_terminal.dll into that PHP installation's extension directory, for example:
C:\xampp\php\ext
Then edit:
C:\xampp\php\php.ini
Add:
extension=php_terminal.dllTest it with XAMPP's CLI PHP:
%PHP_BIN% -m | findstr terminal
%PHP_BIN% examples\doctor.php
%PHP_BIN% examples\prompt.phpOnce enabled in that PHP runtime, any CLI app using the same php.exe can detect and use terminal. The app or framework still needs integration code; the extension provides the native Windows terminal primitives.
Each PHP installation has its own extension directory and php.ini. Build terminal with the phpize and php-config that belong to the PHP binary your application actually runs.
Example target PHP binaries:
- XAMPP on macOS:
/Applications/XAMPP/xamppfiles/bin/php - MAMP on macOS:
/Applications/MAMP/bin/php/php8.x.x/bin/php - LAMPP/XAMPP on Linux:
/opt/lampp/bin/php - Homebrew or system PHP:
$(command -v php)
Build for that target PHP:
git clone https://github.com/prateekbhujel/php-terminal.git
cd php-terminal
PHP_BIN=/Applications/XAMPP/xamppfiles/bin/php
PHPIZE="$(dirname "${PHP_BIN}")/phpize"
PHP_CONFIG="$(dirname "${PHP_BIN}")/php-config"
"${PHPIZE}"
if [ "$(uname -s)" = "Darwin" ]; then
PHP_ARCH=$("${PHP_BIN}" -r 'echo php_uname("m");')
CFLAGS="-arch ${PHP_ARCH}" \
LDFLAGS="-arch ${PHP_ARCH}" \
./configure --with-php-config="${PHP_CONFIG}" --enable-terminal
else
./configure --with-php-config="${PHP_CONFIG}" --enable-terminal
fi
make
make test
sudo make installBuild from a path without spaces. The PHP extension build tooling rejects
whitespace in the build path before configure runs.
The macOS architecture step matters on Apple Silicon when a PHP distribution runs as x86_64 under Rosetta. Without matching that architecture, macOS may build an arm64 terminal.so that the target PHP cannot load.
Find the right php.ini:
"${PHP_BIN}" --iniAdd this to that PHP installation's loaded php.ini:
extension=terminal.soTest it:
"${PHP_BIN}" -m | grep terminal
"${PHP_BIN}" examples/doctor.php
"${PHP_BIN}" examples/prompt.phpOnce enabled there, any PHP application using that same PHP binary can use terminal. If another app uses a different PHP binary, build/install the extension for that PHP too.
The goal is to let Laravel Prompts and similar prompt libraries behave on native Windows the same way they behave on macOS and Linux.
This extension is the native terminal layer for that. It does not monkey-patch Laravel Prompts, Symfony Console, or any other CLI framework by itself. Frameworks still need to opt in, but the hard part is exposed through one API: raw mode, safe terminal restore, single-key reads, terminal size, direct writes, and shared Unix/Windows key names.
That means the end-user path should be: install the matching Windows php_terminal.dll, enable it in the PHP used by the app, install or use an adapter that detects terminal, then run prompts from PowerShell, Command Prompt, or Windows Terminal without WSL.
A Laravel Prompts adapter would keep Laravel's existing prompt code, but swap the terminal backend when this extension is available:
- use
Terminal\Terminal::isTty(Terminal\Stream::Stdin)for interactivity checks - use
Terminal\Terminal::enableAnsi()before rendering ANSI prompts on Windows - use
Terminal\Terminal::enableRawMode()andTerminal\Terminal::restoreMode()instead ofstty - use
Terminal\Terminal::readKey()for input - use
Terminal\Terminal::readSecret()for password/secret prompts instead of shelling out to platform-specific helpers - map
Terminal\Key::*cases back to Laravel Prompts' existingKey::*values - use
Terminal\Terminal::getSize()for columns and rows
The key mapping is intentionally small and predictable:
Terminal\Terminal::readKey() |
Laravel Prompts key |
|---|---|
Terminal\Key::Up |
Key::UP |
Terminal\Key::Down |
Key::DOWN |
Terminal\Key::Left |
Key::LEFT |
Terminal\Key::Right |
Key::RIGHT |
Terminal\Key::Enter |
Key::ENTER |
Terminal\Key::Backspace |
Key::BACKSPACE |
Terminal\Key::Delete |
Key::DELETE |
Terminal\Key::Escape |
Key::ESCAPE |
Terminal\Key::Tab |
Key::TAB |
Terminal\Key::Home |
first Key::HOME value |
Terminal\Key::End |
first Key::END value |
Terminal\Key::PageUp |
Key::PAGE_UP |
Terminal\Key::PageDown |
Key::PAGE_DOWN |
Printable input is returned as the typed character, so normal text prompts do not need special handling.
Until Laravel Prompts has that adapter, existing Laravel Prompts releases will still use their current Windows fallback behavior. XAMPP and Windows users can test the same low-level prompt behavior today with examples/prompt.php.
The bundled examples/prompt.php file is intentionally small so framework authors can see the shape without reading a full TUI library.
Future Laravel Prompts adapter work should target the Terminal\Terminal and enum API from v0.4.1.
For release feedback, open a new issue with the OS, terminal, PHP version, extension version, what you tried, and the behavior you expected.
phpize
./configure --enable-terminal
make
make testconfig.w32 is included, and the CI workflow builds the extension with php/php-windows-builder.
<?php
if (!extension_loaded('terminal')) {
die("terminal is not loaded\n");
}
use Terminal\Terminal;
use Terminal\ModeToken;
var_dump(Terminal::getBackend());
var_dump(Terminal::isTty());
var_dump(Terminal::supportsAnsi());
var_dump(Terminal::getSize());
Terminal::write("hello from terminal\n");
$mode = Terminal::enableRawMode();
if ($mode instanceof ModeToken) {
try {
$key = Terminal::readKey(0.5);
} finally {
Terminal::restoreMode($mode);
}
}There are runnable examples in examples/basic.php, examples/doctor.php, and examples/prompt.php.