Virtual terminal with shell-like commands over a pluggable filesystem.
Project description
termish 📺
Virtual terminal with shell-like commands over a pluggable filesystem.
Parses and executes shell scripts (pipelines, redirects, semicolons) against any object that implements the FileSystem protocol. Zero runtime dependencies. Pure Python.
Features
- Shell parser -- pipes, redirects (
>,>>,<), semicolons, quoted strings, line continuation - 30 builtins -- ls, cat, grep, find, sed, tr, sort, uniq, cut, wc, diff, tar, gzip, zip, jq, xargs, basename, dirname, ...
- Custom commands -- inject your own command handlers alongside builtins; injected commands override builtins and compose in pipelines
- jq engine -- built-in jq filter parser and evaluator (field access, pipes, functions, conditionals)
- Pluggable filesystem --
FileSystemis atyping.Protocol; any object with the right methods works - MemoryFS included -- in-memory filesystem for testing and lightweight use
Install
pip install termish
Quick example
from termish import execute, MemoryFS
fs = MemoryFS()
execute("mkdir -p src", fs)
execute("echo 'def main(): pass' > src/app.py", fs)
execute("echo 'import os' > src/utils.py", fs)
# Pipelines work
output = execute("grep -r 'def' src | wc -l", fs)
print(output) # 1
# jq works
execute('echo \'{"name": "alice", "score": 42}\' > data.json', fs)
output = execute('jq -r ".name" data.json', fs)
print(output) # alice
Custom commands
Inject your own commands via the commands parameter. They receive a CommandContext and compose naturally with builtins in pipelines:
from termish import execute, MemoryFS, CommandContext, CommandResult
def greet(ctx: CommandContext) -> CommandResult | None:
name = ctx.args[0] if ctx.args else "world"
ctx.stdout.write(f"hello {name}\n")
return None
fs = MemoryFS()
output = execute("greet alice | wc -c", fs, commands={"greet": greet})
print(output) # 12
# Injected commands override builtins with the same name
All commands — builtin and injected — use the same CommandContext signature. See CommandContext, CommandResult, and CommandFunc in termish.context and termish.errors.
FileSystem protocol
Any object implementing these 16 methods works with termish -- no inheritance required:
class FileSystem(Protocol):
def getcwd(self) -> str: ...
def chdir(self, path: str) -> None: ...
def read(self, path: str) -> bytes: ...
def write(self, path: str, content: bytes, mode: str = "w") -> None: ...
def exists(self, path: str) -> bool: ...
def isfile(self, path: str) -> bool: ...
def isdir(self, path: str) -> bool: ...
def stat(self, path: str) -> FileMetadata: ...
def mkdir(self, path: str, parents: bool = False, exist_ok: bool = False) -> None: ...
def makedirs(self, path: str, exist_ok: bool = True) -> None: ...
def remove(self, path: str) -> None: ...
def rmdir(self, path: str) -> None: ...
def rename(self, src: str, dst: str) -> None: ...
def list(self, path: str = ".", recursive: bool = False) -> list[str]: ...
def list_detailed(self, path: str = ".", recursive: bool = False) -> list[FileInfo]: ...
def glob(self, pattern: str) -> list[str]: ...
Part of the agex stack
termish provides shell commands for AI agents in agex, operating over virtual filesystems from monkeyfs.
Compatible filesystems
monkeyfs VirtualFS and IsolatedFS both satisfy the termish FileSystem protocol and can be passed directly to execute().
Builtin commands
| Category | Commands |
|---|---|
| Filesystem | pwd, cd, mkdir, ls, touch, cp, mv, rm, basename, dirname |
| I/O | echo, cat, head, tail, tee |
| Search | grep, find |
| Text | wc, sort, uniq, cut, sed, tr |
| Diff | diff |
| Archive | tar, gzip, gunzip, zip, unzip |
| Meta | xargs |
| JSON | jq |
Development
uv sync --extra dev
uv run pytest
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file termish-0.1.6.tar.gz.
File metadata
- Download URL: termish-0.1.6.tar.gz
- Upload date:
- Size: 78.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e99918069a4a6bd74c265f09770dc027faecaaa54310fabd39bfbf76bfc50529
|
|
| MD5 |
31ba0f0a7f4d84df4ff9e795a986b403
|
|
| BLAKE2b-256 |
c5bf24442fb9d299aff973275616ec53d68487962d9192afb8ca79bc9456139a
|
File details
Details for the file termish-0.1.6-py3-none-any.whl.
File metadata
- Download URL: termish-0.1.6-py3-none-any.whl
- Upload date:
- Size: 60.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b8cd2a75df11b0b50021d52b19edf811253b9ad1848ad1f7f8fe242543f6c93c
|
|
| MD5 |
bf62daca2b23c2229dcaf5cf2c6d43d7
|
|
| BLAKE2b-256 |
a10c8c95d6b2e8b83279f098a9ba9eb4f2922b0cb4c4703ff61ffaaef3589b80
|