Skip to main content

Python-first interactive shell for Debian and Unix-like systems

Project description

PySH logo

PySH

Python-first interactive shell for Debian and Unix-like systems.

CI status PyPI version Supported Python versions License GitHub release Debian and Unix-like systems Python-first shell


Python-first interactive shell for Debian and Unix-like systems.

PySH is a small, dependency-free interactive shell written in pure Python. It is packaged as a regular PyPI distribution (pysh-shell), installs a single console command (pysh), and is designed to feel familiar to anyone used to a Bourne-style shell while remaining hackable from Python.

Current development version: 0.2.1. PySH targets Python 3.13+ and is validated primarily on Debian 13 and Unix-like systems.


Features

  • Interactive REPL with persistent history (~/.pysh_history).
  • Bash-like reverse incremental search bound to Ctrl+R when GNU readline is available; degrades silently on other backends.
  • Welcome banner and configurable prompt with user and CWD (~ collapsed).
  • Robust quote-aware parser:
    • Splits chains on ;, &&, || only outside of quotes.
    • Splits pipelines on | only outside of quotes.
  • Pipelines with correctly managed file-descriptor handover.
  • Redirection: <, >, >>, 2>, 2>>, &>, &>>.
  • Command substitution: $(command) and `command`. Quote-aware: evaluated inside double quotes, suppressed inside single quotes. Bounded by a 5-second timeout by default.
  • Local variables (NAME=value) and exported environment variables (export NAME=value) with $NAME / ${NAME} expansion.
  • Aliases with sane defaults; alias and unalias builtins.
  • Migration layer for zsh/bash/sh: source_zsh <file> preserves the existing alias importer, source_zsh_profile <file> and source_sh_aliases <file> statically import simple aliases, exports and assignments without executing profile code, run_script <file> [args...] delegates shebang scripts to their real interpreter, and compat_check <file> reports migration risk before execution.
  • Zsh Transition Layer for explicit delegation: zsh <command> delegates to real zsh when installed, and zsh_fallback can be enabled for controlled fallback experiments.
  • Python-native runtime bridge: py <code> executes one-line Python code in a persistent per-session runtime context.
  • Startup file ~/.pyshrc plus a plugin directory at ~/.pyshrc.d/ whose *.pysh files load in deterministic lexicographic order.
  • Mini rc-interpreter for control flow inside ~/.pyshrc and plugins: if/else/fi, for/do/done, while/do/done (with a hard iteration safety limit).
  • Directory stack: pushd, popd, dirs.
  • svc builtin for PyInit-style service control by PID file: svc list, svc status, svc stop, svc restart, svc start.
  • PyInit service metadata parser with dependency tracking (depends: [network]).
  • Safe ANSI color helpers that respect NO_COLOR and TERM=dumb, used for the banner and diagnostics. The input line itself is left untouched so editing remains stable.
  • Basic tab completion for aliases, builtins, files and directories.
  • Clean Ctrl+C (cancels current line, keeps the shell alive) and Ctrl+D (exits the shell).

Installation

From PyPI

pip install pysh-shell

Then start the shell with:

pysh

Quick start

pysh
pysh --version
pysh -c "echo hi"
alias ll='ls -lah'
source_zsh_profile ~/.zshrc
source_sh_aliases ~/.bash_aliases
compat_check ~/scripts/maintenance.sh
run_script ~/scripts/maintenance.sh --dry-run
py import platform; print(platform.platform())

Development install

python3.13 -m venv .venv
source .venv/bin/activate
python -m pip install --upgrade pip
python -m pip install -e ".[dev]"

Running

pysh           # console entry point installed by the wheel
python -m pysh # equivalent module entry point
pysh --version # print version and exit
pysh -c "echo hi; echo there"  # run one command line and exit

Documentation

Full documentation lives under docs/:

  • Installation — installing from PyPI and dev install.
  • Usage — invocation, operators, pipelines, redirection, command substitution, variables, builtins.
  • Builtins — syntax, examples, return behavior and limitations for every builtin.
  • Operators — chains, pipelines, redirection, command substitution, quoting and parser limitations.
  • Configuration~/.pyshrc, plugins under ~/.pyshrc.d/, aliases, exports, prompt behavior.
  • Migration — static profile import, script transition runner, and compatibility reporting.
  • Zsh compatibility — transition bridge, safe profile import, explicit zsh delegation, optional fallback mode.
  • Python runtime — persistent Python-native py execution context.
  • Development — running the test suite, linting, building artifacts, repository layout.
  • Release process — how PySH 0.2.1 ships via GitHub Actions and PyPI Trusted Publishing.
  • Limitations — explicit non-goals and compatibility boundaries.
  • Documentation policy — required coverage for new commands, parser behavior, migration helpers and limitations.

