Metadata-Version: 2.4
Name: colabsh
Version: 1.0.2
Summary: CLI tool for Google Colab - execute code and interact with Colab from the terminal
Project-URL: Documentation, https://github.com/onuralpszr/colabsh#readme
Project-URL: Issues, https://github.com/onuralpszr/colabsh/issues
Project-URL: Source, https://github.com/onuralpszr/colabsh
Author-email: Onuralp Sezer <thunderbirdtr@gmail.com>
License-Expression: Apache-2.0
License-File: LICENSE
Keywords: cli,colab,google,jupyter,notebook
Classifier: Development Status :: 3 - Alpha
Classifier: Environment :: Console
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: Science/Research
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Classifier: Topic :: Scientific/Engineering
Classifier: Topic :: Software Development
Classifier: Typing :: Typed
Requires-Python: >=3.10
Requires-Dist: click>=8.1.0
Requires-Dist: pydantic<3.0.0,>=2.12.3
Requires-Dist: websockets>=15.0.1
Provides-Extra: all
Requires-Dist: playwright>=1.40.0; extra == 'all'
Requires-Dist: qrcode>=8.0; extra == 'all'
Provides-Extra: auto
Requires-Dist: playwright>=1.40.0; extra == 'auto'
Provides-Extra: qr
Requires-Dist: qrcode>=8.0; extra == 'qr'
Description-Content-Type: text/markdown

<div align="center">

<img src="https://raw.githubusercontent.com/onuralpszr/colabsh/main/assets/colabsh-logo.png" alt="colabsh logo" width="480">

</div>

# colabsh

<div align="center">

