Compare commits
20 Commits
fish-confi
...
4997598194
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4997598194 | ||
|
|
01cefdc6fd | ||
|
|
c5ba20c814 | ||
|
|
161c2c2291 | ||
|
|
dfca731f77 | ||
|
|
2b9d0835f0 | ||
|
|
07e034d5f4 | ||
|
|
7884d4fb61 | ||
|
|
ac6c6203c3 | ||
| 566225ac04 | |||
| 977d11b8de | |||
| a64e7bceff | |||
|
|
7de1042823 | ||
|
|
6982026a88 | ||
|
|
d96ef0af90 | ||
|
|
3049391a4f | ||
|
|
138d76a19f | ||
|
|
57d0ebda20 | ||
|
|
726e6098b2 | ||
|
|
6b963ee277 |
33
CLAUDE.md
Normal file
33
CLAUDE.md
Normal file
@@ -0,0 +1,33 @@
|
||||
# CLAUDE.md
|
||||
|
||||
Dotfiles & workstation automation for Linux, macOS, WSL. `bashrc` loads `rc/*.sh` (ordered by prefix) and adds `bin/` to PATH. `$LWS` points to repo root.
|
||||
|
||||
## Directory Layout
|
||||
|
||||
- `rc/` — Shell init files, loaded in `0X-name.sh` order. Per-tool env setup goes here.
|
||||
- `install/` — Idempotent one-off installer scripts (run manually).
|
||||
- `bin/` — Scripts auto-added to PATH. New CLI tools go here.
|
||||
- `scripts/` — Setup scripts NOT on PATH (git-config, docker fixes, gnome config).
|
||||
- `conf/` — Tool configs (vim, WireGuard, WSL).
|
||||
- `functions.sh` — Core lib: `append_path`, `prepend_path`, `set_uniq_path`, `source_directory_sh`, `is_fast_init`/`is_slow_init`, WSL detection, debug logging.
|
||||
|
||||
## Conventions
|
||||
|
||||
All new bash scripts MUST use:
|
||||
```bash
|
||||
#!/usr/bin/env bash
|
||||
set -E -o errexit -o nounset -o pipefail
|
||||
```
|
||||
|
||||
Before writing shell code, check `functions.sh` for existing helper functions (e.g. `can_run`, `append_path_try`, `source_try`) and prefer them over raw commands.
|
||||
|
||||
RC files MUST guard slow tools and check availability:
|
||||
```bash
|
||||
if is_slow_init && can_run pyenv; then
|
||||
# initialize pyenv
|
||||
fi
|
||||
```
|
||||
|
||||
Installers (`install/*.sh`) MUST be idempotent — check before installing.
|
||||
|
||||
Env vars: `LWS_FAST=1` skips slow tools, `LWS_DEBUG=1` enables debug logging.
|
||||
75
bin/xclaude
Executable file
75
bin/xclaude
Executable file
@@ -0,0 +1,75 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
# https://code.claude.com/docs/en/settings
|
||||
|
||||
import argparse
|
||||
import os
|
||||
import sys
|
||||
import urllib.parse
|
||||
|
||||
env = os.environ
|
||||
|
||||
|
||||
def die(msg):
|
||||
print(f"Error: {msg}", file=sys.stderr)
|
||||
sys.exit(1)
|
||||
|
||||
|
||||
def validate_url(url):
|
||||
parsed = urllib.parse.urlparse(url)
|
||||
if parsed.scheme not in ("http", "https"):
|
||||
die(f"invalid URL scheme '{parsed.scheme}' in --url (expected http or https)")
|
||||
|
||||
parser = argparse.ArgumentParser(
|
||||
prog="xclaude",
|
||||
description="Run Claude Code with any OpenAI-compatible API backend.",
|
||||
epilog="""\
|
||||
Environment variables (overridable by arguments):
|
||||
CLAUDE_URL - API base URL (e.g. https://openrouter.ai/api/v1)
|
||||
CLAUDE_TOKEN - API token/key
|
||||
CLAUDE_MODEL - model name
|
||||
|
||||
If neither URL nor token is set, defaults to Ollama:
|
||||
OLLAMA_HOST - Ollama server address (default: nvidia.hell), or use --host
|
||||
OLLAMA_MODEL - Ollama model (default: glm-5:cloud)
|
||||
|
||||
Any extra arguments are passed through directly to the claude CLI.
|
||||
|
||||
Examples:
|
||||
xclaude
|
||||
xclaude --model qwen3-coder
|
||||
xclaude --url https://openrouter.ai/api/v1 --token sk-XXX --model openai/gpt-4o
|
||||
""",
|
||||
formatter_class=argparse.RawDescriptionHelpFormatter,
|
||||
)
|
||||
parser.add_argument("--url", default=env.get("CLAUDE_URL", ""))
|
||||
parser.add_argument("--token", default=env.get("CLAUDE_TOKEN", ""))
|
||||
parser.add_argument("--model", default=env.get("CLAUDE_MODEL", ""))
|
||||
parser.add_argument("--host", default=env.get("OLLAMA_HOST", "nvidia.hell"))
|
||||
args, rest = parser.parse_known_args()
|
||||
|
||||
env["DISABLE_TELEMETRY"] = "1"
|
||||
|
||||
is_external_api = bool(args.url or args.token)
|
||||
if is_external_api:
|
||||
if not args.url:
|
||||
die("no URL specified (--url or CLAUDE_URL)")
|
||||
validate_url(args.url)
|
||||
if not args.token:
|
||||
die("no token specified (--token or CLAUDE_TOKEN)")
|
||||
if not args.model:
|
||||
die("no model specified (--model or CLAUDE_MODEL)")
|
||||
env["ANTHROPIC_BASE_URL"] = args.url
|
||||
env["ANTHROPIC_API_KEY"] = args.token
|
||||
# Remove Ollama auth token if switching to external API
|
||||
env.pop("ANTHROPIC_AUTH_TOKEN", None)
|
||||
model = args.model
|
||||
else:
|
||||
host = args.host
|
||||
model = args.model or env.get("OLLAMA_MODEL", "glm-5:cloud")
|
||||
env["ANTHROPIC_BASE_URL"] = f"http://{host}:11434"
|
||||
env["ANTHROPIC_AUTH_TOKEN"] = "ollama"
|
||||
env.pop("ANTHROPIC_API_KEY", None)
|
||||
|
||||
cmd = ["claude", "--model", model] + rest
|
||||
os.execvp("claude", cmd)
|
||||
5
bin/xvim
Executable file
5
bin/xvim
Executable file
@@ -0,0 +1,5 @@
|
||||
#!/bin/bash
|
||||
|
||||
vim "$@" --cmd 'au VimLeave * :!clear'
|
||||
clear
|
||||
|
||||
21
conf/wsl.conf
Normal file
21
conf/wsl.conf
Normal file
@@ -0,0 +1,21 @@
|
||||
# https://learn.microsoft.com/en-us/windows/wsl/wsl-config
|
||||
|
||||
[boot]
|
||||
systemd=false
|
||||
|
||||
[user]
|
||||
default=lachtan
|
||||
|
||||
[automount]
|
||||
enabled = true
|
||||
|
||||
[network]
|
||||
generateHosts = true
|
||||
generateResolvConf = false
|
||||
|
||||
[interop]
|
||||
enabled = false
|
||||
appendWindowsPath = false
|
||||
|
||||
[gpu]
|
||||
enabled = true
|
||||
3
fishrc
Normal file
3
fishrc
Normal file
@@ -0,0 +1,3 @@
|
||||
for file in (dirname (status -f))/rc.fish/*.fish
|
||||
source $file
|
||||
end
|
||||
@@ -1,4 +1,5 @@
|
||||
# Linux Workspace Functions
|
||||
# shellcheck shell=bash
|
||||
|
||||
if [ -z "$time_ms" ]; then
|
||||
readonly time_ms='date +%s%3N'
|
||||
@@ -134,3 +135,7 @@ function true_false() {
|
||||
# [[ "${1,,}" =~ ^(1|true|yes)$ ]]
|
||||
[[ "${1@L}" =~ ^(1|true|yes)$ ]]
|
||||
}
|
||||
|
||||
function is_wsl() {
|
||||
grep -q -i wsl /proc/version
|
||||
}
|
||||
|
||||
34
install.sh
34
install.sh
@@ -2,11 +2,29 @@
|
||||
|
||||
set -E -o errexit -o nounset -o pipefail
|
||||
|
||||
if grep -q -e 'source\s+.*/linux-workspace/bashrc\s*$' $HOME/.bashrc; then
|
||||
echo "Linux Workspace configuration already exists in .bashrc !"
|
||||
exit 1
|
||||
else
|
||||
readonly SCRIPT_DIR=$(cd "$(dirname "${BASH_SOURCE[0]}")" &>/dev/null && pwd -P)
|
||||
echo "Adding to $HOME/.bashrc"
|
||||
echo "source '$SCRIPT_DIR/bashrc'" >> "$HOME/.bashrc"
|
||||
fi
|
||||
function install_lws() {
|
||||
local LWS_DIR=$(cd "$(dirname "${BASH_SOURCE[0]}")" &>/dev/null && pwd -P)
|
||||
|
||||
if grep -q -e "source.*$LWS_DIR/bashrc" $HOME/.bashrc; then
|
||||
echo "SKIP: Linux Workspace configuration already exists in $HOME/.bashrc !"
|
||||
else
|
||||
echo "Adding to $HOME/.bashrc"
|
||||
echo "source '$LWS_DIR/bashrc'" >> "$HOME/.bashrc"
|
||||
fi
|
||||
|
||||
local FISH_CONF_DIR="$HOME/.config/fish"
|
||||
local FISH_CONF_LWS="$FISH_CONF_DIR/conf.d/lws.fish"
|
||||
|
||||
if [[ -d "$FISH_CONF_DIR" ]]; then
|
||||
if [[ -f "$FISH_CONF_LWS" ]]; then
|
||||
echo "SKIP: Linux Workspace configuration already exists in $FISH_CONF_LWS !"
|
||||
else
|
||||
echo "Adding to $FISH_CONF_LWS"
|
||||
echo "source $LWS_DIR/fishrc" > "$FISH_CONF_LWS"
|
||||
fi
|
||||
else
|
||||
echo "Fish configuration directory not found, skipping fish setup."
|
||||
fi
|
||||
}
|
||||
|
||||
install_lws
|
||||
|
||||
9
install/claude-code.sh
Executable file
9
install/claude-code.sh
Executable file
@@ -0,0 +1,9 @@
|
||||
#!/bin/bash
|
||||
|
||||
# https://code.claude.com/docs/en/setup
|
||||
|
||||
set -E -o errexit -o nounset -o pipefail
|
||||
set -x
|
||||
|
||||
curl -fsSL https://claude.ai/install.sh | bash
|
||||
|
||||
8
install/gemini-cli.sh
Executable file
8
install/gemini-cli.sh
Executable file
@@ -0,0 +1,8 @@
|
||||
#!/bin/bash
|
||||
|
||||
# https://github.com/google-gemini/gemini-cli
|
||||
|
||||
set -E -o errexit -o nounset -o pipefail
|
||||
set -x
|
||||
|
||||
npm install -g @google/gemini-cli
|
||||
10
install/nvm-npm.sh
Executable file
10
install/nvm-npm.sh
Executable file
@@ -0,0 +1,10 @@
|
||||
#!/bin/bash
|
||||
|
||||
# https://github.com/nvm-sh/nvm
|
||||
|
||||
set -E -o errexit -o nounset -o pipefail
|
||||
set -x
|
||||
|
||||
curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.40.3/install.sh | bash
|
||||
nvm install node
|
||||
nvm use node
|
||||
5
rc.fish/01-init.fish
Normal file
5
rc.fish/01-init.fish
Normal file
@@ -0,0 +1,5 @@
|
||||
set -gx PAGER less
|
||||
set -g fish_prompt_pwd_dir_length 0
|
||||
|
||||
# set -g fish_autosuggestion_enabled 0
|
||||
# fish_config theme choose "fish default"
|
||||
@@ -1 +1,2 @@
|
||||
prepend_path_try "$HOME/.local/bin"
|
||||
prepend_path_try "$HOME/bin"
|
||||
|
||||
@@ -1,5 +1,11 @@
|
||||
alias nop=':'
|
||||
|
||||
# Long format, human-readable, with indicators
|
||||
alias ll='ls -lhF'
|
||||
# Same as ll but including hidden files
|
||||
alias la='ls -lhAF'
|
||||
|
||||
|
||||
alias exit-when-error='set -o errexit'
|
||||
alias exit-when-unset-variable='set -o nounset'
|
||||
alias exit-when-pipe-fail='set -o pipefail'
|
||||
@@ -9,6 +15,7 @@ alias clws='cd $LWS'
|
||||
alias chs=cheatsheet
|
||||
|
||||
alias dc=docker-compose
|
||||
alias pc=podman-compose
|
||||
alias dcl='dcm logs'
|
||||
alias dcfilter="sed -r 's/^[^|]+\| //g'"
|
||||
|
||||
|
||||
@@ -1,3 +1 @@
|
||||
if [[ -f "$HOME/.cargo/env" ]]; then
|
||||
source "$HOME/.cargo/env"
|
||||
fi
|
||||
source_try "$HOME/.cargo/env"
|
||||
|
||||
1
rc/claude-code.sh
Normal file
1
rc/claude-code.sh
Normal file
@@ -0,0 +1 @@
|
||||
append_path_try $HOME/.local/bin
|
||||
@@ -5,7 +5,7 @@
|
||||
#COLORS_FILE="$LWS/conf/dircolors/ansi-light.dircolors"
|
||||
COLORS_FILE="$LWS/conf/dircolors/dracula.dircolors"
|
||||
|
||||
if [[ -f "$COLORS_FILE" ]]; then
|
||||
if can_run dircolors && [[ -f "$COLORS_FILE" ]]; then
|
||||
xlog "Loading colors $COLORS_FILE"
|
||||
eval "$(dircolors "$COLORS_FILE")"
|
||||
else
|
||||
|
||||
8
rc/nvm.sh
Normal file
8
rc/nvm.sh
Normal file
@@ -0,0 +1,8 @@
|
||||
NVM_DIR="$HOME/.nvm"
|
||||
if [ -d "$NVM_DIR" ]; then
|
||||
export NVM_DIR
|
||||
[ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh"
|
||||
[ -s "$NVM_DIR/bash_completion" ] && \. "$NVM_DIR/bash_completion"
|
||||
else
|
||||
unset NVM_DIR
|
||||
fi
|
||||
1
rc/opencode.sh
Normal file
1
rc/opencode.sh
Normal file
@@ -0,0 +1 @@
|
||||
append_path_try "$HOME/.opencode/bin"
|
||||
@@ -1,9 +1,51 @@
|
||||
# Start ssh agent for private key
|
||||
|
||||
function _log() {
|
||||
xlog "[ssh-agent] $*"
|
||||
}
|
||||
|
||||
_SSH_AGENT_NO_KEYS=1
|
||||
_SSH_AGENT_NOT_RUNNING=2
|
||||
|
||||
for key in id_ecdsa id_rsa; do
|
||||
key_filename="$HOME/.ssh/$key"
|
||||
if [ -f $key_filename ]; then
|
||||
eval $(keychain --eval --quiet --agents ssh $key)
|
||||
|
||||
if [ -f "$key_filename" ]; then
|
||||
_log "SSH key: $key_filename"
|
||||
|
||||
if is_wsl; then
|
||||
_log "WSL ON"
|
||||
|
||||
if [ -z "$SSH_AUTH_SOCK" ]; then
|
||||
export SSH_AUTH_SOCK=$HOME/.ssh/ssh-agent.sock
|
||||
fi
|
||||
|
||||
_log "SSH_AUTH_SOCK: $SSH_AUTH_SOCK"
|
||||
|
||||
ssh-add -L &> /dev/null
|
||||
ssh_add_status=$?
|
||||
|
||||
if [[ -S "$SSH_AUTH_SOCK" && $ssh_add_status -eq $_SSH_AGENT_NOT_RUNNING ]]; then
|
||||
_log "SSH agent does not running, delete $SSH_AUTH_SOCK"
|
||||
rm -f "$SSH_AUTH_SOCK"
|
||||
fi
|
||||
|
||||
if [[ ! -S "$SSH_AUTH_SOCK" || $ssh_add_status -eq $_SSH_AGENT_NOT_RUNNING ]]; then
|
||||
_log "Starting ssh-agent"
|
||||
eval "$(ssh-agent -s -a "$SSH_AUTH_SOCK")"
|
||||
fi
|
||||
|
||||
ssh-add -L &> /dev/null
|
||||
ssh_add_status=$?
|
||||
|
||||
if [[ $ssh_add_status -eq $_SSH_AGENT_NO_KEYS ]]; then
|
||||
_log "Adding SSH key to agent"
|
||||
ssh-add "$key_filename"
|
||||
fi
|
||||
else
|
||||
eval "$(keychain --eval --quiet --agents ssh $key)"
|
||||
fi
|
||||
|
||||
break
|
||||
fi
|
||||
done
|
||||
|
||||
Reference in New Issue
Block a user