Builtins

Available shell builtins. Most run inside the shell process; transition builtins such as zsh and run_script may delegate explicitly as documented.

Builtin Description
cd Change the current working directory.
pwd Print the current working directory.
alias Define or display aliases.
unalias Remove one or more aliases.
export Define or display exported environment vars.
source Execute commands from a file (also .).
source_zsh Safely import simple aliases from a zsh-compatible file.
source_zsh_profile Statically import simple zsh profile entries.
source_sh_aliases Statically import simple sh/bash aliases and vars.
run_script Run a script through a shebang interpreter or native PySH lines.
compat_check Produce a static migration report for a shell file.
zsh Execute one command through real zsh -lc.
zsh_fallback Enable or disable explicit zsh fallback mode.
py Execute Python code in the persistent PySH runtime.
pushd Push CWD onto the directory stack and cd to a path.
popd Pop the directory stack and cd to the popped entry.
dirs Print the current directory followed by the stack.
svc Query / signal PyInit services. See svc / PyInit.
exit Exit the shell with an optional status code.
quit Same as exit.

Operators

Operator Meaning
cmd1 ; cmd2 Run cmd1, then unconditionally run cmd2.
cmd1 && cmd2 Run cmd2 only if cmd1 exits with status 0.
cmd1 || cmd2 Run cmd2 only if cmd1 exits with non-zero status.
cmd1 | cmd2 Pipe cmd1's stdout into cmd2's stdin.

Operators inside single or double quotes are treated as literal text.

echo "🐍 PySH v0.2.1 | Python 3.13.5"
echo "Test | pipe & semicolon; && ok"
python3.13 -c "import subprocess; print('ok')"

Redirection

Syntax Effect
< file Read stdin from file.
> file Write stdout to file (truncate).
>> file Write stdout to file (append).
2> file Write stderr to file (truncate).
2>> file Write stderr to file (append).
&> file Write stdout + stderr to file.
&>> file Append stdout + stderr to file.
ls -la 2>/dev/null | head -3
python3.13 -c "import sys; print('err', file=sys.stderr)" 2> err.log
echo "hello" > out.txt
echo "again" >> out.txt

Redirection operators inside quotes are kept as literal characters.


Pipelines

PySH connects each stage with a real OS pipe and closes the parent's duplicate handles after the child is spawned, so neither side deadlocks.

ls -la | head -3
apt list --upgradable 2>/dev/null | grep -c "/"

Command substitution

echo "Kernel: `uname -r`"
echo "Date: $(date '+%Y-%m-%d')"
echo 'No substitution: $(date)'
  • $(...) and `...` are both supported.
  • Quotes are honoured: substitutions inside single quotes are kept literally; substitutions inside double quotes are evaluated.
  • Each substitution runs with a 5-second timeout by default. On timeout or failure the substitution expands to an empty string and the shell remains usable.

Variables

NAME=world           # local shell variable
export GREETING=hi   # exported environment variable
echo "$GREETING, $NAME"

Local variables shadow environment variables when expanded. Single quotes suppress expansion; double quotes do not.


Zsh Transition Layer

PySH is Python-first, not a full zsh clone. The zsh compatibility bridge is for transition: it lets users move aliases and selected legacy commands into PySH without pretending that every zsh grammar feature is native.

source_zsh ~/.zsh_aliases
source_zsh_profile ~/.zshrc
source_sh_aliases ~/.bash_aliases
compat_check ~/scripts/maintenance.sh
run_script ~/scripts/maintenance.sh --dry-run
zsh 'source ~/.zshrc; my_old_alias'
zsh 'print -r -- hello'

source_zsh <file> statically imports supported simple alias definitions such as alias ll='ls -lah', ignores comments and unsupported constructs, and never executes the file as code. It prints imported=N skipped=M file=<path> and reports malformed alias lines deterministically on stderr.