[![PyPI version](https://img.shields.io/pypi/v/colabsh.svg)](https://pypi.org/project/colabsh/)
[![PyPI downloads](https://img.shields.io/pypi/dm/colabsh.svg)](https://pypistats.org/packages/colabsh)
[![Python versions](https://img.shields.io/pypi/pyversions/colabsh.svg)](https://pypi.org/project/colabsh/)
[![License](https://img.shields.io/badge/license-Apache--2.0-blue.svg)](LICENSE)

[![CI](https://img.shields.io/github/actions/workflow/status/onuralpszr/colabsh/ci.yml?branch=main)](https://github.com/onuralpszr/colabsh/actions)
[![coverage](https://codecov.io/gh/onuralpszr/colabsh/graph/badge.svg)](https://codecov.io/gh/onuralpszr/colabsh)
[![Release](https://img.shields.io/github/v/release/onuralpszr/colabsh)](https://github.com/onuralpszr/colabsh/releases)
[![Dependabot Updates](https://github.com/onuralpszr/colabsh/actions/workflows/dependabot/dependabot-updates/badge.svg)](https://github.com/onuralpszr/colabsh/actions/workflows/dependabot/dependabot-updates)
[![CodeFactor](https://www.codefactor.io/repository/github/onuralpszr/colabsh/badge/main)](https://www.codefactor.io/repository/github/onuralpszr/colabsh/overview/main)

[![pre-commit.ci status](https://results.pre-commit.ci/badge/github/onuralpszr/colabsh/main.svg)](https://results.pre-commit.ci/latest/github/onuralpszr/colabsh/main)
[![autofix enabled](https://shields.io/badge/autofix.ci-yes-success?logo=autofix-ci)](https://autofix.ci)
[![ruff](https://img.shields.io/badge/ruff-enabled-7b0cff?logo=ruff)](https://github.com/charliermarsh/ruff)
[![mypy](https://img.shields.io/badge/mypy-passing-brightgreen?logo=mypy)](https://github.com/onuralpszr/colabsh/actions)

[![stars](https://img.shields.io/github/stars/onuralpszr/colabsh?style=social)](https://github.com/onuralpszr/colabsh/stargazers)
[![forks](https://img.shields.io/github/forks/onuralpszr/colabsh?style=social)](https://github.com/onuralpszr/colabsh/network/members)
[![issues](https://img.shields.io/github/issues/onuralpszr/colabsh)](https://github.com/onuralpszr/colabsh/issues)
[![contributors](https://img.shields.io/github/contributors/onuralpszr/colabsh)](https://github.com/onuralpszr/colabsh/graphs/contributors)

</div>

> A CLI tool for Google Colab. Execute code, download notebooks, and interact
> with Google Colab directly from your terminal. Connects through your browser
> via WebSocket. No API keys needed.

---

## Table of Contents

- [Installation](#installation)
- [Quick Start](#quick-start)
- [How It Works](#how-it-works)
- [Auto Mode (Playwright)](#auto-mode-playwright)
- [Commands](#commands)
- [Headless Mode](#headless-mode)
- [Security](#security)
- [Configuration](#configuration)
- [Docker](#docker)
- [Development](#development)
- [Architecture](#architecture)
- [License](#license)

---

## Installation

| Method               | Command                                                                        |
| -------------------- | ------------------------------------------------------------------------------ |
| **pip**              | `pip install colabsh`                                                          |
| **uvx** (no install) | `uvx colabsh exec "print('hello')"`                                            |
| **From source**      | `git clone https://github.com/onuralpszr/colabsh.git && cd colabsh && uv sync` |

## Quick Start

```bash
# Start the server (opens Google Colab in browser once)
colabsh start

# Execute code (reuses the same browser tab)
colabsh exec "print('hello from Google Colab')"
colabsh exec -f script.py
echo "import sys; print(sys.version)" | colabsh exec -

# Interactive REPL with readline history
colabsh repl

# Download notebook
colabsh download notebook.ipynb
colabsh download script.py -f analysis.py

# Stop the server when done
colabsh stop
```

## How It Works

```
Terminal                     Background Server              Browser
--------                     -----------------              -------
colabsh start ---------------> WebSocket server <----------> Colab
colabsh exec "code" ----TCP--> routes to Colab -----------> executes
colabsh exec "more" ----TCP--> same connection -----------> executes
colabsh stop ----------------> shuts down
```

1. `colabsh start` launches a background server and opens Colab in your browser
2. Colab's frontend connects back via WebSocket (MCP proxy protocol)
3. All subsequent commands go through this persistent connection
4. **One browser tab** serves all commands, no new tabs per command
5. The server auto-starts if you run `exec`/`repl` without starting first

## Auto Mode (Playwright)

For fully headless operation with no manual browser interaction. Playwright
controls a real Chrome browser to connect to Colab automatically.

### Setup

```bash
pip install colabsh[auto]
playwright install chromium

# Login once (opens a visible browser)
colabsh login

# Then use auto mode
colabsh start --auto
```

### GPU selection

```bash
# Select GPU on start
colabsh start --auto --gpu t4

# Change GPU while running
colabsh gpu t4
colabsh gpu a100
colabsh gpu cpu
```

Available GPU types: `cpu`, `t4`, `a100`, `v100`, `l4`, `tpu`

### Health check

```bash
colabsh status --health
```

Shows runtime type (CPU/GPU), whether the runtime is alive, and connection state.

### Debugging

```bash
# Show the browser window
colabsh start --auto --show-browser

# Check server log
cat ~/.config/colabsh/server.log
```

## Commands

### Server

| Command                    | Description                                |
| -------------------------- | ------------------------------------------ |
| `colabsh start`            | Start server and open browser              |
| `colabsh start --auto`     | Fully headless with Playwright             |
| `colabsh start --auto --gpu t4` | Auto-connect with T4 GPU              |
| `colabsh start --headless` | Print URL instead of opening browser       |
| `colabsh start --qr`       | Print QR code + URL for easy copy-paste    |
| `colabsh stop`             | Stop the background server                 |
| `colabsh status`           | Check connection state                     |
| `colabsh status --health`  | Full health check (GPU/CPU, runtime alive) |

### Execute

| Command                         | Description         |
| ------------------------------- | ------------------- |
| `colabsh exec "code"`           | Execute inline code |
| `colabsh exec -f script.py`     | Execute a file      |
| `echo "code" \| colabsh exec -` | Execute from stdin  |
| `colabsh repl`                  | Interactive REPL    |

### REPL Features

| Feature            | Details                                                          |
| ------------------ | ---------------------------------------------------------------- |
| Arrow keys         | Navigate previous commands (readline history)                    |
| Persistent history | Saved across sessions                                            |
| Multiline input    | Lines ending with `:` or `\` start a block (end with empty line) |
| Commands           | `/quit`, `/tools`, `/cells`                                      |

### Download

| Command                                        | Description                               |
| ---------------------------------------------- | ----------------------------------------- |
| `colabsh download notebook.ipynb`              | Download as Jupyter notebook              |
| `colabsh download script.py`                   | Download as Python script                 |
| `colabsh download output.ipynb -f analysis.py` | Execute first, then download with results |

### Other

| Command                            | Description                                |
| ---------------------------------- | ------------------------------------------ |
| `colabsh login`                    | Sign in to Google for auto mode            |
| `colabsh gpu <type>`               | Change GPU on the fly (t4, a100, cpu, ...) |
| `colabsh tools`                    | List available Google Colab frontend tools |
| `colabsh history list`             | Show tracked sessions                      |
| `colabsh history show <id>`        | Show detailed history for a notebook       |
| `colabsh history clear`            | Delete all local history                   |
| `colabsh history toggle [on\|off]` | Enable/disable local history tracking      |
| `colabsh history path`             | Show history file path                     |
| `colabsh --json <command>`         | JSON output for scripting/LLM tools        |
| `colabsh -v <command>`             | Enable debug logging                       |

## Headless Mode

For SSH sessions, containers, or remote machines where there's no desktop
browser:

```bash
colabsh start --headless
```

This prints the connection URL instead of opening a browser. Open the URL on the
same machine in any browser.

### SSH Port Forwarding

To use colabsh on a remote server:

```bash
# On remote server
colabsh start --headless
# Output: https://colab.research.google.com/notebooks/empty.ipynb#mcpProxyToken=...&mcpProxyPort=45123

# On your local machine (forward the port)
ssh -L 45123:localhost:45123 remote-server

# Open the printed URL in your local browser
# It connects to localhost:45123 which is forwarded to the remote server

# Now run commands on the remote server
colabsh exec "print('running on remote')"
```

> **Why not LAN/phone access?** Google Colab's frontend JavaScript **always
> connects WebSocket to `localhost`**. This is hardcoded in Google's code. When
> you open the URL on a different device, the browser tries to connect to
> `localhost` on _that_ device, which doesn't have the colabsh server. The only
> workaround is **SSH port forwarding**, which makes the remote port appear as
> `localhost` on your local machine.

## Security

### What colabsh does

- Runs a **localhost-only** WebSocket server, not accessible from the network
- Uses a random **authentication token** for every session
- Communicates with Google Colab via Google's MCP proxy protocol
- Stores data locally in your [config directory](#config-directory). Nothing is
  sent to third parties

### What to be aware of

- The connection URL contains a secret token. Treat it like a password
- Anyone with the URL can execute code in your Colab session
- The background server runs until you stop it (`colabsh stop`)
- Code execution happens on Google's Colab VMs, subject to Google's terms of
  service
- The Google Colab session has your Google account's permissions

### Token lifecycle

| Event           | Behavior                                                             |
| --------------- | -------------------------------------------------------------------- |
| `colabsh start` | A new random token is generated                                      |
| While running   | Token stored in `~/.config/colabsh/server.json` (user-readable only) |
| `colabsh stop`  | Token is deleted                                                     |

## Configuration

### Config directory

| Platform | Path                                       |
| -------- | ------------------------------------------ |
| Linux    | `~/.config/colabsh/`                       |
| macOS    | `~/Library/Application Support/colabsh/`   |
| Windows  | `C:\Users\<user>\AppData\Roaming\colabsh\` |

### Config files

| File            | Description                             |
| --------------- | --------------------------------------- |
| `server.json`   | Running server state (port, PID, token) |
| `server.log`    | Server logs                             |
| `settings.json` | User preferences (headless mode, etc.)  |
| `history.json`  | Local usage history                     |
| `repl_history`  | Readline command history                |

### Output format

Human-readable output is the default:

```bash
colabsh status
# status: running
# connected: true
# pid: 12345

colabsh --json status
# {"status": "running", "connected": true, "pid": 12345}
```

Use `--json` when piping to other tools or LLMs.

## Docker

```bash
# Build
docker build -t colabsh .

# Login (interactive, mount profile volume)
docker run -it -v colabsh-data:/root/.config/colabsh colabsh login

# Run with auto-connect
docker run -d -v colabsh-data:/root/.config/colabsh colabsh start --auto

# Execute code
docker exec <container> colabsh exec "print('hello')"
```

## Development

| Task                 | Command                          |
| -------------------- | -------------------------------- |
| Install dependencies | `uv sync`                        |
| Run tests            | `uv run pytest`                  |
| Lint                 | `uv run ruff check src/ tests/`  |
| Format               | `uv run ruff format src/ tests/` |
| Type check           | `uv run mypy src/colabsh/`       |

## Architecture

```
src/colabsh/
|-- main.py              # Click CLI entry point
|-- commands.py          # All commands (exec, repl, start, stop, gpu, login, ...)
|-- history.py           # history list/show/clear/toggle
+-- core/
    |-- models.py        # Pydantic models (Settings, ServerState, ConnectionInfo)
    |-- config.py        # Platform config dirs, settings persistence
    |-- server.py        # Background server (WebSocket + TCP control)
    |-- proxy.py         # WebSocket + JSON-RPC to Colab frontend
    |-- browser.py       # Playwright automation (auto-connect, GPU, login)
    |-- cells.py         # Shared helpers for notebook cell parsing
    |-- output.py        # JSON/human formatter
    |-- history.py       # Local usage tracking
    |-- repl.py          # Shared REPL with readline
    +-- qr.py            # QR code generation (optional)
```

## License

Apache-2.0

## Inspiration

Inspired by [colab-mcp](https://github.com/googlecolab/colab-mcp) but with a
focus on CLI usability, persistent server, and local history.

## Disclaimer

This project has no affiliation with Google. It reverse-engineers Google Colab's
frontend protocol to enable terminal access. Use responsibly and in accordance
with Google's terms of service.
