23 Commits

Author SHA1 Message Date
08f7a1a9ee feat: add WireGuard Waybar custom module
- Added scripts/wireguard-status.sh — JSON output with lock/unlock icons,
  human-readable TX/RX stats, tooltip with peer/endpoint/IP details
- Added custom/wireguard module to Waybar modules.json (10s interval, JSON
  return type, click left/right for up/down)
- Added custom/wireguard to modules-right bar config
- Added CSS for #custom-wireguard (connected/disconnected classes)
- Added openresolv install and Waybar refresh instructions to README.org
2026-05-20 14:33:28 -04:00
7fd549fe00 feat: add WireGuard VPN keybinds (Super+Ctrl+V connect, Super+Ctrl+Shift+V disconnect) 2026-05-20 14:30:42 -04:00
7b1998a40d Merge pull request 'fix: replace deprecated hyprland dwindle options (pseudotile, togglesplit)' (#7) from fix/hyprland-deprecated-options into master
Reviewed-on: #7
2026-05-15 15:24:55 +00:00
9b4c678fd9 Merge branch 'master' into fix/hyprland-deprecated-options 2026-05-15 11:23:35 -04:00
53b376e6e3 fix: replace deprecated dwindle pseudotile and togglesplit options
- pseudotile variable removed from dwindle block in Hyprland 0.55+
- togglesplit standalone dispatcher removed, use pseudo dispatcher instead (matches original intent)
- Skill updated with absolute paths for common edit locations
2026-05-15 11:18:15 -04:00
6f6b24d3c3 Merge pull request 'fix: add local stow package to Makefile CONFIG_DIRS' (#5) from fix/add-local-stow into master
Reviewed-on: #5
2026-05-14 22:16:42 +00:00
7a0948a43f fix: add local stow package to Makefile 2026-05-14 18:16:08 -04:00
addf9fbe83 Merge pull request 'feat: add QET Xephyr launch script for Wayland drag-drop workaround' (#4) from feat/qet-xephyr-launch into master
Reviewed-on: #4
2026-05-14 22:14:23 +00:00
fb1d0cd2d9 fix: restore .desktop file for QET Xephyr 2026-05-14 17:38:46 -04:00
5f465ecb5a Revert "feat: add .desktop file for QET Xephyr wofi launch"
This reverts commit 8f5d2e73bc.
2026-05-14 17:38:23 -04:00
48038df872 Merge remote-tracking branch 'origin/feat/qet-xephyr-launch' into feat/qet-xephyr-launch 2026-05-14 17:38:13 -04:00
d5095064c2 feat: add wofi .desktop file for QET Xephyr launch 2026-05-14 17:37:51 -04:00
8f5d2e73bc feat: add .desktop file for QET Xephyr wofi launch 2026-05-14 17:23:55 -04:00
69c2447f04 Merge pull request 'feat: Add Hermes Agent vterm integration with SPC z h keybinding' (#1) from feat/hermes-vterm-integration into master
Reviewed-on: #1
2026-05-14 21:19:55 +00:00
4982831883 Command to send stuff to hermes 2026-05-14 17:12:10 -04:00
311a2eae57 Emacs config with the hermes shortcut 2026-05-14 17:11:08 -04:00
79ce045657 feat: add QET Xephyr launch script 2026-05-14 16:51:04 -04:00
Thierry Pouplier
386d1a3bf8 fix: add ssh -t to zsh alias, use vterm-mode instead of vterm in Emacs function
- zsh alias: added -t to SSH (docker exec -it needs a TTY)
- Emacs function: vterm-mode instead of vterm (vterm splits windows,
  vterm-mode is just a major mode, keeps current window layout)
- The SSH command in both cases is now consistent:
  ssh -t lazyworkhorse 'docker exec -it hermes /opt/hermes/.venv/bin/hermes'
2026-05-01 15:06:05 +00:00
Thierry Pouplier
a09a7c03ef fix: update zsh alias to direct SSH, fix Emacs function with delete-other-windows and correct docker exec path
- Zsh alias now runs ssh + docker exec directly (no emacsclient)
- Emacs function uses delete-other-windows for clean single-window vterm
- SSH command uses correct docker exec path: /opt/hermes/.venv/bin/hermes
- config.el not included - user will tangle from README.org
2026-05-01 15:02:19 +00:00
Thierry Pouplier
d73ed29733 fix: move hermes config from config.el to README.org org-babel source
- Revert config.el changes (file is generated from README.org via tangle)
- Add dedicated ** +Hermes Agent+ section under * AI in README.org
- Add SPC z h keybinding to existing map! block in Minimize window section
- config.el should NEVER be edited directly
2026-05-01 14:50:36 +00:00
Thierry Pouplier
a2a3baf7da feat: add hermes zsh alias (hermes) and Hyprland binding ( + C)
- zsh: alias hermes=emacsclient -n -c -e '(gortium/hermes-vterm)'
- hyprland: bind = , C to same command
- C for CLI/Chat, unused key in current layout
2026-05-01 14:40:02 +00:00
Thierry Pouplier
aa06568852 refactor: replace sleep-for vterm hack with process-send-string for Hermes launch
- Replaced sleep-for 0.5 + vterm-send-string timing hack with
  direct process-send-string (no timing race)
- Removed TUI chat command, SSH uses -t PTY for clean CLI mode
- Lines 947-957 in config.el
2026-05-01 14:35:48 +00:00
Thierry Pouplier
390ba2ef61 feat: Add Hermes Agent vterm integration with SPC z h keybinding 2026-04-30 14:35:33 +00:00
13 changed files with 285 additions and 147 deletions

View File

@@ -1,5 +1,5 @@
# Define list of directories you want to stow
CONFIG_DIRS := btop doom hypr kitty nvim rclone starship tmux wallpapers waybar wireplumber wofi yazi zsh
CONFIG_DIRS := btop doom hypr kitty local nvim rclone starship tmux wallpapers waybar wireplumber wofi yazi zsh
all: stow_all
wal -i ~/.config/wallpapers/green_yellow_forest.jpg -o ~/.config/waybar/launch.sh

View File

@@ -17,6 +17,42 @@ I use stow to deploy all this to where it need to go (mostly ~/.confg).
** starship
** tmux
** waybar
*** WireGuard Module
The Waybar includes a custom WireGuard VPN status module that shows connection status, TX/RX stats, and peer info.
Setup:
1. Install openresolv (needed by wg-quick for DNS):
#+begin_src bash :tangle no
sudo pacman -S openresolv
#+end_src
2. Configure sudo for wg-quick (required for click-to-toggle binds):
#+begin_src bash :tangle no
echo "$USER ALL=(ALL) NOPASSWD: /usr/bin/wg-quick" | sudo tee /etc/sudoers.d/wireguard
#+end_src
3. Create a WireGuard config (if not exists):
#+begin_src bash :tangle no
sudo cp /etc/wireguard/work-laptop.conf /etc/wireguard/
sudo chmod 600 /etc/wireguard/work-laptop.conf
#+end_src
4. Refresh Waybar to pick up new config:
#+begin_src bash :tangle no
killall -q waybar && waybar -c ~/.config/waybar/config.jsonc & -s ~/.config/waybar/style.css &
#+end_src
Or use the launch script:
#+begin_src bash :tangle no
~/.config/waybar/launch.sh
#+end_src
Keybinds:
- ~$mainMod+Ctrl+V~ — Connect WireGuard (work-laptop)
- ~$mainMod+Ctrl+Shift+V~ — Disconnect WireGuard
The module polls /usr/bin/wg every 10 seconds. It shows a lock icon with TX/RX when connected, or an unlock icon when disconnected.
** Wireplumber
** wofi
** yazi

View File

@@ -112,7 +112,9 @@ My current workflow consist in having the 3-5 files I work on open in vertical s
#+begin_src emacs-lisp
(map! :leader
(:prefix ("w" . "window")
:desc "Minimize window" "O" #'minimize-window))
:desc "Minimize window" "O" #'minimize-window)
(:prefix ("z" . "Personal")
:desc "Hermes Agent" "h" #'gortium/hermes-vterm))
#+end_src
** Ultra-scroll
@@ -1830,6 +1832,25 @@ Org-AI
(org-ai-install-yasnippets)) ; if you are using yasnippet and want `ai` snippets
#+end_src
** +Hermes Agent+
Hermes Agent runs on lazyworkhorse server. Launched via SSH in a vterm buffer.
#+begin_src emacs-lisp
(after! vterm
(defun gortium/hermes-vterm ()
"Launch Hermes Agent in vterm via SSH to lazyworkhorse (CLI mode)."
(interactive)
(let ((buf (get-buffer-create "*hermes*")))
(switch-to-buffer buf)
(unless (and (eq major-mode 'vterm-mode)
(get-buffer-process buf))
(vterm-mode)
(let ((proc (get-buffer-process buf)))
(when proc
(process-send-string
proc "ssh -t lazyworkhorse 'docker exec -it hermes /opt/hermes/.venv/bin/hermes'\n")))))))
#+end_src
* Tramp
Remote connection to server inside emacs (with all my config) <3

View File

@@ -18,7 +18,9 @@
(map! :leader
(:prefix ("w" . "window")
:desc "Minimize window" "O" #'minimize-window))
:desc "Minimize window" "O" #'minimize-window)
(:prefix ("z" . "Personal")
:desc "Hermes Agent" "h" #'gortium/hermes-vterm))
(use-package! ultra-scroll
:init
@@ -1053,6 +1055,20 @@ DIFF:
:desc "GPT generate commit"
"g" #'gortium/magit-gptel-generate-commit)
(after! vterm
(defun gortium/hermes-vterm ()
"Launch Hermes Agent in vterm via SSH to lazyworkhorse (CLI mode)."
(interactive)
(let ((buf (get-buffer-create "*hermes*")))
(switch-to-buffer buf)
(unless (and (eq major-mode 'vterm-mode)
(get-buffer-process buf))
(vterm-mode)
(let ((proc (get-buffer-process buf)))
(when proc
(process-send-string
proc "ssh -t lazyworkhorse 'docker exec -it hermes /opt/hermes/.venv/bin/hermes'\n")))))))
;; set `tramp-direct-async-process' locally in all ssh connections
(connection-local-set-profile-variables
'remote-direct-async-process

View File

@@ -73,7 +73,6 @@ exec-once = swww img ~/.config/wallpapers/green_yellow_forest.jpg --outputs $EXT
exec-once = swww img ~/.config/wallpapers/buck_head.jpg --outputs $EXTERNAL_R1
exec-once = swww img ~/.config/wallpapers/buck_head.jpg --outputs $EXTERNAL_R2
exec-once = swaync
exec-once = ~/.config/waybar/scripts/disk-monitor.sh daemon
exec-once = hypridle
exec-once = hyprlock
exec-once = /usr/bin/emacs --daemon
@@ -148,7 +147,8 @@ animations {
# See https://wiki.hyprland.org/Configuring/Dwindle-Layout/ for more
dwindle {
pseudotile = true # Master switch for pseudotiling. Enabling is bound to mainMod + P in the keybinds section below
# Note: pseudotile variable removed in 0.55+. Use `pseudo` dispatcher to toggle per-window.
# Bound to $mainMod + P below.
# This prevents the layout from collapsing if you close one window
preserve_split = true # You probably want this
@@ -220,14 +220,19 @@ bind = $mainMod, E, exec, $editor
bind = $mainMod, V, togglefloating,
bind = $mainMod, R, exec, $menu
bind = $mainMod, Z, exec, $zen
#bind = $mainMod, P, pseudo, # dwindle
bind = $mainMod, P, togglesplit, # dwindle
# togglesplit removed as standalone dispatcher in 0.55+; use `layoutmsg togglesplit` instead
bind = $mainMod, P, pseudo, # dwindle
bind = $mainMod, N, movecurrentworkspacetomonitor, -1
bind = $mainMod SHIFT, N, movecurrentworkspacetomonitor, +1
bind = $mainMod, X, swapwindow
bind = $mainMod, B, exec, python /home/tpouplier/ExoKortex/4-Automata/recordntrans/record_transcribe.py
# bind = $mainMod, G, exec, nerd-dictation end
bind = $mainMod, A, exec, $music
bind = $mainMod, C, exec, emacsclient -n -c -e '(gortium/hermes-vterm)'
# WireGuard VPN toggle (uses config file: /etc/wireguard/work-laptop.conf)
bind = $mainMod CTRL, V, exec, sudo wg-quick up work-laptop
bind = $mainMod CTRL SHIFT, V, exec, sudo wg-quick down work-laptop
# Full screen
bind = SUPER, F, fullscreen

View File

@@ -0,0 +1,8 @@
[Desktop Entry]
Name=QET (Xephyr)
Comment=QElectroTech in nested X server for Wayland compatibility
Exec=/home/tpouplier/ExoKortex/4-Automata/dotfiles/scripts/qet-xephyr.sh
Icon=qelectrotech
Type=Application
Categories=Graphics;Engineering;
Terminal=false

21
scripts/qet-xephyr.sh Executable file
View File

@@ -0,0 +1,21 @@
#!/bin/bash
# Launch QElectroTech in a nested Xephyr X server with Openbox
# Works around Wayland drag-and-drop issue in QET
XEPHYR_PID=""
cleanup() {
kill $XEPHYR_PID 2>/dev/null
exit 0
}
trap cleanup SIGINT SIGTERM
Xephyr -screen 1920x1080 :1 -title QET -br &
XEPHYR_PID=$!
sleep 1
DISPLAY=:1 openbox &
sleep 0.5
DISPLAY=:1 qelectrotech &
wait $XEPHYR_PID

129
scripts/wireguard-status.sh Executable file
View File

@@ -0,0 +1,129 @@
#!/usr/bin/env bash
# wireguard-status.sh -- JSON status for Waybar
# Outputs: {"text":"...","class":"connected|disconnected","tooltip":"..."}
#
# Waybar config example (in ~/.config/waybar/config.jsonc):
# "custom/wireguard": {
# "exec": "$HOME/.config/waybar/scripts/wireguard-status.sh",
# "interval": 10,
# "return-type": "json"
# }
set -euo pipefail
# --- Config -------------------------------------------------------
ICON_LOCKED="\uf023" # nf-fa-lock (FontAwesome)
ICON_UNLOCKED="\uf09e" # nf-fa-unlock
# Override with environment variables if set:
ICON_LOCKED="${WG_ICON_LOCKED:-$ICON_LOCKED}"
ICON_UNLOCKED="${WG_ICON_UNLOCKED:-$ICON_UNLOCKED}"
# --- Helpers ------------------------------------------------------
die() { echo "$*" >&2; exit 1; }
# Find the wg binary — works on NixOS, standard Linux, etc.
find_wg() {
if command -v wg &>/dev/null; then
echo "wg"
elif [ -x /run/current-system/sw/bin/wg ]; then
echo "/run/current-system/sw/bin/wg"
elif [ -x /usr/bin/wg ]; then
echo "/usr/bin/wg"
else
return 1
fi
}
# Find the ip binary
find_ip() {
if command -v ip &>/dev/null; then
echo "ip"
elif [ -x /run/current-system/sw/bin/ip ]; then
echo "/run/current-system/sw/bin/ip"
elif [ -x /usr/sbin/ip ]; then
echo "/usr/sbin/ip"
elif [ -x /sbin/ip ]; then
echo "/sbin/ip"
else
return 1
fi
}
format_bytes() {
local bytes=$1
if [ "$bytes" -lt 1024 ]; then
echo "${bytes}B"
elif [ "$bytes" -lt $((1024 * 1024)) ]; then
echo "$(awk "BEGIN { printf \"%.1f KiB\", $bytes / 1024 }")"
elif [ "$bytes" -lt $((1024 * 1024 * 1024)) ]; then
echo "$(awk "BEGIN { printf \"%.1f MiB\", $bytes / (1024 * 1024) }")"
else
echo "$(awk "BEGIN { printf \"%.2f GiB\", $bytes / (1024 * 1024 * 1024) }")"
fi
}
# --- Main ---------------------------------------------------------
WG_BIN=$(find_wg 2>/dev/null) || {
# wg not installed — report as disconnected
printf '{"text":"%s","class":"disconnected","tooltip":"WireGuard not installed"}\n' "$ICON_UNLOCKED"
exit 0
}
IP_BIN=$(find_ip 2>/dev/null) || true
# Find all WireGuard interfaces
INTERFACES=$("$WG_BIN" show interfaces 2>/dev/null) || {
# No WireGuard interfaces exist (wg module not loaded?)
printf '{"text":"%s","class":"disconnected","tooltip":"No WireGuard interfaces"}\n' "$ICON_UNLOCKED"
exit 0
}
# Parse into array, handling possible single-line space-separated output
IFS=' ' read -ra ifaces <<< "$INTERFACES"
if [ ${#ifaces[@]} -eq 0 ]; then
printf '{"text":"%s","class":"disconnected","tooltip":"No WireGuard interfaces"}\n' "$ICON_UNLOCKED"
exit 0
fi
# Just use the first interface (most setups have one)
IFACE="${ifaces[0]}"
# Get transfer stats
TRANSFER=$("$WG_BIN" show "$IFACE" transfer 2>/dev/null) || {
printf '{"text":"%s","class":"disconnected","tooltip":"Cannot read stats for %s"}\n' "$ICON_UNLOCKED" "$IFACE"
exit 0
}
# Parse: "rx_bytes tx_bytes"
RX_BYTES=$(echo "$TRANSFER" | awk '{print $1}')
TX_BYTES=$(echo "$TRANSFER" | awk '{print $2}')
RX_HUMAN=$(format_bytes "${RX_BYTES:-0}")
TX_HUMAN=$(format_bytes "${TX_BYTES:-0}")
# Get endpoint info for tooltip
ENDPOINT=$("$WG_BIN" show "$IFACE" endpoints 2>/dev/null | head -1 | awk '{print $2}') || ENDPOINT=""
PEER=$("$WG_BIN" show "$IFACE" peers 2>/dev/null | head -1) || PEER=""
# Build status line
STATUS_TEXT="${ICON_LOCKED} \u2193${RX_HUMAN} \u2191${TX_HUMAN}"
# Build tooltip
TOOLTIP="${IFACE}"
if [ -n "$PEER" ]; then
TOOLTIP="${TOOLTIP}\nPeer: ${PEER}"
fi
if [ -n "$ENDPOINT" ]; then
TOOLTIP="${TOOLTIP}\nEndpoint: ${ENDPOINT}"
fi
# Add IP info if ip is available
if [ -n "$IP_BIN" ]; then
IFACE_IP=$("$IP_BIN" -4 addr show dev "$IFACE" 2>/dev/null | awk '/inet / {print $2}') || IFACE_IP=""
if [ -n "$IFACE_IP" ]; then
TOOLTIP="${TOOLTIP}\nIP: ${IFACE_IP}"
fi
fi
TOOLTIP="${TOOLTIP}\nRX: ${RX_HUMAN}\nTX: ${TX_HUMAN}"
printf '{"text":"%s","class":"connected","tooltip":"%s"}\n' "$STATUS_TEXT" "$TOOLTIP"

View File

@@ -25,10 +25,10 @@
"modules-right": [
"group/hardware",
"custom/disk-monitor",
"pulseaudio",
"bluetooth",
"network",
"custom/wireguard",
"battery",
"custom/exit"
]

View File

@@ -138,6 +138,15 @@
//"on-click": "~/dotfiles/.settings/networkmanager.sh"
},
// wireguard
"custom/wireguard": {
"exec": "$HOME/.config/waybar/scripts/wireguard-status.sh",
"interval": 10,
"return-type": "json",
"on-click": "kitty sudo wg-quick up work-laptop",
"on-click-right": "kitty sudo wg-quick down work-laptop"
},
// battery
"battery": {
"states": {
@@ -158,12 +167,4 @@
"on-click": "wlogout -b 2",
"tooltip": false
},
// disk space monitor (background daemon for notifications)
"custom/disk-monitor": {
"exec": "~/.config/waybar/scripts/disk-monitor.sh waybar",
"return-type": "json",
"interval": 60,
"signal": 8
}
}

View File

@@ -1,119 +0,0 @@
#!/usr/bin/env bash
# Disk Space Monitor — warns you before you hit the wall.
#
# Two integration points:
# 1. Waybar: add "custom/disk" to modules-right in config.jsonc
# 2. Hyprland: add exec-once = ... disk-monitor.sh daemon
#
# Install:
# cp disk-monitor.sh ~/.config/waybar/scripts/
# chmod +x ~/.config/waybar/scripts/disk-monitor.sh
set -euo pipefail
WARN_PCT=85
CRIT_PCT=92
# ─── helpers ────────────────────────────────────────────────────────────────
_notify() {
local urgency="$1" title="$2" body="$3"
notify-send -u "$urgency" -a "disk-monitor" "$title" "$body"
}
# ─── commands ───────────────────────────────────────────────────────────────
cmd_check() {
local alerts=""
local any_alert=false
while IFS= read -r line; do
# Skip headers and pseudo-filesystems
target=$(echo "$line" | awk '{print $1}')
pct=$(echo "$line" | awk '{print $2}' | tr -d '%')
used=$(echo "$line" | awk '{print $3}')
size=$(echo "$line" | awk '{print $4}')
avail=$(echo "$line" | awk '{print $5}')
case "$target" in
/dev|/sys|/proc|/run|tmpfs|devtmpfs|none|udev|overlay|shm|devpts|mqueue|pts) continue ;;
/boot*) continue ;;
*) ;;
esac
if [ "$pct" -ge "$CRIT_PCT" ]; then
any_alert=true
alerts+="🔴 $target ${pct}% (${avail} free)\n"
elif [ "$pct" -ge "$WARN_PCT" ]; then
any_alert=true
alerts+="🟡 $target ${pct}% (${avail} free)\n"
fi
done < <(df -h --output=target,pcent,used,size,avail | tail -n +2)
if [ -z "$alerts" ]; then
return 0
fi
if echo "$alerts" | grep -q "🔴"; then
_notify critical "🚨 Disk CRITICAL" "$alerts"
else
_notify normal "⚠️ Disk space low" "$alerts"
fi
}
cmd_daemon() {
local interval="${1:-300}"
echo "disk-monitor: checking every ${interval}s, warn@${WARN_PCT}% crit@${CRIT_PCT}%"
cmd_check
while true; do
sleep "$interval"
cmd_check
done
}
cmd_waybar() {
local worst_pct=0
local worst_target="/"
local tooltip=""
while IFS= read -r line; do
target=$(echo "$line" | awk '{print $1}')
pct=$(echo "$line" | awk '{print $2}' | tr -d '%')
used=$(echo "$line" | awk '{print $3}')
size=$(echo "$line" | awk '{print $4}')
avail=$(echo "$line" | awk '{print $5}')
case "$target" in
/dev|/sys|/proc|/run|tmpfs|devtmpfs|none|udev|overlay|shm|devpts|mqueue|pts) continue ;;
/boot*) continue ;;
*) ;;
esac
tooltip+="${target}: ${pct}% (${avail} free)\\n"
[ "$pct" -gt "$worst_pct" ] && { worst_pct=$pct; worst_target=$target; }
done < <(df -h --output=target,pcent,used,size,avail | tail -n +2)
local icon class
if [ "$worst_pct" -ge "$CRIT_PCT" ]; then
icon="🔴"; class="critical"
elif [ "$worst_pct" -ge "$WARN_PCT" ]; then
icon="🟡"; class="warning"
else
icon="💾"; class="normal"
fi
printf '{"text": " %s %s %d%% ", "tooltip": "%s", "class": "%s"}\n' \
"$icon" "$worst_target" "$worst_pct" "$tooltip" "$class"
}
case "${1:-help}" in
daemon) cmd_daemon "${2:-300}" ;;
waybar|w) cmd_waybar ;;
check|notify) cmd_check ;;
*)
echo "Usage: $(basename "$0") {daemon|waybar|check}"
echo " daemon Background check every 5 min, desktop notifications"
echo " waybar One-shot JSON for Waybar custom module"
echo " check Single check, prints to stdout"
exit 1
;;
esac

View File

@@ -213,7 +213,7 @@ tooltip label {
}
/* custom modules */
#custom-disk-monitor {
#custom-wireguard {
margin: 0px 0px 0px 5px;
padding: 1px 8px 0px 8px;
font-size: 16px;
@@ -221,18 +221,11 @@ tooltip label {
border-radius: 5px;
background-color: @set;
}
#custom-disk-monitor.warning {
background-color: #ff9a3c;
color: #FFFFFF;
#custom-wireguard.connected {
background-color: @color5;
}
#custom-disk-monitor.critical {
background-color: #dc2f2f;
color: #FFFFFF;
animation-name: blink;
animation-duration: 0.5s;
animation-timing-function: linear;
animation-iteration-count: infinite;
animation-direction: alternate;
#custom-wireguard.disconnected {
background-color: @set;
}
#custom-exit {

View File

@@ -19,6 +19,33 @@ alias emacs="emacsclient -c -a emacs"
# Opencode
alias oc="opencode"
# Hermes Agent - SSH directly to remote server
alias hermes="ssh -t lazyworkhorse 'docker exec -it hermes /opt/hermes/.venv/bin/hermes'"
hermes-cp() {
if [ $# -eq 0 ]; then
echo "Usage: hermes-cp <file_or_directory> [...]"
echo ""
echo "Copies files/directories from your Arch laptop into the Hermes"
echo "agent container at /opt/data/home/ preserving directory structure."
echo ""
echo "Examples:"
echo " hermes-cp document.pdf # single file"
echo " hermes-cp *.pdf # multiple files (glob)"
echo " hermes-cp manuals/ # whole directory"
echo " hermes-cp manuals/ keyence-docs/ # multiple items"
echo ""
echo "To copy to a different destination inside the container:"
echo " tar -czf - . | ssh gortium@lazyworkhorse \\"
echo " \"docker exec -i hermes sh -c 'cd /opt/data/projects && tar -xzf -'\""
return 1
fi
# tar everything into a single SSH pipe, extract inside container
# Supports: single files, directories, glob patterns, mixed args
tar -czf - "$@" | ssh gortium@lazyworkhorse \
"docker exec -i hermes sh -c 'cd /opt/data/home && tar -xzf -'"
}
# Yazi
alias y="yazi"
function yy() {