source_zsh_profile <file> and source_sh_aliases <file> use the 0.2.1 static profile importer. They read files such as ~/.zshrc, .profile and .bash_aliases without executing them, import supported simple aliases, exports and local assignments, and print aliases=N exports=N vars=N skipped=M file=<path>.

compat_check <file> produces a static report with supported, delegated, skipped and risky counts. Risky constructs such as eval, command substitution, source and shell functions cause exit status 2.

run_script <file> [args...] is an explicit transition runner. Scripts with zsh, bash or sh shebangs are delegated to the real interpreter through an argv list; no-shebang scripts are executed line-by-line by PySH's native engine where possible.

zsh <command> delegates explicitly to real zsh -lc <command>. If zsh is not installed, it returns 127 with a deterministic error.

Fallback mode is off by default. It can be enabled only explicitly:

zsh_fallback on
zsh_fallback off
PYSH_ZSH_FALLBACK=1

When fallback is on, PySH may delegate commands it cannot parse or execute natively to zsh. Builtins already handled by PySH stay native, and native command failures are not hidden.


Python Runtime

The py builtin executes one-line Python code in a persistent runtime context owned by the current PySH session:

py print("hello from python")
py import platform; print(platform.platform())
py from pathlib import Path; print(Path(".").resolve())

Variables and imports persist across py invocations:

py x = 10
py print(x)
py import pathlib
py print(pathlib.Path(".").exists())

Exceptions are printed to stderr and return non-zero without terminating the shell.


~/.pyshrc and ~/.pyshrc.d/

At startup PySH first executes ~/.pyshrc (if present), then every file in ~/.pyshrc.d/ whose name ends with .pysh. Plugins are loaded in deterministic lexicographic order so prefix numbering (10-…, 20-…) gives predictable layering.

You can also re-source any file at any time:

source ~/.pyshrc

A failing line is reported on stderr and the next line is still executed. A broken plugin does not prevent later plugins from loading.

Example ~/.pyshrc

export EDITOR="nano"
export PAGER="less"
export LANG="pl_PL.UTF-8"
export PYTHONDONTWRITEBYTECODE="1"

alias rm="rm -i"
alias cp="cp -i"
alias mv="mv -i"
alias python="python3.13"
alias pip="pip3.13"

if [ -d /opt/local/bin ]; then
    export PATH="/opt/local/bin:$PATH"
fi

for dir in ~/bin ~/.local/bin; do
    if [ -d "$dir" ]; then
        export PATH="$dir:$PATH"
    fi
done

echo "🐍 PySH 0.2.1 | Python 3.13+"
echo "💡 Operators: && || ; | > >> < 2> 2>> &> &>>  + \$() and backticks"

Example plugin ~/.pyshrc.d/10-aliases.pysh

# Loaded after ~/.pyshrc, in lexicographic order.
alias gs="git status -sb"
alias gd="git diff"
alias gl="git log --oneline --decorate"

if [ -f ~/.work_aliases ]; then
    source ~/.work_aliases
fi

Mini rc-interpreter cheat sheet

Construct Notes
if [ <cond> ]; then ... fi else block optional
for VAR in a b c; do ... done Iterates literal words
while [ <cond> ]; do ... done Bounded by a safety iteration limit

The canonical else keyword is else. The form else: is accepted as a compatibility alias.

Supported test operators:

Test Meaning
[ -f path ] path exists and is a regular file
[ -d path ] path exists and is a directory
[ -e path ] path exists
[ -z "$VAR" ] $VAR is empty
[ -n "$VAR" ] $VAR is non-empty
[ "$A" = "$B" ] string equality
[ "$A" == "$B" ] string equality (alias)
[ "$A" != "$B" ] string inequality
[ ! -f path ] etc. negate any of the above

Directory stack

pushd /tmp
pushd /var/log
dirs
popd
popd

pushd path pushes the current directory onto the stack and changes to path. popd returns to the most-recently-pushed directory. dirs prints the current directory followed by the stack contents. Popping an empty stack produces a deterministic error and a non-zero exit status.


svc / PyInit

PySH ships a small service client used by the svc builtin. It is deliberately PID-file based so it can be useful on systems that do not run a full supervisor, but it is also designed to plug into PyInit when a control interface is present.

