Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 84b5ed4412 | |||
| 8535f73ad3 | |||
| e4063b5559 | |||
| cb4971b4a6 | |||
| 3d632ef02f |
@@ -227,6 +227,7 @@ Understanding what survives container restarts and what doesn't:
|
|||||||
| `/home/developer/.aws` | Host bind mount | ✅ Yes — lives on host | AWS credentials/SSO cache |
|
| `/home/developer/.aws` | Host bind mount | ✅ Yes — lives on host | AWS credentials/SSO cache |
|
||||||
| `/home/developer/.local/share/opencode` | Named volume (if configured) | ✅ Yes — Docker volume | Session history, memory, auth tokens |
|
| `/home/developer/.local/share/opencode` | Named volume (if configured) | ✅ Yes — Docker volume | Session history, memory, auth tokens |
|
||||||
| `/home/developer/.local/state/opencode` | Named volume (if configured) | ✅ Yes — Docker volume | TUI settings (theme, toggles) |
|
| `/home/developer/.local/state/opencode` | Named volume (if configured) | ✅ Yes — Docker volume | TUI settings (theme, toggles) |
|
||||||
|
| `/home/developer/.cache/bash` | Named volume `devbox-shell-history` | ✅ Yes — Docker volume | Bash history (`$HISTFILE`) — survives container recreate |
|
||||||
| `/home/developer/.local/share/uv` | Named volume (if configured) | ✅ Yes — Docker volume | Python installs, uv tool installs |
|
| `/home/developer/.local/share/uv` | Named volume (if configured) | ✅ Yes — Docker volume | Python installs, uv tool installs |
|
||||||
| `/home/developer/.rustup` | Named volume (if configured) | ✅ Yes — Docker volume | Rust toolchains |
|
| `/home/developer/.rustup` | Named volume (if configured) | ✅ Yes — Docker volume | Rust toolchains |
|
||||||
| `/home/developer/.cargo` | Named volume (if configured) | ✅ Yes — Docker volume | Cargo binaries, registry cache |
|
| `/home/developer/.cargo` | Named volume (if configured) | ✅ Yes — Docker volume | Cargo binaries, registry cache |
|
||||||
@@ -239,6 +240,7 @@ Understanding what survives container restarts and what doesn't:
|
|||||||
- **opencode config** is auto-generated from `OPENCODE_PROVIDER` env var on each start if no existing config is found. To persist config changes, mount the config directory from the host (see Custom opencode Config below).
|
- **opencode config** is auto-generated from `OPENCODE_PROVIDER` env var on each start if no existing config is found. To persist config changes, mount the config directory from the host (see Custom opencode Config below).
|
||||||
- **opencode data** (session history, memory) is lost on container recreation unless you add a named volume.
|
- **opencode data** (session history, memory) is lost on container recreation unless you add a named volume.
|
||||||
- **TUI settings** (theme, toggles) are lost on container recreation unless you add the `devbox-state` named volume.
|
- **TUI settings** (theme, toggles) are lost on container recreation unless you add the `devbox-state` named volume.
|
||||||
|
- **Bash history** persists via the `devbox-shell-history` volume mounted at `~/.cache/bash`. `HISTFILE` is pre-configured; no setup required.
|
||||||
- **Python installs** via `uv python install` are lost on container recreation unless you add the `devbox-uv` named volume.
|
- **Python installs** via `uv python install` are lost on container recreation unless you add the `devbox-uv` named volume.
|
||||||
- **Rust toolchains** via `rustup-init` are lost on container recreation unless you add the `devbox-rustup` and `devbox-cargo` named volumes.
|
- **Rust toolchains** via `rustup-init` are lost on container recreation unless you add the `devbox-rustup` and `devbox-cargo` named volumes.
|
||||||
- **AWS SSO tokens** persist across restarts when `~/.aws` is mounted (recommended for Bedrock users).
|
- **AWS SSO tokens** persist across restarts when `~/.aws` is mounted (recommended for Bedrock users).
|
||||||
@@ -463,6 +465,19 @@ docker compose run --rm devbox # direct to opencode
|
|||||||
docker compose run --rm devbox bash # interactive shell
|
docker compose run --rm devbox bash # interactive shell
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Shell defaults
|
||||||
|
|
||||||
|
The image ships baked `.bash_aliases` and `.inputrc` in `/etc/skel-devbox/`. On first container start the entrypoint copies them to `/home/developer/` **only if the target file does not already exist**, so your host bind-mounts or any in-container customization are preserved across upgrades.
|
||||||
|
|
||||||
|
- **Prefix history search** on Up/Down arrows (type `git `, press Up, walk back through prior `git ...` commands only). Ctrl-Up / Ctrl-Down still step through full history.
|
||||||
|
- **Persistent history** via `$HISTFILE=~/.cache/bash/history`, backed by the `devbox-shell-history` named volume. Survives container recreate. 100 000 entries, time-stamped, dedup.
|
||||||
|
- **Case-insensitive tab completion** and coloured completion lists.
|
||||||
|
- **Aliases** — `ls`/`ll`/`la` → `eza`, `cat` → `bat`, `gs`/`gd`/`gl` for git, interactive `rm`/`mv`/`cp`.
|
||||||
|
- **Integrations** — `zoxide` (`z <fragment>`), `fzf` key bindings (`Ctrl-R`, `Ctrl-T`).
|
||||||
|
- **`[devbox]` prompt prefix** so you always know you're in the container.
|
||||||
|
|
||||||
|
To override with your host's own files, uncomment the matching bind-mount lines in `docker-compose.yml`. To restore the baked defaults any time: `cp /etc/skel-devbox/.bash_aliases ~/` (or delete the file and recreate the container).
|
||||||
|
|
||||||
## What's Included
|
## What's Included
|
||||||
|
|
||||||
### Base image (`latest`)
|
### Base image (`latest`)
|
||||||
|
|||||||
+14
@@ -200,9 +200,23 @@ RUN mkdir -p /workspace \
|
|||||||
/home/${USER_NAME}/.config/opencode/skills \
|
/home/${USER_NAME}/.config/opencode/skills \
|
||||||
/home/${USER_NAME}/.agents/skills \
|
/home/${USER_NAME}/.agents/skills \
|
||||||
/home/${USER_NAME}/.local/share/opencode \
|
/home/${USER_NAME}/.local/share/opencode \
|
||||||
|
/home/${USER_NAME}/.cache/bash \
|
||||||
/home/${USER_NAME}/.ssh && \
|
/home/${USER_NAME}/.ssh && \
|
||||||
chown -R ${USER_NAME}:${USER_NAME} /workspace /home/${USER_NAME}
|
chown -R ${USER_NAME}:${USER_NAME} /workspace /home/${USER_NAME}
|
||||||
|
|
||||||
|
# ── Shell defaults (bash history, aliases, readline) ─────────────────
|
||||||
|
# Shipped under /etc/skel-devbox/ rather than copied directly to the
|
||||||
|
# user's home. The entrypoint copies them to /home/developer/ only if
|
||||||
|
# the target file does not already exist, so host bind-mounts and
|
||||||
|
# previously-customized files are never overwritten. Users can restore
|
||||||
|
# the baked defaults anytime via:
|
||||||
|
# cp /etc/skel-devbox/.bash_aliases ~/.bash_aliases
|
||||||
|
# History itself persists via the devbox-shell-history named volume
|
||||||
|
# mounted at ~/.cache/bash (HISTFILE points there).
|
||||||
|
RUN mkdir -p /etc/skel-devbox
|
||||||
|
COPY rootfs/home/developer/.bash_aliases /etc/skel-devbox/.bash_aliases
|
||||||
|
COPY rootfs/home/developer/.inputrc /etc/skel-devbox/.inputrc
|
||||||
|
|
||||||
# ── Entrypoint ────────────────────────────────────────────────────────
|
# ── Entrypoint ────────────────────────────────────────────────────────
|
||||||
COPY entrypoint.sh /usr/local/bin/entrypoint.sh
|
COPY entrypoint.sh /usr/local/bin/entrypoint.sh
|
||||||
COPY entrypoint-user.sh /usr/local/bin/entrypoint-user.sh
|
COPY entrypoint-user.sh /usr/local/bin/entrypoint-user.sh
|
||||||
|
|||||||
@@ -438,6 +438,49 @@ The `--use-device-code` flag outputs a URL and short code instead of trying to o
|
|||||||
|
|
||||||
SSO sessions typically last 8–12 hours before requiring re-authentication. Since `~/.aws` is mounted from the host, tokens persist across container restarts.
|
SSO sessions typically last 8–12 hours before requiring re-authentication. Since `~/.aws` is mounted from the host, tokens persist across container restarts.
|
||||||
|
|
||||||
|
## Shell defaults
|
||||||
|
|
||||||
|
The image ships a baked `.bash_aliases` and `.inputrc` with quality-of-life defaults. On first container start they are copied from `/etc/skel-devbox/` into `/home/developer/` **only if the target file does not already exist** — so host bind-mounts and any version you've customized inside the container are never overwritten on upgrade.
|
||||||
|
|
||||||
|
Defaults you get out of the box:
|
||||||
|
|
||||||
|
- **Prefix history search** on Up/Down arrows (type `git `, press Up, walk back through prior `git ...` commands only). Ctrl-Up / Ctrl-Down still step through full history.
|
||||||
|
- **Persistent history** — `$HISTFILE` points at `~/.cache/bash/history`, backed by the `devbox-shell-history` named volume so history survives container recreation. Timestamps, 100 000 entries, dedup.
|
||||||
|
- **Case-insensitive tab completion**, coloured completion lists, `show-all-if-ambiguous`.
|
||||||
|
- **Aliases** — `ls`/`ll`/`la` use `eza`, `cat` uses `bat`, `gs`/`gd`/`gl` for git, safe `rm`/`mv`/`cp`.
|
||||||
|
- **Integrations** — `zoxide` (`z <fragment>` to jump), `fzf` Ctrl-R / Ctrl-T key bindings.
|
||||||
|
- **Prompt marker** — `[devbox]` prefix so it's always obvious you're inside the container.
|
||||||
|
|
||||||
|
### Overriding the defaults
|
||||||
|
|
||||||
|
**Option A — bind-mount host files.** Uncomment the bind-mount lines in `docker-compose.yml`:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
- ~/.bash_aliases:/home/developer/.bash_aliases:ro
|
||||||
|
- ~/.inputrc:/home/developer/.inputrc:ro
|
||||||
|
```
|
||||||
|
|
||||||
|
**Option B — customize inside the container.** Just edit `~/.bash_aliases` or `~/.inputrc` as normal. Pair this with a bind-mount or named volume on the home dir if you want the edits to survive container recreation.
|
||||||
|
|
||||||
|
### Restoring or diffing defaults
|
||||||
|
|
||||||
|
The skel files remain available inside every container at `/etc/skel-devbox/`. Useful commands:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# See what the image currently ships
|
||||||
|
cat /etc/skel-devbox/.bash_aliases
|
||||||
|
|
||||||
|
# Diff your current config against the upstream defaults
|
||||||
|
diff ~/.bash_aliases /etc/skel-devbox/.bash_aliases
|
||||||
|
|
||||||
|
# Reset to the baked defaults
|
||||||
|
cp /etc/skel-devbox/.bash_aliases ~/.bash_aliases
|
||||||
|
|
||||||
|
# …or delete the file and recreate the container — the entrypoint
|
||||||
|
# copies from /etc/skel-devbox/ on next start if the target is absent
|
||||||
|
rm ~/.bash_aliases
|
||||||
|
```
|
||||||
|
|
||||||
## Secret Scanning
|
## Secret Scanning
|
||||||
|
|
||||||
A [gitleaks](https://github.com/gitleaks/gitleaks) pre-commit hook prevents accidentally committing API keys, passwords, or other secrets.
|
A [gitleaks](https://github.com/gitleaks/gitleaks) pre-commit hook prevents accidentally committing API keys, passwords, or other secrets.
|
||||||
@@ -496,6 +539,7 @@ Container (Debian trixie)
|
|||||||
| `/home/developer/.aws` | Host bind mount (if configured) | ✅ Yes | AWS credentials/SSO cache |
|
| `/home/developer/.aws` | Host bind mount (if configured) | ✅ Yes | AWS credentials/SSO cache |
|
||||||
| `/home/developer/.local/share/opencode` | Named volume `devbox-data` | ✅ Yes | Session history, memory |
|
| `/home/developer/.local/share/opencode` | Named volume `devbox-data` | ✅ Yes | Session history, memory |
|
||||||
| `/home/developer/.local/state/opencode` | Named volume `devbox-state` | ✅ Yes | TUI settings (theme, toggles) |
|
| `/home/developer/.local/state/opencode` | Named volume `devbox-state` | ✅ Yes | TUI settings (theme, toggles) |
|
||||||
|
| `/home/developer/.cache/bash` | Named volume `devbox-shell-history` | ✅ Yes | Bash history (`$HISTFILE`), survives container recreate |
|
||||||
| `/home/developer/.local/share/uv` | Named volume `devbox-uv` (if configured) | ✅ Yes | Python installs, uv tool installs |
|
| `/home/developer/.local/share/uv` | Named volume `devbox-uv` (if configured) | ✅ Yes | Python installs, uv tool installs |
|
||||||
| `/home/developer/.rustup` | Named volume `devbox-rustup` (if configured) | ✅ Yes | Rust toolchains |
|
| `/home/developer/.rustup` | Named volume `devbox-rustup` (if configured) | ✅ Yes | Rust toolchains |
|
||||||
| `/home/developer/.cargo` | Named volume `devbox-cargo` (if configured) | ✅ Yes | Cargo binaries, registry cache |
|
| `/home/developer/.cargo` | Named volume `devbox-cargo` (if configured) | ✅ Yes | Cargo binaries, registry cache |
|
||||||
|
|||||||
@@ -197,3 +197,58 @@ After editing `docker-compose.yml` on the VM to uncomment the bind mounts you ne
|
|||||||
```
|
```
|
||||||
|
|
||||||
The script reads `docker-compose.yml` on the remote VM, detects which bind mounts are active, and syncs only those directories from your local machine. It also creates the remote directories if they don't exist.
|
The script reads `docker-compose.yml` on the remote VM, detects which bind mounts are active, and syncs only those directories from your local machine. It also creates the remote directories if they don't exist.
|
||||||
|
|
||||||
|
### Shell defaults inside the container
|
||||||
|
|
||||||
|
The image ships baked `.bash_aliases` and `.inputrc` in `/etc/skel-devbox/` — quality-of-life defaults (prefix history search on Up/Down arrows, persistent history across container recreates via the `devbox-shell-history` named volume, `[devbox]` prompt marker, sensible aliases). On first container start the entrypoint copies them to `/home/developer/` **only if the target file does not already exist**.
|
||||||
|
|
||||||
|
This means:
|
||||||
|
|
||||||
|
- Fresh containers get the defaults automatically.
|
||||||
|
- If you bind-mount your host's `~/.bash_aliases` / `~/.inputrc` (see the commented lines in `docker-compose.yml`), your host versions win.
|
||||||
|
- If you edit the files inside a running container and store them via a home-dir bind-mount or equivalent, subsequent upgrades never overwrite them.
|
||||||
|
- To restore the baked defaults any time: `cp /etc/skel-devbox/.bash_aliases ~/` (or delete the file and recreate the container).
|
||||||
|
- To diff your current config against what the image ships: `diff ~/.bash_aliases /etc/skel-devbox/.bash_aliases`.
|
||||||
|
|
||||||
|
### Troubleshooting: SSH hangs or "banner exchange" timeouts
|
||||||
|
|
||||||
|
If SSH to the VM intermittently fails with `Connection timed out during banner exchange` or pure TCP connect timeouts — especially after the first few successful connects in a short window — the cause is almost certainly your ISP's CGNAT (Carrier-Grade NAT), not the VM.
|
||||||
|
|
||||||
|
**Symptoms**
|
||||||
|
|
||||||
|
- First 3–4 SSH connects succeed, then subsequent ones fail hard for 20–30 minutes
|
||||||
|
- `ping` to the VM works perfectly throughout (ICMP isn't tracked the same way)
|
||||||
|
- `mosh` sessions stay stable once established (UDP, different flow table)
|
||||||
|
- Happens on residential ISPs (Tele2, Comhem, Telia, most European consumer broadband)
|
||||||
|
- VM-side logs show SSH is idle — the SYNs never reach it
|
||||||
|
|
||||||
|
**Cause**
|
||||||
|
|
||||||
|
Residential CGNAT boxes keep a per-subscriber TCP flow table with a small concurrent-flow cap (~4) per destination IP. Once exhausted, new SYNs to that destination are silently dropped until old flows age out (typically 20–30 min after TCP close).
|
||||||
|
|
||||||
|
**Fix**
|
||||||
|
|
||||||
|
Add SSH connection multiplexing on your client so all SSH sessions (interactive, `scp`, `rsync`, scripts) share a single TCP connection to the VM:
|
||||||
|
|
||||||
|
```ssh-config
|
||||||
|
# ~/.ssh/config
|
||||||
|
Host <vm-alias>
|
||||||
|
HostName <vm-ip>
|
||||||
|
User devbox
|
||||||
|
IdentityFile ~/.ssh/id_ed25519
|
||||||
|
ControlMaster auto
|
||||||
|
ControlPath ~/.ssh/cm/%r@%h:%p
|
||||||
|
ControlPersist 4h
|
||||||
|
ServerAliveInterval 30
|
||||||
|
ServerAliveCountMax 6
|
||||||
|
```
|
||||||
|
|
||||||
|
Then create the socket directory:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mkdir -p ~/.ssh/cm && chmod 700 ~/.ssh/cm
|
||||||
|
```
|
||||||
|
|
||||||
|
All SSH to the VM now multiplexes over a single flow slot, regardless of how many parallel sessions you open. `sync-to-vm.sh` already does this internally for its own rsync/scp calls.
|
||||||
|
|
||||||
|
For a more robust long-term fix (especially if you access the VM from multiple hosts), run a WireGuard tunnel on the VM and route SSH through that — UDP bypasses the TCP flow table entirely.
|
||||||
|
|||||||
@@ -39,6 +39,9 @@ services:
|
|||||||
# Persist opencode data (auth, memory, session history)
|
# Persist opencode data (auth, memory, session history)
|
||||||
- devbox-data:/home/developer/.local/share/opencode
|
- devbox-data:/home/developer/.local/share/opencode
|
||||||
|
|
||||||
|
# Persist bash history across container recreations
|
||||||
|
- devbox-shell-history:/home/developer/.cache/bash
|
||||||
|
|
||||||
# Persist uv data (Python installs)
|
# Persist uv data (Python installs)
|
||||||
- devbox-uv:/home/developer/.local/share/uv
|
- devbox-uv:/home/developer/.local/share/uv
|
||||||
|
|
||||||
@@ -47,4 +50,5 @@ services:
|
|||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
devbox-data:
|
devbox-data:
|
||||||
|
devbox-shell-history:
|
||||||
devbox-uv:
|
devbox-uv:
|
||||||
|
|||||||
@@ -52,6 +52,16 @@ services:
|
|||||||
# Optional: persist opencode TUI settings (theme, toggles, etc.)
|
# Optional: persist opencode TUI settings (theme, toggles, etc.)
|
||||||
- devbox-state:/home/developer/.local/state/opencode
|
- devbox-state:/home/developer/.local/state/opencode
|
||||||
|
|
||||||
|
# Persist bash history across container recreations.
|
||||||
|
# Without this, ~/.bash_history is lost on 'docker compose up --force-recreate'.
|
||||||
|
- devbox-shell-history:/home/developer/.cache/bash
|
||||||
|
|
||||||
|
# Optional: override baked shell defaults with your host's rc files.
|
||||||
|
# The image ships sensible defaults (history tuning, prefix-search on
|
||||||
|
# Up/Down arrows, fzf/zoxide integration). Uncomment to use your own:
|
||||||
|
# - ~/.bash_aliases:/home/developer/.bash_aliases:ro
|
||||||
|
# - ~/.inputrc:/home/developer/.inputrc:ro
|
||||||
|
|
||||||
# Optional: persist uv data (Python installs, tool installs)
|
# Optional: persist uv data (Python installs, tool installs)
|
||||||
# Without this, 'uv python install' must be re-run after container removal.
|
# Without this, 'uv python install' must be re-run after container removal.
|
||||||
- devbox-uv:/home/developer/.local/share/uv
|
- devbox-uv:/home/developer/.local/share/uv
|
||||||
@@ -70,6 +80,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
devbox-data:
|
devbox-data:
|
||||||
devbox-state:
|
devbox-state:
|
||||||
|
devbox-shell-history:
|
||||||
devbox-uv:
|
devbox-uv:
|
||||||
# devbox-rustup:
|
# devbox-rustup:
|
||||||
# devbox-cargo:
|
# devbox-cargo:
|
||||||
|
|||||||
@@ -1,6 +1,20 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
|
# ── Shell defaults: copy baked files from /etc/skel-devbox/ if absent
|
||||||
|
# Respects host bind-mounts and user customizations — existing files
|
||||||
|
# are never overwritten. To restore defaults: rm ~/.bash_aliases (or
|
||||||
|
# .inputrc) and recreate the container, or cp from /etc/skel-devbox/
|
||||||
|
# directly.
|
||||||
|
SKEL_DIR="/etc/skel-devbox"
|
||||||
|
if [ -d "$SKEL_DIR" ]; then
|
||||||
|
for f in .bash_aliases .inputrc; do
|
||||||
|
if [ -f "$SKEL_DIR/$f" ] && [ ! -e "$HOME/$f" ]; then
|
||||||
|
cp "$SKEL_DIR/$f" "$HOME/$f"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
# ── Git config defaults ──────────────────────────────────────────────
|
# ── Git config defaults ──────────────────────────────────────────────
|
||||||
if [ -n "${GIT_USER_NAME:-}" ] && ! git config --global user.name &>/dev/null; then
|
if [ -n "${GIT_USER_NAME:-}" ] && ! git config --global user.name &>/dev/null; then
|
||||||
git config --global user.name "$GIT_USER_NAME"
|
git config --global user.name "$GIT_USER_NAME"
|
||||||
|
|||||||
+16
-9
@@ -6,18 +6,22 @@ CURRENT_UID=$(id -u "$USER_NAME")
|
|||||||
CURRENT_GID=$(id -g "$USER_NAME")
|
CURRENT_GID=$(id -g "$USER_NAME")
|
||||||
|
|
||||||
# ── UID/GID adjustment ───────────────────────────────────────────────
|
# ── UID/GID adjustment ───────────────────────────────────────────────
|
||||||
# Priority: env vars > auto-detect from /workspace > default (1000)
|
# Priority per dimension: env var > auto-detect from /workspace > no-op
|
||||||
|
# UID and GID are detected independently so a GID-only mismatch (e.g. host
|
||||||
|
# user has UID 1000 but primary group at GID 1001) is still corrected.
|
||||||
TARGET_UID="${USER_UID:-}"
|
TARGET_UID="${USER_UID:-}"
|
||||||
TARGET_GID="${USER_GID:-}"
|
TARGET_GID="${USER_GID:-}"
|
||||||
|
|
||||||
# Auto-detect from /workspace owner if env vars not set
|
if [ -d /workspace ]; then
|
||||||
if [ -z "$TARGET_UID" ] && [ -d /workspace ]; then
|
WORKSPACE_UID=$(stat -c '%u' /workspace 2>/dev/null || stat -f '%u' /workspace 2>/dev/null || echo "")
|
||||||
WORKSPACE_UID=$(stat -c '%u' /workspace 2>/dev/null || stat -f '%u' /workspace 2>/dev/null)
|
WORKSPACE_GID=$(stat -c '%g' /workspace 2>/dev/null || stat -f '%g' /workspace 2>/dev/null || echo "")
|
||||||
WORKSPACE_GID=$(stat -c '%g' /workspace 2>/dev/null || stat -f '%g' /workspace 2>/dev/null)
|
# Adopt workspace UID if env var not set and workspace is non-root-owned
|
||||||
# Only adjust if workspace is owned by a non-root user
|
if [ -z "$TARGET_UID" ] && [ -n "$WORKSPACE_UID" ] && [ "$WORKSPACE_UID" != "0" ] && [ "$WORKSPACE_UID" != "$CURRENT_UID" ]; then
|
||||||
if [ "$WORKSPACE_UID" != "0" ] && [ "$WORKSPACE_UID" != "$CURRENT_UID" ]; then
|
|
||||||
TARGET_UID="$WORKSPACE_UID"
|
TARGET_UID="$WORKSPACE_UID"
|
||||||
TARGET_GID="${TARGET_GID:-$WORKSPACE_GID}"
|
fi
|
||||||
|
# Adopt workspace GID if env var not set and workspace group differs
|
||||||
|
if [ -z "$TARGET_GID" ] && [ -n "$WORKSPACE_GID" ] && [ "$WORKSPACE_GID" != "0" ] && [ "$WORKSPACE_GID" != "$CURRENT_GID" ]; then
|
||||||
|
TARGET_GID="$WORKSPACE_GID"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -25,12 +29,13 @@ fi
|
|||||||
if [ -n "$TARGET_GID" ] && [ "$TARGET_GID" != "$CURRENT_GID" ]; then
|
if [ -n "$TARGET_GID" ] && [ "$TARGET_GID" != "$CURRENT_GID" ]; then
|
||||||
groupmod -g "$TARGET_GID" "$USER_NAME" 2>/dev/null || true
|
groupmod -g "$TARGET_GID" "$USER_NAME" 2>/dev/null || true
|
||||||
find /home/"$USER_NAME" -not -path "/home/$USER_NAME/.ssh/*" -group "$CURRENT_GID" -exec chgrp "$TARGET_GID" {} + 2>/dev/null || true
|
find /home/"$USER_NAME" -not -path "/home/$USER_NAME/.ssh/*" -group "$CURRENT_GID" -exec chgrp "$TARGET_GID" {} + 2>/dev/null || true
|
||||||
|
echo "Adjusted developer GID to $TARGET_GID"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ -n "$TARGET_UID" ] && [ "$TARGET_UID" != "$CURRENT_UID" ]; then
|
if [ -n "$TARGET_UID" ] && [ "$TARGET_UID" != "$CURRENT_UID" ]; then
|
||||||
usermod -u "$TARGET_UID" "$USER_NAME" 2>/dev/null || true
|
usermod -u "$TARGET_UID" "$USER_NAME" 2>/dev/null || true
|
||||||
find /home/"$USER_NAME" -not -path "/home/$USER_NAME/.ssh/*" -user "$CURRENT_UID" -exec chown "$TARGET_UID" {} + 2>/dev/null || true
|
find /home/"$USER_NAME" -not -path "/home/$USER_NAME/.ssh/*" -user "$CURRENT_UID" -exec chown "$TARGET_UID" {} + 2>/dev/null || true
|
||||||
echo "Adjusted developer UID:GID to $TARGET_UID:${TARGET_GID:-$CURRENT_GID}"
|
echo "Adjusted developer UID to $TARGET_UID"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# ── SSH key permissions ──────────────────────────────────────────────
|
# ── SSH key permissions ──────────────────────────────────────────────
|
||||||
@@ -61,6 +66,7 @@ for parent in \
|
|||||||
/home/"$USER_NAME"/.local \
|
/home/"$USER_NAME"/.local \
|
||||||
/home/"$USER_NAME"/.local/share \
|
/home/"$USER_NAME"/.local/share \
|
||||||
/home/"$USER_NAME"/.local/state \
|
/home/"$USER_NAME"/.local/state \
|
||||||
|
/home/"$USER_NAME"/.cache \
|
||||||
/home/"$USER_NAME"/.config; do
|
/home/"$USER_NAME"/.config; do
|
||||||
if [ -d "$parent" ] && [ "$(stat -c '%u' "$parent" 2>/dev/null)" != "$FINAL_UID" ]; then
|
if [ -d "$parent" ] && [ "$(stat -c '%u' "$parent" 2>/dev/null)" != "$FINAL_UID" ]; then
|
||||||
chown "$FINAL_UID":"$FINAL_GID" "$parent" 2>/dev/null || true
|
chown "$FINAL_UID":"$FINAL_GID" "$parent" 2>/dev/null || true
|
||||||
@@ -71,6 +77,7 @@ for dir in \
|
|||||||
/home/"$USER_NAME"/.local/share/opencode \
|
/home/"$USER_NAME"/.local/share/opencode \
|
||||||
/home/"$USER_NAME"/.local/state/opencode \
|
/home/"$USER_NAME"/.local/state/opencode \
|
||||||
/home/"$USER_NAME"/.local/share/uv \
|
/home/"$USER_NAME"/.local/share/uv \
|
||||||
|
/home/"$USER_NAME"/.cache/bash \
|
||||||
/home/"$USER_NAME"/.rustup \
|
/home/"$USER_NAME"/.rustup \
|
||||||
/home/"$USER_NAME"/.cargo \
|
/home/"$USER_NAME"/.cargo \
|
||||||
/home/"$USER_NAME"/.vscode-server \
|
/home/"$USER_NAME"/.vscode-server \
|
||||||
|
|||||||
@@ -0,0 +1,82 @@
|
|||||||
|
# opencode-devbox bash aliases and customizations
|
||||||
|
# Sourced by the Debian-default ~/.bashrc on shell startup.
|
||||||
|
# To override, bind-mount your host's ~/.bash_aliases over this file
|
||||||
|
# via docker-compose.yml.
|
||||||
|
|
||||||
|
# ── History persistence and quality ──────────────────────────────────
|
||||||
|
# The named volume devbox-shell-history is mounted at ~/.cache/bash
|
||||||
|
# so history survives container recreation.
|
||||||
|
export HISTFILE="${HOME}/.cache/bash/history"
|
||||||
|
mkdir -p "$(dirname "$HISTFILE")" 2>/dev/null || true
|
||||||
|
|
||||||
|
# Large, time-stamped, deduplicated history. Append rather than overwrite.
|
||||||
|
export HISTSIZE=100000
|
||||||
|
export HISTFILESIZE=200000
|
||||||
|
export HISTCONTROL=ignoreboth:erasedups
|
||||||
|
export HISTTIMEFORMAT='%F %T '
|
||||||
|
shopt -s histappend 2>/dev/null
|
||||||
|
shopt -s cmdhist 2>/dev/null
|
||||||
|
# Note: PROMPT_COMMAND="history -a" is installed LATER in this file,
|
||||||
|
# after zoxide's init runs. Installing it here would create a
|
||||||
|
# "history -a;;__zoxide_hook" chain because zoxide's init uses ';'
|
||||||
|
# as its separator and prepends itself; two adjacent ';' breaks the
|
||||||
|
# parser. See https://github.com/ajeetdsouza/zoxide/issues/722.
|
||||||
|
|
||||||
|
# ── Common aliases ───────────────────────────────────────────────────
|
||||||
|
# Prefer eza (modern ls) when available
|
||||||
|
if command -v eza >/dev/null 2>&1; then
|
||||||
|
alias ls='eza --group-directories-first'
|
||||||
|
alias ll='eza -lh --group-directories-first --git'
|
||||||
|
alias la='eza -lha --group-directories-first --git'
|
||||||
|
alias tree='eza --tree'
|
||||||
|
else
|
||||||
|
alias ll='ls -lh'
|
||||||
|
alias la='ls -lha'
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Prefer bat (syntax-highlighted cat) when available
|
||||||
|
if command -v bat >/dev/null 2>&1; then
|
||||||
|
alias cat='bat --style=plain --paging=never'
|
||||||
|
alias less='bat --paging=always'
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Git shortcuts
|
||||||
|
alias gs='git status'
|
||||||
|
alias gd='git diff'
|
||||||
|
alias gl='git log --oneline --graph --decorate -20'
|
||||||
|
|
||||||
|
# Safety: confirm before destructive ops
|
||||||
|
alias rm='rm -i'
|
||||||
|
alias mv='mv -i'
|
||||||
|
alias cp='cp -i'
|
||||||
|
|
||||||
|
# ── Shell integrations ───────────────────────────────────────────────
|
||||||
|
# zoxide — smarter cd. Use 'z <fragment>' to jump to previously-visited dirs.
|
||||||
|
if command -v zoxide >/dev/null 2>&1; then
|
||||||
|
eval "$(zoxide init bash)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# fzf — fuzzy finder key bindings (Ctrl-R for history, Ctrl-T for files).
|
||||||
|
# We install fzf from GitHub releases (not apt), so sourcing from the
|
||||||
|
# apt-path /usr/share/doc/fzf/examples/* would find nothing. Use the
|
||||||
|
# binary's own --bash flag (available since fzf 0.48) for setup.
|
||||||
|
if command -v fzf >/dev/null 2>&1; then
|
||||||
|
eval "$(fzf --bash)" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
# ── PROMPT_COMMAND: flush history every prompt ───────────────────────
|
||||||
|
# Installed AFTER zoxide init so zoxide's hook is already in place;
|
||||||
|
# we append with a newline separator to avoid the ';;' parse error
|
||||||
|
# described at the top of this file. Guarded so repeated sourcing
|
||||||
|
# (e.g. `exec bash`) doesn't stack duplicates.
|
||||||
|
if [ -z "${DEVBOX_HIST_SET:-}" ]; then
|
||||||
|
PROMPT_COMMAND="${PROMPT_COMMAND:+$PROMPT_COMMAND$'\n'}history -a"
|
||||||
|
export DEVBOX_HIST_SET=1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# ── Prompt: show [opencode-devbox] tag so it's obvious you're in the container
|
||||||
|
# Preserves the default Debian PS1 logic but prefixes with a container marker.
|
||||||
|
if [ -n "${PS1:-}" ] && [ -z "${DEVBOX_PS1_SET:-}" ]; then
|
||||||
|
PS1='\[\e[38;5;39m\][devbox]\[\e[0m\] '"${PS1}"
|
||||||
|
export DEVBOX_PS1_SET=1
|
||||||
|
fi
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
# opencode-devbox readline defaults
|
||||||
|
# To override, bind-mount your host's ~/.inputrc over this file
|
||||||
|
# via docker-compose.yml.
|
||||||
|
|
||||||
|
# Inherit system-wide defaults (colour, 8-bit input, …) if present
|
||||||
|
$include /etc/inputrc
|
||||||
|
|
||||||
|
# ── History search on Up/Down ────────────────────────────────────────
|
||||||
|
# Type a prefix, press Up, and walk through previous commands starting
|
||||||
|
# with that prefix. Ctrl-Up / Ctrl-Down keep the unconditional stepper.
|
||||||
|
"\e[A": history-search-backward
|
||||||
|
"\e[B": history-search-forward
|
||||||
|
"\e[1;5A": previous-history
|
||||||
|
"\e[1;5B": next-history
|
||||||
|
|
||||||
|
# ── Completion quality ───────────────────────────────────────────────
|
||||||
|
set show-all-if-ambiguous on # single Tab shows matches on ambiguity
|
||||||
|
set completion-ignore-case on # case-insensitive file/dir completion
|
||||||
|
set colored-stats on # colour ls-style completion list entries
|
||||||
|
set colored-completion-prefix on # highlight the matched prefix
|
||||||
|
set visible-stats on # append /*@ type indicators in completion
|
||||||
|
set mark-symlinked-directories on # add trailing / to symlinks to dirs
|
||||||
|
set skip-completed-text on # don't re-insert already-typed text
|
||||||
|
|
||||||
|
# Treat hyphens and underscores as equivalent when completing (e.g.
|
||||||
|
# typing `foo-` matches both `foo-bar` and `foo_bar`).
|
||||||
|
set completion-map-case on
|
||||||
Reference in New Issue
Block a user