svc list
svc status <name>
svc start <name>
svc stop <name>
svc restart <name>
  • svc list walks /run/pyinit/*.pid and prints each service as name\tactive|dead\tpid=N.
  • svc status <name> checks the PID file at /run/pyinit/<name>.pid.
  • svc stop <name> reads the PID file and sends SIGTERM.
  • svc restart <name> sends SIGTERM. Without a registered PyInit control interface it then reports that restart requires supervision.
  • svc start <name> requires a PyInit control interface. Without one it fails deterministically rather than pretending to work.

PySH never calls sudo. To control system-wide PyInit services, run PySH under an account that already has permission.

PyInit service metadata

PyInit-style service files live in ~/.config/pyinit/services/ (or a path chosen by your integration). PySH ships a strict metadata parser:

# ~/.config/pyinit/services/example.service
name: example
command: python3.13 -m http.server 8080
depends: [network]

Recognised fields:

  • name — required, identifier
  • command — required, the launch command line
  • depends — optional list of dependency names; accepts [a, b] or comma-separated form

Invalid metadata (unknown syntax, malformed dependencies, missing fields) produces a deterministic ServiceMetadataError rather than silently dropping content. This is metadata support for PyInit integration, not a complete replacement for systemd.


Tab completion

Tab completes aliases and builtins for the first word, and filesystem paths for any word. Inaccessible directories are silently skipped.


Limitations

  • No job control (&, bg, fg, jobs, Ctrl+Z job suspension).
  • No full POSIX shell grammar — only the constructs documented above.
  • No glob expansion is performed by PySH itself; external commands still receive globs through their own expansion logic when run via a system shell, but plain pipelines do not expand * / ? in arguments.
  • No full zsh compatibility. The zsh compatibility bridge is a transition layer with safe static alias import and explicit delegation to real zsh.
  • svc start and svc restart to actually re-launch a process require a PyInit control interface; without one they return a deterministic error.

Testing and quality gates

pytest -q
ruff check src tests
python -m build
twine check dist/*

The project ships unit tests for the parser, the redirection module, the rc loader and mini-interpreter, command substitution, the history manager, the highlighting helpers, the plugin loader, directory stack, unalias, the svc builtin, the PyInit metadata parser, the zsh transition layer and the Python runtime bridge.


Publishing

This repository is configured for PyPI Trusted Publishing via GitHub Actions. See .github/workflows/publish.yml — it uses pypa/gh-action-pypi-publish@release/v1 with id-token: write and the pypi environment. Tagging a release on GitHub publishes the build.

Do not publish from a developer machine; let the workflow do it.


Target platform

  • Primary target: Debian 13 with Python 3.13+.
  • Should work on any POSIX system with a working subprocess and readline, but only Debian 13 is regularly validated.

License

PySH is distributed under the GNU General Public License v3.0 or later (GPL-3.0-or-later). See LICENSE for the full text.

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

pysh_shell-0.2.1.tar.gz (60.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pysh_shell-0.2.1-py3-none-any.whl (59.9 kB view details)

Uploaded Python 3

File details

Details for the file pysh_shell-0.2.1.tar.gz.

File metadata

  • Download URL: pysh_shell-0.2.1.tar.gz
  • Upload date:
  • Size: 60.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pysh_shell-0.2.1.tar.gz
Algorithm Hash digest
SHA256 a63cca7bd3d5ede1fa843c0b06715fcca10c90c74ce56f1187b235a33d565302
MD5 d1f51ad748cc15124060989a07a434b2
BLAKE2b-256 51025cfece2d003f26f31eb0d17ac7745e5b217c8a93c90d75ee8a59d3054e9e

See more details on using hashes here.

Provenance

The following attestation bundles were made for pysh_shell-0.2.1.tar.gz:

Publisher: publish.yml on SSobol77/pysh

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pysh_shell-0.2.1-py3-none-any.whl.

File metadata

  • Download URL: pysh_shell-0.2.1-py3-none-any.whl
  • Upload date:
  • Size: 59.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pysh_shell-0.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 3496199cc4594d052836de08bf91815f88a12ef283cf50d8ffcffb1b65552c9a
MD5 922dd085237c6a54cee0111265b6d45e
BLAKE2b-256 c85d21ab16c8035f25cada347ffbdfea6ce5639938f40259852886e6ef76282a

See more details on using hashes here.

Provenance

The following attestation bundles were made for pysh_shell-0.2.1-py3-none-any.whl:

Publisher: publish.yml on SSobol77/pysh

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page