mirror of
https://github.com/Gu1llaum-3/sshm.git
synced 2025-10-22 10:57:20 +02:00
Compare commits
9 Commits
main
...
v1.6.0-dev
Author | SHA1 | Date | |
---|---|---|---|
ab5f430ee1 | |||
3da3a33530 | |||
12c1ab476c | |||
e0e50ebfd0 | |||
947afb2bbe | |||
fe529792e3 | |||
5ee623d054 | |||
09423287fd | |||
4767267387 |
136
.github/workflows/build.yml
vendored
Normal file
136
.github/workflows/build.yml
vendored
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
name: Build Binaries
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- '*'
|
||||||
|
release:
|
||||||
|
types: [created]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
name: Build
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
# Linux AMD64
|
||||||
|
- goos: linux
|
||||||
|
goarch: amd64
|
||||||
|
suffix: linux-amd64
|
||||||
|
# Linux ARM64
|
||||||
|
- goos: linux
|
||||||
|
goarch: arm64
|
||||||
|
suffix: linux-arm64
|
||||||
|
# macOS AMD64 (Intel)
|
||||||
|
- goos: darwin
|
||||||
|
goarch: amd64
|
||||||
|
suffix: darwin-amd64
|
||||||
|
# macOS ARM64 (Apple Silicon)
|
||||||
|
- goos: darwin
|
||||||
|
goarch: arm64
|
||||||
|
suffix: darwin-arm64
|
||||||
|
# Windows AMD64
|
||||||
|
- goos: windows
|
||||||
|
goarch: amd64
|
||||||
|
suffix: windows-amd64
|
||||||
|
# Windows ARM64
|
||||||
|
- goos: windows
|
||||||
|
goarch: arm64
|
||||||
|
suffix: windows-arm64
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Set up Go
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: '1.23'
|
||||||
|
|
||||||
|
- name: Cache Go modules
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ~/go/pkg/mod
|
||||||
|
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-go-
|
||||||
|
|
||||||
|
- name: Build binary
|
||||||
|
env:
|
||||||
|
GOOS: ${{ matrix.goos }}
|
||||||
|
GOARCH: ${{ matrix.goarch }}
|
||||||
|
CGO_ENABLED: 0
|
||||||
|
run: |
|
||||||
|
mkdir -p dist
|
||||||
|
VERSION=${GITHUB_REF#refs/tags/}
|
||||||
|
# Remove 'v' prefix if present for version injection
|
||||||
|
VERSION_CLEAN=${VERSION#v}
|
||||||
|
if [ "${{ matrix.goos }}" = "windows" ]; then
|
||||||
|
go build -ldflags="-s -w -X github.com/Gu1llaum-3/sshm/cmd.version=${VERSION_CLEAN}" -o dist/sshm-${{ matrix.suffix }}.exe .
|
||||||
|
else
|
||||||
|
go build -ldflags="-s -w -X github.com/Gu1llaum-3/sshm/cmd.version=${VERSION_CLEAN}" -o dist/sshm-${{ matrix.suffix }} .
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Create archive
|
||||||
|
run: |
|
||||||
|
cd dist
|
||||||
|
if [ "${{ matrix.goos }}" = "windows" ]; then
|
||||||
|
zip sshm-${{ matrix.suffix }}.zip sshm-${{ matrix.suffix }}.exe
|
||||||
|
rm sshm-${{ matrix.suffix }}.exe
|
||||||
|
else
|
||||||
|
tar -czf sshm-${{ matrix.suffix }}.tar.gz sshm-${{ matrix.suffix }}
|
||||||
|
rm sshm-${{ matrix.suffix }}
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Upload artifacts
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: sshm-${{ matrix.suffix }}
|
||||||
|
path: |
|
||||||
|
dist/sshm-${{ matrix.suffix }}.tar.gz
|
||||||
|
dist/sshm-${{ matrix.suffix }}.zip
|
||||||
|
|
||||||
|
release:
|
||||||
|
name: Create Release
|
||||||
|
needs: build
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Download all artifacts
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
path: ./artifacts
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Prepare release assets
|
||||||
|
run: |
|
||||||
|
mkdir -p release
|
||||||
|
find ./artifacts -name "*.tar.gz" -exec cp {} ./release/ \;
|
||||||
|
find ./artifacts -name "*.zip" -exec cp {} ./release/ \;
|
||||||
|
ls -la ./release/
|
||||||
|
|
||||||
|
- name: Check if pre-release
|
||||||
|
id: check_prerelease
|
||||||
|
run: |
|
||||||
|
if [[ "${GITHUB_REF#refs/tags/}" == *"-beta"* ]] || [[ "${GITHUB_REF#refs/tags/}" == *"-alpha"* ]] || [[ "${GITHUB_REF#refs/tags/}" == *"-rc"* ]] || [[ "${GITHUB_REF#refs/tags/}" == *"-dev"* ]]; then
|
||||||
|
echo "is_prerelease=true" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "is_prerelease=false" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Create Release
|
||||||
|
uses: softprops/action-gh-release@v2
|
||||||
|
with:
|
||||||
|
files: ./release/*
|
||||||
|
draft: false
|
||||||
|
prerelease: ${{ steps.check_prerelease.outputs.is_prerelease }}
|
||||||
|
generate_release_notes: true
|
||||||
|
name: ${{ github.ref_name }}${{ steps.check_prerelease.outputs.is_prerelease == 'true' && ' (Pre-release)' || '' }}
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
39
.github/workflows/release.yml
vendored
39
.github/workflows/release.yml
vendored
@ -1,39 +0,0 @@
|
|||||||
name: Release with GoReleaser
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- '*'
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
# Required for Homebrew tap updates
|
|
||||||
issues: write
|
|
||||||
pull-requests: write
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
goreleaser:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
# Fetch full history for changelog generation
|
|
||||||
fetch-depth: 0
|
|
||||||
|
|
||||||
- name: Set up Go
|
|
||||||
uses: actions/setup-go@v5
|
|
||||||
with:
|
|
||||||
go-version: '1.23'
|
|
||||||
cache: true
|
|
||||||
|
|
||||||
- name: Run GoReleaser
|
|
||||||
uses: goreleaser/goreleaser-action@v6
|
|
||||||
with:
|
|
||||||
distribution: goreleaser
|
|
||||||
version: '~> v2'
|
|
||||||
args: release --clean
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
# Token for updating Homebrew tap (create this secret in your repo settings)
|
|
||||||
HOMEBREW_TAP_GITHUB_TOKEN: ${{ secrets.HOMEBREW_TAP_GITHUB_TOKEN }}
|
|
155
.goreleaser.yaml
155
.goreleaser.yaml
@ -1,155 +0,0 @@
|
|||||||
version: 2
|
|
||||||
project_name: sshm
|
|
||||||
|
|
||||||
before:
|
|
||||||
hooks:
|
|
||||||
- go mod tidy
|
|
||||||
- go test ./...
|
|
||||||
|
|
||||||
builds:
|
|
||||||
- id: sshm
|
|
||||||
main: ./main.go
|
|
||||||
binary: sshm
|
|
||||||
goos:
|
|
||||||
- linux
|
|
||||||
- windows
|
|
||||||
- darwin
|
|
||||||
goarch:
|
|
||||||
- amd64
|
|
||||||
- arm64
|
|
||||||
- "386"
|
|
||||||
- arm
|
|
||||||
goarm:
|
|
||||||
- "6"
|
|
||||||
- "7"
|
|
||||||
ignore:
|
|
||||||
# Skip ARM for Windows (not commonly used)
|
|
||||||
- goos: windows
|
|
||||||
goarch: arm
|
|
||||||
- goos: windows
|
|
||||||
goarch: arm64
|
|
||||||
env:
|
|
||||||
- CGO_ENABLED=0
|
|
||||||
ldflags:
|
|
||||||
- -s -w
|
|
||||||
- -X github.com/Gu1llaum-3/sshm/cmd.AppVersion={{.Version}}
|
|
||||||
flags:
|
|
||||||
- -trimpath
|
|
||||||
|
|
||||||
archives:
|
|
||||||
- id: sshm
|
|
||||||
formats: [ "tar.gz" ]
|
|
||||||
# Use zip for Windows
|
|
||||||
format_overrides:
|
|
||||||
- goos: windows
|
|
||||||
formats: [ "zip" ]
|
|
||||||
# Template for archive name
|
|
||||||
name_template: >-
|
|
||||||
{{ .ProjectName }}_ {{- title .Os }}_ {{- if eq .Arch "amd64" }}x86_64 {{- else if eq .Arch "386" }}i386 {{- else }}{{ .Arch }}{{ end }} {{- if .Arm }}v{{ .Arm }}{{ end }}
|
|
||||||
files:
|
|
||||||
- LICENSE
|
|
||||||
- README.md
|
|
||||||
|
|
||||||
checksum:
|
|
||||||
name_template: "checksums.txt"
|
|
||||||
algorithm: sha256
|
|
||||||
|
|
||||||
changelog:
|
|
||||||
use: github
|
|
||||||
sort: asc
|
|
||||||
filters:
|
|
||||||
exclude:
|
|
||||||
- "^docs:"
|
|
||||||
- "^test:"
|
|
||||||
- "^ci:"
|
|
||||||
- "^chore:"
|
|
||||||
- "^build:"
|
|
||||||
groups:
|
|
||||||
- title: Features
|
|
||||||
regexp: '^.*?feat(\([[:word:]]+\))??!?:.+$'
|
|
||||||
order: 0
|
|
||||||
- title: Bug fixes
|
|
||||||
regexp: '^.*?fix(\([[:word:]]+\))??!?:.+$'
|
|
||||||
order: 1
|
|
||||||
- title: Others
|
|
||||||
order: 999
|
|
||||||
|
|
||||||
# Homebrew tap configuration (Formula pour CLI)
|
|
||||||
brews:
|
|
||||||
- name: sshm
|
|
||||||
repository:
|
|
||||||
owner: Gu1llaum-3
|
|
||||||
name: homebrew-sshm
|
|
||||||
# Token with repo permissions for your homebrew-sshm repo
|
|
||||||
token: "{{ .Env.HOMEBREW_TAP_GITHUB_TOKEN }}"
|
|
||||||
commit_author:
|
|
||||||
name: goreleaserbot
|
|
||||||
email: bot@goreleaser.com
|
|
||||||
commit_msg_template: "Brew formula update for {{ .ProjectName }} version {{ .Tag }}"
|
|
||||||
homepage: "https://github.com/Gu1llaum-3/sshm"
|
|
||||||
description: "A modern SSH connection manager for your terminal"
|
|
||||||
license: MIT
|
|
||||||
skip_upload: auto
|
|
||||||
# Test command to verify installation
|
|
||||||
test: |
|
|
||||||
system "#{bin}/sshm --version"
|
|
||||||
|
|
||||||
# Release configuration
|
|
||||||
release:
|
|
||||||
github:
|
|
||||||
owner: Gu1llaum-3
|
|
||||||
name: sshm
|
|
||||||
prerelease: auto
|
|
||||||
draft: false
|
|
||||||
replace_existing_draft: true
|
|
||||||
target_commitish: "{{ .Commit }}"
|
|
||||||
name_template: "{{.ProjectName}} {{.Version}}"
|
|
||||||
header: |
|
|
||||||
## SSHM {{.Version}}
|
|
||||||
|
|
||||||
Thank you for downloading SSHM!
|
|
||||||
|
|
||||||
footer: |
|
|
||||||
## Installation
|
|
||||||
|
|
||||||
### Homebrew (macOS/Linux)
|
|
||||||
```bash
|
|
||||||
brew tap Gu1llaum-3/sshm
|
|
||||||
brew install sshm
|
|
||||||
```
|
|
||||||
|
|
||||||
### Installation Script (Recommended)
|
|
||||||
**Unix/Linux/macOS:**
|
|
||||||
```bash
|
|
||||||
curl -sSL https://github.com/Gu1llaum-3/sshm/raw/main/install/unix.sh | bash
|
|
||||||
```
|
|
||||||
|
|
||||||
**Windows (PowerShell):**
|
|
||||||
```powershell
|
|
||||||
iwr -useb https://github.com/Gu1llaum-3/sshm/raw/main/install/windows.ps1 | iex
|
|
||||||
```
|
|
||||||
|
|
||||||
### Manual Installation
|
|
||||||
Download the appropriate binary for your platform from the assets above, extract it, and place it in your PATH.
|
|
||||||
|
|
||||||
## Full Changelog
|
|
||||||
|
|
||||||
See all changes at https://github.com/Gu1llaum-3/sshm/compare/{{.PreviousTag}}...{{.Tag}}
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
📖 **Documentation:** See the updated [README](https://github.com/Gu1llaum-3/sshm/blob/main/README.md)
|
|
||||||
|
|
||||||
🐛 **Issues:** Found a bug? Open an [issue](https://github.com/Gu1llaum-3/sshm/issues)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
Released with ❤️ by [GoReleaser](https://github.com/goreleaser/goreleaser)
|
|
||||||
|
|
||||||
# Snapshot builds (for non-tag builds)
|
|
||||||
snapshot:
|
|
||||||
version_template: "{{ .Tag }}-snapshot-{{.ShortCommit}}"
|
|
||||||
|
|
||||||
# Metadata for package managers
|
|
||||||
metadata:
|
|
||||||
mod_timestamp: "{{ .CommitTimestamp }}"
|
|
44
Makefile
44
Makefile
@ -1,44 +0,0 @@
|
|||||||
.PHONY: build build-local test clean release snapshot
|
|
||||||
|
|
||||||
# Version can be overridden via environment variable or command line
|
|
||||||
VERSION ?= dev
|
|
||||||
|
|
||||||
# Go build flags
|
|
||||||
LDFLAGS := -s -w -X github.com/Gu1llaum-3/sshm/cmd.AppVersion=$(VERSION)
|
|
||||||
|
|
||||||
# Build with specific version
|
|
||||||
build:
|
|
||||||
@mkdir -p dist
|
|
||||||
go build -ldflags="$(LDFLAGS)" -o dist/sshm .
|
|
||||||
|
|
||||||
# Build with git version
|
|
||||||
build-local: VERSION := $(shell git describe --tags --always --dirty 2>/dev/null || echo "dev")
|
|
||||||
build-local: build
|
|
||||||
|
|
||||||
# Run tests
|
|
||||||
test:
|
|
||||||
go test ./...
|
|
||||||
|
|
||||||
# Clean build artifacts
|
|
||||||
clean:
|
|
||||||
rm -rf dist
|
|
||||||
|
|
||||||
# Release with GoReleaser (requires tag)
|
|
||||||
release:
|
|
||||||
@if [ -z "$(shell git tag --points-at HEAD)" ]; then \
|
|
||||||
echo "Error: No git tag found at current commit. Create a tag first with: git tag vX.Y.Z"; \
|
|
||||||
exit 1; \
|
|
||||||
fi
|
|
||||||
goreleaser release --clean
|
|
||||||
|
|
||||||
# Build snapshot (without tag)
|
|
||||||
snapshot:
|
|
||||||
goreleaser release --snapshot --clean
|
|
||||||
|
|
||||||
# Check GoReleaser config
|
|
||||||
release-check:
|
|
||||||
goreleaser check
|
|
||||||
|
|
||||||
# Run GoReleaser in dry-run mode
|
|
||||||
release-dry-run:
|
|
||||||
goreleaser release --snapshot --skip=publish --clean
|
|
222
README.md
222
README.md
@ -25,39 +25,40 @@ SSHM is a beautiful command-line tool that transforms how you manage and connect
|
|||||||
|
|
||||||
## ✨ Features
|
## ✨ Features
|
||||||
|
|
||||||
### 🚀 **Core Capabilities**
|
### 🎯 **Core Features**
|
||||||
- **🎨 Beautiful TUI Interface** - Navigate your SSH hosts with an elegant, interactive terminal UI
|
- **🎨 Beautiful TUI Interface** - Navigate your SSH hosts with an elegant, interactive terminal UI
|
||||||
- **⚡ Quick Connect** - Connect to any host instantly through the TUI or the CLI with `sshm <host>`
|
- **⚡ Quick Connect** - Connect to any host instantly
|
||||||
- **🔄 Port Forwarding** - Easy setup for Local, Remote, and Dynamic (SOCKS) forwarding with history persistence
|
- **🔄 Port Forwarding** - Easy setup for Local, Remote, and Dynamic (SOCKS) forwarding
|
||||||
- **📝 Easy Management** - Add, edit, move, and manage SSH configurations seamlessly
|
- **📝Easy Management** - Add, edit, and manage SSH configurations seamlessly
|
||||||
- **🏷️ Tag Support** - Organize your hosts with custom tags for better categorization
|
- **🏷️ Tag Support** - Organize your hosts with custom tags for better categorization
|
||||||
- **🔍 Smart Search** - Find hosts quickly with built-in filtering and search
|
- **🔍 Smart Search** - Find hosts quickly with built-in filtering and search
|
||||||
- **📝 Real-time Status** - Live SSH connectivity indicators with asynchronous ping checks and color-coded status
|
|
||||||
- **🔔 Smart Updates** - Automatic version checking with update notifications
|
|
||||||
- **📈 Connection History** - Track your SSH connections with last login timestamps
|
|
||||||
|
|
||||||
### 🛠️ **Technical Features**
|
|
||||||
- **🔒 Secure** - Works directly with your existing `~/.ssh/config` file
|
- **🔒 Secure** - Works directly with your existing `~/.ssh/config` file
|
||||||
- **📁 Custom Config Support** - Use any SSH configuration file with the `-c` flag
|
- **📁 Custom Config Support** - Use any SSH configuration file with the `-c` flag
|
||||||
- **📂 SSH Include Support** - Full support for SSH Include directives to organize configurations across multiple files
|
- **📂 SSH Include Support** - Full support for SSH Include directives to organize configurations across multiple files
|
||||||
- **⚙️ SSH Options Support** - Add any SSH configuration option through intuitive forms
|
- **⚙️ SSH Options Support** - Add any SSH configuration option through intuitive forms
|
||||||
- **🔄 Automatic Conversion** - Seamlessly converts between command-line and config formats
|
- **🔄 Automatic Conversion** - Seamlessly converts between command-line and config formats
|
||||||
- **🔄 Automatic Backups** - Backup configurations automatically before changes
|
|
||||||
- **✅ Validation** - Prevent configuration errors with built-in validation
|
### 🛠️ **Management Operations**
|
||||||
- **🔗 ProxyJump Support** - Secure connection tunneling through bastion hosts
|
- **Add new SSH hosts** with interactive forms
|
||||||
- **⌨️ Keyboard Shortcuts** - Power user navigation with vim-like shortcuts
|
- **Edit existing configurations** in-place
|
||||||
- **🌐 Cross-platform** - Supports Linux, macOS (Intel & Apple Silicon), and Windows
|
- **Delete hosts** with confirmation prompts
|
||||||
- **⚡ Lightweight** - Single binary with no dependencies, zero configuration required
|
- **Port forwarding setup** with intuitive interface for Local (-L), Remote (-R), and Dynamic (-D) forwarding
|
||||||
|
- **Backup configurations** automatically before changes
|
||||||
|
- **Validate settings** to prevent configuration errors
|
||||||
|
- **ProxyJump support** for secure connection tunneling through bastion hosts
|
||||||
|
- **SSH Options management** - Add any SSH option with automatic format conversion
|
||||||
|
- **Full SSH compatibility** - Maintains compatibility with standard SSH tools
|
||||||
|
|
||||||
|
### 🎮 **User Experience**
|
||||||
|
- **Zero configuration** - Works out of the box with your existing SSH setup
|
||||||
|
- **Keyboard shortcuts** for power users
|
||||||
|
- **Cross-platform** - Supports Linux, macOS (Intel & Apple Silicon), and Windows
|
||||||
|
- **Lightweight** - Single binary with no dependencies
|
||||||
|
|
||||||
## 🚀 Quick Start
|
## 🚀 Quick Start
|
||||||
|
|
||||||
### Installation
|
### Installation
|
||||||
|
|
||||||
**Homebrew (Recommended for macOS):**
|
|
||||||
```bash
|
|
||||||
brew install Gu1llaum-3/sshm/sshm
|
|
||||||
```
|
|
||||||
|
|
||||||
**Unix/Linux/macOS (One-line install):**
|
**Unix/Linux/macOS (One-line install):**
|
||||||
```bash
|
```bash
|
||||||
curl -sSL https://raw.githubusercontent.com/Gu1llaum-3/sshm/main/install/unix.sh | bash
|
curl -sSL https://raw.githubusercontent.com/Gu1llaum-3/sshm/main/install/unix.sh | bash
|
||||||
@ -104,17 +105,10 @@ sshm
|
|||||||
- `a` - Add new host
|
- `a` - Add new host
|
||||||
- `e` - Edit selected host
|
- `e` - Edit selected host
|
||||||
- `d` - Delete selected host
|
- `d` - Delete selected host
|
||||||
- `m` - Move host to another config file (requires SSH Include directives)
|
|
||||||
- `f` - Port forwarding setup
|
- `f` - Port forwarding setup
|
||||||
- `q` - Quit
|
- `q` - Quit
|
||||||
- `/` - Search/filter hosts
|
- `/` - Search/filter hosts
|
||||||
|
|
||||||
**Real-time Status Indicators:**
|
|
||||||
- 🟢 **Online** - Host is reachable via SSH
|
|
||||||
- 🟡 **Connecting** - Currently checking host connectivity
|
|
||||||
- 🔴 **Offline** - Host is unreachable or SSH connection failed
|
|
||||||
- ⚫ **Unknown** - Connectivity status not yet determined
|
|
||||||
|
|
||||||
**Sorting & Filtering:**
|
**Sorting & Filtering:**
|
||||||
- `s` - Switch between sorting modes (name ↔ last login)
|
- `s` - Switch between sorting modes (name ↔ last login)
|
||||||
- `n` - Sort by **name** (alphabetical)
|
- `n` - Sort by **name** (alphabetical)
|
||||||
@ -164,7 +158,6 @@ SSHM provides an intuitive interface for setting up SSH port forwarding. Press `
|
|||||||
- Configure ports and addresses with guided forms
|
- Configure ports and addresses with guided forms
|
||||||
- Optional bind address configuration (defaults to 127.0.0.1)
|
- Optional bind address configuration (defaults to 127.0.0.1)
|
||||||
- Real-time validation of port numbers and addresses
|
- Real-time validation of port numbers and addresses
|
||||||
- **Port forwarding history** - Save frequently used configurations for quick reuse
|
|
||||||
- Connect automatically with configured forwarding options
|
- Connect automatically with configured forwarding options
|
||||||
|
|
||||||
**Troubleshooting Port Forwarding:**
|
**Troubleshooting Port Forwarding:**
|
||||||
@ -225,15 +218,9 @@ SSHM provides both command-line operations and an interactive TUI interface:
|
|||||||
# Launch interactive TUI mode for browsing and connecting to hosts
|
# Launch interactive TUI mode for browsing and connecting to hosts
|
||||||
sshm
|
sshm
|
||||||
|
|
||||||
# Connect directly to a specific host (with history tracking)
|
|
||||||
sshm my-server
|
|
||||||
|
|
||||||
# Launch TUI with custom SSH config file
|
# Launch TUI with custom SSH config file
|
||||||
sshm -c /path/to/custom/ssh_config
|
sshm -c /path/to/custom/ssh_config
|
||||||
|
|
||||||
# Connect directly with custom SSH config file
|
|
||||||
sshm my-server -c /path/to/custom/ssh_config
|
|
||||||
|
|
||||||
# Add a new host using interactive form
|
# Add a new host using interactive form
|
||||||
sshm add
|
sshm add
|
||||||
|
|
||||||
@ -249,42 +236,13 @@ sshm edit my-server
|
|||||||
# Edit host with custom SSH config file
|
# Edit host with custom SSH config file
|
||||||
sshm edit my-server -c /path/to/custom/ssh_config
|
sshm edit my-server -c /path/to/custom/ssh_config
|
||||||
|
|
||||||
# Move a host to another SSH config file (requires Include directives)
|
# Show version information
|
||||||
sshm move my-server
|
|
||||||
|
|
||||||
# Move host with custom SSH config file (requires Include directives)
|
|
||||||
sshm move my-server -c /path/to/custom/ssh_config
|
|
||||||
|
|
||||||
# Search for hosts (interactive filter)
|
|
||||||
sshm search
|
|
||||||
|
|
||||||
# Show version information (includes update check)
|
|
||||||
sshm --version
|
sshm --version
|
||||||
|
|
||||||
# Show help and available commands
|
# Show help and available commands
|
||||||
sshm --help
|
sshm --help
|
||||||
```
|
```
|
||||||
|
|
||||||
### Direct Host Connection
|
|
||||||
|
|
||||||
SSHM supports direct connection to hosts via the command line, making it easy to integrate into your existing workflow:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Connect directly to any configured host
|
|
||||||
sshm production-server
|
|
||||||
sshm db-staging
|
|
||||||
sshm web-01
|
|
||||||
|
|
||||||
# All direct connections are tracked in your history
|
|
||||||
# Use the TUI to see your most recently connected hosts
|
|
||||||
```
|
|
||||||
|
|
||||||
**Features of Direct Connection:**
|
|
||||||
- **Instant connection** - No TUI navigation required
|
|
||||||
- **History tracking** - All connections are recorded with timestamps
|
|
||||||
- **Error handling** - Clear messages if host doesn't exist or configuration issues
|
|
||||||
- **Config file support** - Works with custom config files using `-c` flag
|
|
||||||
|
|
||||||
### Backup Configuration
|
### Backup Configuration
|
||||||
|
|
||||||
SSHM automatically creates backups of your SSH configuration files before making any changes to ensure your configurations are safe.
|
SSHM automatically creates backups of your SSH configuration files before making any changes to ensure your configurations are safe.
|
||||||
@ -299,10 +257,6 @@ SSHM automatically creates backups of your SSH configuration files before making
|
|||||||
- Stored separately to avoid SSH Include conflicts
|
- Stored separately to avoid SSH Include conflicts
|
||||||
- Easy manual recovery if needed
|
- Easy manual recovery if needed
|
||||||
|
|
||||||
**Additional Storage:**
|
|
||||||
- **Connection History**: Stored in the same config directory for persistent tracking
|
|
||||||
- **Port Forwarding History**: Saved configurations for quick reuse of common forwarding setups
|
|
||||||
|
|
||||||
**Quick Recovery:**
|
**Quick Recovery:**
|
||||||
```bash
|
```bash
|
||||||
# Unix/Linux/macOS
|
# Unix/Linux/macOS
|
||||||
@ -323,96 +277,8 @@ sshm -c /path/to/custom/ssh_config
|
|||||||
# Use custom config file with commands
|
# Use custom config file with commands
|
||||||
sshm add hostname -c /path/to/custom/ssh_config
|
sshm add hostname -c /path/to/custom/ssh_config
|
||||||
sshm edit hostname -c /path/to/custom/ssh_config
|
sshm edit hostname -c /path/to/custom/ssh_config
|
||||||
sshm move hostname -c /path/to/custom/ssh_config
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Advanced Features
|
|
||||||
|
|
||||||
#### Host Movement Between Config Files
|
|
||||||
|
|
||||||
SSHM provides a powerful `move` command to relocate SSH hosts between different configuration files. **This feature requires SSH Include directives to be present in your SSH configuration.**
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Move a host to another config file (requires Include directives)
|
|
||||||
sshm move my-server
|
|
||||||
|
|
||||||
# Move with custom config file (requires Include directives)
|
|
||||||
sshm move my-server -c /path/to/custom/ssh_config
|
|
||||||
```
|
|
||||||
|
|
||||||
**⚠️ Important Requirements:**
|
|
||||||
- **SSH Include directives must be present** in your SSH config file (either `~/.ssh/config` or the file specified with `-c`)
|
|
||||||
- The config file must contain `Include` statements referencing other SSH configuration files
|
|
||||||
- Without Include directives, the move command will display an error message
|
|
||||||
|
|
||||||
**Features:**
|
|
||||||
- **Interactive file selector** - Choose destination config file from Include directives
|
|
||||||
- **Include support** - Works seamlessly with SSH Include directives structure
|
|
||||||
- **Atomic operations** - Safe host movement with automatic backups
|
|
||||||
- **Validation** - Prevents conflicts and ensures configuration integrity
|
|
||||||
- **Error handling** - Clear messages when Include files are needed but not found
|
|
||||||
|
|
||||||
**Use Cases:**
|
|
||||||
- Reorganize hosts from main config to specialized include files
|
|
||||||
- Move development hosts to separate environment-specific configs
|
|
||||||
- Consolidate configurations for better organization
|
|
||||||
|
|
||||||
**Example Setup Required:**
|
|
||||||
Your main SSH config file must contain Include directives like:
|
|
||||||
```ssh
|
|
||||||
# ~/.ssh/config
|
|
||||||
Include ~/.ssh/config.d/*
|
|
||||||
Include work-servers.conf
|
|
||||||
Include projects/*.conf
|
|
||||||
|
|
||||||
Host personal-server
|
|
||||||
HostName personal.example.com
|
|
||||||
User myuser
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Real-time Connectivity Status
|
|
||||||
|
|
||||||
SSHM features asynchronous SSH connectivity checking that provides visual indicators of host availability:
|
|
||||||
|
|
||||||
**Status Indicators:**
|
|
||||||
- 🟢 **Online** - SSH connection successful (shows response time)
|
|
||||||
- 🟡 **Connecting** - Currently testing connectivity
|
|
||||||
- 🔴 **Offline** - SSH connection failed or host unreachable
|
|
||||||
- ⚫ **Unknown** - Status not yet determined
|
|
||||||
|
|
||||||
**Features:**
|
|
||||||
- **Non-blocking checks** - Status updates happen in the background
|
|
||||||
- **Response time tracking** - See connection latency for online hosts
|
|
||||||
- **Automatic refresh** - Status indicators update continuously
|
|
||||||
- **Error details** - Detailed error information for failed connections
|
|
||||||
|
|
||||||
#### Automatic Update Checking
|
|
||||||
|
|
||||||
SSHM includes built-in version checking that notifies you of available updates:
|
|
||||||
|
|
||||||
**Features:**
|
|
||||||
- **Background checking** - Version check happens asynchronously
|
|
||||||
- **Release notifications** - Clear indicators when updates are available
|
|
||||||
- **Pre-release detection** - Identifies beta and development versions
|
|
||||||
- **GitHub integration** - Direct links to release pages
|
|
||||||
- **Non-intrusive** - Updates don't interrupt your workflow
|
|
||||||
|
|
||||||
**Update notifications appear:**
|
|
||||||
- In the main TUI interface as a subtle notification
|
|
||||||
- In the `sshm --version` command output
|
|
||||||
- Only when a newer stable version is available
|
|
||||||
|
|
||||||
#### Port Forwarding History
|
|
||||||
|
|
||||||
SSHM remembers your port forwarding configurations for easy reuse:
|
|
||||||
|
|
||||||
**Features:**
|
|
||||||
- **Automatic saving** - Successful forwarding setups are saved automatically
|
|
||||||
- **Quick reuse** - Previously used configurations appear as suggestions
|
|
||||||
- **Per-host history** - Forwarding history is tracked per SSH host
|
|
||||||
- **All forward types** - Supports Local (-L), Remote (-R), and Dynamic (-D) forwarding history
|
|
||||||
- **Persistent storage** - History survives application restarts
|
|
||||||
|
|
||||||
### Platform-Specific Notes
|
### Platform-Specific Notes
|
||||||
|
|
||||||
**Windows:**
|
**Windows:**
|
||||||
@ -553,34 +419,6 @@ This will be automatically converted to:
|
|||||||
StrictHostKeyChecking no
|
StrictHostKeyChecking no
|
||||||
```
|
```
|
||||||
|
|
||||||
### Custom Key Bindings
|
|
||||||
|
|
||||||
SSHM supports customizable key bindings through a configuration file. This is particularly useful for users who want to modify the default quit behavior.
|
|
||||||
|
|
||||||
**Configuration File Location:**
|
|
||||||
- **Linux/macOS**: `~/.config/sshm/config.json`
|
|
||||||
- **Windows**: `%APPDATA%\sshm\config.json`
|
|
||||||
|
|
||||||
**Example Configuration:**
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"key_bindings": {
|
|
||||||
"quit_keys": ["q", "ctrl+c"],
|
|
||||||
"disable_esc_quit": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
**Available Options:**
|
|
||||||
- **quit_keys**: Array of keys that will quit the application. Default: `["q", "ctrl+c"]`
|
|
||||||
- **disable_esc_quit**: Boolean flag to disable ESC key from quitting the application. Default: `false`
|
|
||||||
|
|
||||||
**For Vim Users:**
|
|
||||||
If you frequently press ESC accidentally causing the application to quit, set `disable_esc_quit` to `true`. This will disable ESC as a quit key while preserving all other functionality.
|
|
||||||
|
|
||||||
**Default Configuration:**
|
|
||||||
If no configuration file exists, SSHM will automatically create one with default settings that maintain backward compatibility.
|
|
||||||
|
|
||||||
## 🛠️ Development
|
## 🛠️ Development
|
||||||
|
|
||||||
### Prerequisites
|
### Prerequisites
|
||||||
@ -611,29 +449,20 @@ sshm/
|
|||||||
│ ├── root.go # Root command and interactive mode
|
│ ├── root.go # Root command and interactive mode
|
||||||
│ ├── add.go # Add host command
|
│ ├── add.go # Add host command
|
||||||
│ ├── edit.go # Edit host command
|
│ ├── edit.go # Edit host command
|
||||||
│ ├── move.go # Move host command
|
|
||||||
│ └── search.go # Search command
|
│ └── search.go # Search command
|
||||||
├── internal/
|
├── internal/
|
||||||
│ ├── config/ # SSH configuration management
|
│ ├── config/ # SSH configuration management
|
||||||
│ │ └── ssh.go # Config parsing and manipulation
|
│ │ └── ssh.go # Config parsing and manipulation
|
||||||
│ ├── connectivity/ # SSH connectivity checking
|
|
||||||
│ │ └── ping.go # Asynchronous SSH ping functionality
|
|
||||||
│ ├── history/ # Connection history tracking
|
│ ├── history/ # Connection history tracking
|
||||||
│ │ ├── history.go # History management and last login tracking
|
│ │ └── history.go # History management and last login tracking
|
||||||
│ │ └── port_forward_test.go # Port forwarding history tests
|
|
||||||
│ ├── version/ # Version checking and updates
|
|
||||||
│ │ ├── version.go # GitHub release checking and version comparison
|
|
||||||
│ │ └── version_test.go # Version parsing and comparison tests
|
|
||||||
│ ├── ui/ # Terminal UI components (Bubble Tea)
|
│ ├── ui/ # Terminal UI components (Bubble Tea)
|
||||||
│ │ ├── tui.go # Main TUI interface and program setup
|
│ │ ├── tui.go # Main TUI interface and program setup
|
||||||
│ │ ├── model.go # Core TUI model and state
|
│ │ ├── model.go # Core TUI model and state
|
||||||
│ │ ├── update.go # Message handling and state updates
|
│ │ ├── update.go # Message handling and state updates
|
||||||
│ │ ├── view.go # UI rendering and layout
|
│ │ ├── view.go # UI rendering and layout
|
||||||
│ │ ├── table.go # Host list table component with status indicators
|
│ │ ├── table.go # Host list table component
|
||||||
│ │ ├── add_form.go # Add host form interface
|
│ │ ├── add_form.go # Add host form interface
|
||||||
│ │ ├── edit_form.go# Edit host form interface
|
│ │ ├── edit_form.go# Edit host form interface
|
||||||
│ │ ├── move_form.go# Move host form interface
|
|
||||||
│ │ ├── port_forward_form.go # Port forwarding setup with history
|
|
||||||
│ │ ├── styles.go # Lip Gloss styling definitions
|
│ │ ├── styles.go # Lip Gloss styling definitions
|
||||||
│ │ ├── sort.go # Sorting and filtering logic
|
│ │ ├── sort.go # Sorting and filtering logic
|
||||||
│ │ └── utils.go # UI utility functions
|
│ │ └── utils.go # UI utility functions
|
||||||
@ -661,7 +490,6 @@ sshm/
|
|||||||
- [Bubble Tea](https://github.com/charmbracelet/bubbletea) - TUI framework
|
- [Bubble Tea](https://github.com/charmbracelet/bubbletea) - TUI framework
|
||||||
- [Bubbles](https://github.com/charmbracelet/bubbles) - TUI components
|
- [Bubbles](https://github.com/charmbracelet/bubbles) - TUI components
|
||||||
- [Lipgloss](https://github.com/charmbracelet/lipgloss) - Styling
|
- [Lipgloss](https://github.com/charmbracelet/lipgloss) - Styling
|
||||||
- [Go Crypto SSH](https://golang.org/x/crypto/ssh) - SSH connectivity checking
|
|
||||||
|
|
||||||
## 📦 Releases
|
## 📦 Releases
|
||||||
|
|
||||||
@ -697,8 +525,6 @@ This project is licensed under the MIT License - see the [LICENSE](LICENSE) file
|
|||||||
- [Charm](https://charm.sh/) for the amazing TUI libraries
|
- [Charm](https://charm.sh/) for the amazing TUI libraries
|
||||||
- [Cobra](https://cobra.dev/) for the excellent CLI framework
|
- [Cobra](https://cobra.dev/) for the excellent CLI framework
|
||||||
- [@yimeng](https://github.com/yimeng) for contributing SSH Include directive support
|
- [@yimeng](https://github.com/yimeng) for contributing SSH Include directive support
|
||||||
- [@ldreux](https://github.com/ldreux) for contributing multi-word search functionality
|
|
||||||
- [@qingfengzxr](https://github.com/qingfengzxr) for contributing custom key bindings support
|
|
||||||
- The Go community for building such fantastic tools
|
- The Go community for building such fantastic tools
|
||||||
|
|
||||||
---
|
---
|
||||||
|
31
cmd/root.go
31
cmd/root.go
@ -103,18 +103,27 @@ func runInteractiveMode() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func connectToHost(hostName string) {
|
func connectToHost(hostName string) {
|
||||||
// Quick check if host exists without full parsing (optimized for connection)
|
// Parse SSH configurations to verify host exists
|
||||||
var hostFound bool
|
var hosts []config.SSHHost
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
if configFile != "" {
|
if configFile != "" {
|
||||||
hostFound, err = config.QuickHostExistsInFile(hostName, configFile)
|
hosts, err = config.ParseSSHConfigFile(configFile)
|
||||||
} else {
|
} else {
|
||||||
hostFound, err = config.QuickHostExists(hostName)
|
hosts, err = config.ParseSSHConfig()
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Error checking SSH config: %v", err)
|
log.Fatalf("Error reading SSH config file: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if host exists
|
||||||
|
var hostFound bool
|
||||||
|
for _, host := range hosts {
|
||||||
|
if host.Name == hostName {
|
||||||
|
hostFound = true
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !hostFound {
|
if !hostFound {
|
||||||
@ -140,17 +149,11 @@ func connectToHost(hostName string) {
|
|||||||
fmt.Printf("Connecting to %s...\n", hostName)
|
fmt.Printf("Connecting to %s...\n", hostName)
|
||||||
|
|
||||||
var sshCmd *exec.Cmd
|
var sshCmd *exec.Cmd
|
||||||
var args []string
|
|
||||||
|
|
||||||
if configFile != "" {
|
if configFile != "" {
|
||||||
args = append(args, "-F", configFile)
|
sshCmd = exec.Command("ssh", "-F", configFile, hostName)
|
||||||
|
} else {
|
||||||
|
sshCmd = exec.Command("ssh", hostName)
|
||||||
}
|
}
|
||||||
args = append(args, hostName)
|
|
||||||
|
|
||||||
// Note: We don't add RemoteCommand here because if it's configured in SSH config,
|
|
||||||
// SSH will handle it automatically. Adding it as a command line argument would conflict.
|
|
||||||
|
|
||||||
sshCmd = exec.Command("ssh", args...)
|
|
||||||
|
|
||||||
// Set up the command to use the same stdin, stdout, and stderr as the parent process
|
// Set up the command to use the same stdin, stdout, and stderr as the parent process
|
||||||
sshCmd.Stdin = os.Stdin
|
sshCmd.Stdin = os.Stdin
|
||||||
|
@ -7,7 +7,6 @@ USE_SUDO="false"
|
|||||||
OS=""
|
OS=""
|
||||||
ARCH=""
|
ARCH=""
|
||||||
FORCE_INSTALL="${FORCE_INSTALL:-false}"
|
FORCE_INSTALL="${FORCE_INSTALL:-false}"
|
||||||
SSHM_VERSION="${SSHM_VERSION:-latest}"
|
|
||||||
|
|
||||||
RED='\033[0;31m'
|
RED='\033[0;31m'
|
||||||
PURPLE='\033[0;35m'
|
PURPLE='\033[0;35m'
|
||||||
@ -15,27 +14,13 @@ GREEN='\033[0;32m'
|
|||||||
YELLOW='\033[1;33m'
|
YELLOW='\033[1;33m'
|
||||||
NC='\033[0m'
|
NC='\033[0m'
|
||||||
|
|
||||||
usage() {
|
|
||||||
printf "${PURPLE}SSHM Installation Script${NC}\n\n"
|
|
||||||
printf "Usage:\n"
|
|
||||||
printf " Default (latest stable): ${GREEN}bash install.sh${NC}\n"
|
|
||||||
printf " Specific version: ${GREEN}SSHM_VERSION=v1.8.0 bash install.sh${NC}\n"
|
|
||||||
printf " Beta/pre-release: ${GREEN}SSHM_VERSION=v1.8.1-beta bash install.sh${NC}\n"
|
|
||||||
printf " Force install: ${GREEN}FORCE_INSTALL=true bash install.sh${NC}\n"
|
|
||||||
printf " Custom install directory: ${GREEN}INSTALL_DIR=/opt/bin bash install.sh${NC}\n\n"
|
|
||||||
printf "Environment variables:\n"
|
|
||||||
printf " SSHM_VERSION - Version to install (default: latest)\n"
|
|
||||||
printf " FORCE_INSTALL - Skip confirmation prompts (default: false)\n"
|
|
||||||
printf " INSTALL_DIR - Installation directory (default: /usr/local/bin)\n\n"
|
|
||||||
}
|
|
||||||
|
|
||||||
setSystem() {
|
setSystem() {
|
||||||
ARCH=$(uname -m)
|
ARCH=$(uname -m)
|
||||||
case $ARCH in
|
case $ARCH in
|
||||||
i386|i686) ARCH="amd64" ;;
|
i386|i686) ARCH="amd64" ;;
|
||||||
x86_64) ARCH="amd64";;
|
x86_64) ARCH="amd64";;
|
||||||
armv6*) ARCH="armv6" ;;
|
armv6*) ARCH="arm64" ;;
|
||||||
armv7*) ARCH="armv7" ;;
|
armv7*) ARCH="arm64" ;;
|
||||||
aarch64*) ARCH="arm64" ;;
|
aarch64*) ARCH="arm64" ;;
|
||||||
arm64) ARCH="arm64" ;;
|
arm64) ARCH="arm64" ;;
|
||||||
esac
|
esac
|
||||||
@ -61,48 +46,17 @@ runAsRoot() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
getLatestVersion() {
|
getLatestVersion() {
|
||||||
if [ "$SSHM_VERSION" = "latest" ]; then
|
printf "${YELLOW}Fetching latest version...${NC}\n"
|
||||||
printf "${YELLOW}Fetching latest stable version...${NC}\n"
|
LATEST_VERSION=$(curl -s https://api.github.com/repos/Gu1llaum-3/sshm/releases/latest | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/')
|
||||||
LATEST_VERSION=$(curl -s https://api.github.com/repos/Gu1llaum-3/sshm/releases/latest | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/')
|
if [ -z "$LATEST_VERSION" ]; then
|
||||||
if [ -z "$LATEST_VERSION" ]; then
|
printf "${RED}Failed to fetch latest version${NC}\n"
|
||||||
printf "${RED}Failed to fetch latest version${NC}\n"
|
exit 1
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
printf "${YELLOW}Using specified version: $SSHM_VERSION${NC}\n"
|
|
||||||
# Validate that the specified version exists
|
|
||||||
RELEASE_CHECK=$(curl -s "https://api.github.com/repos/Gu1llaum-3/sshm/releases/tags/$SSHM_VERSION" | grep '"tag_name":')
|
|
||||||
if [ -z "$RELEASE_CHECK" ]; then
|
|
||||||
printf "${RED}Version $SSHM_VERSION not found. Available versions:${NC}\n"
|
|
||||||
curl -s https://api.github.com/repos/Gu1llaum-3/sshm/releases | grep '"tag_name":' | head -10 | sed -E 's/.*"([^"]+)".*/ - \1/'
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
LATEST_VERSION="$SSHM_VERSION"
|
|
||||||
fi
|
fi
|
||||||
printf "${GREEN}Installing version: $LATEST_VERSION${NC}\n"
|
printf "${GREEN}Latest version: $LATEST_VERSION${NC}\n"
|
||||||
}
|
}
|
||||||
|
|
||||||
downloadBinary() {
|
downloadBinary() {
|
||||||
# Map OS names to match GoReleaser format
|
GITHUB_FILE="sshm-${OS}-${ARCH}.tar.gz"
|
||||||
local GORELEASER_OS="$OS"
|
|
||||||
case $OS in
|
|
||||||
"darwin") GORELEASER_OS="Darwin" ;;
|
|
||||||
"linux") GORELEASER_OS="Linux" ;;
|
|
||||||
"windows") GORELEASER_OS="Windows" ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# Map architecture names to match GoReleaser format
|
|
||||||
local GORELEASER_ARCH="$ARCH"
|
|
||||||
case $ARCH in
|
|
||||||
"amd64") GORELEASER_ARCH="x86_64" ;;
|
|
||||||
"arm64") GORELEASER_ARCH="arm64" ;;
|
|
||||||
"386") GORELEASER_ARCH="i386" ;;
|
|
||||||
"armv6") GORELEASER_ARCH="armv6" ;;
|
|
||||||
"armv7") GORELEASER_ARCH="armv7" ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# GoReleaser format: sshm_Linux_armv7.tar.gz
|
|
||||||
GITHUB_FILE="sshm_${GORELEASER_OS}_${GORELEASER_ARCH}.tar.gz"
|
|
||||||
GITHUB_URL="https://github.com/Gu1llaum-3/sshm/releases/download/$LATEST_VERSION/$GITHUB_FILE"
|
GITHUB_URL="https://github.com/Gu1llaum-3/sshm/releases/download/$LATEST_VERSION/$GITHUB_FILE"
|
||||||
|
|
||||||
printf "${YELLOW}Downloading $GITHUB_FILE...${NC}\n"
|
printf "${YELLOW}Downloading $GITHUB_FILE...${NC}\n"
|
||||||
@ -120,8 +74,8 @@ downloadBinary() {
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# GoReleaser extracts the binary as just "sshm", not with the platform suffix
|
# Check if the expected binary exists (no find needed)
|
||||||
EXTRACTED_BINARY="./sshm"
|
EXTRACTED_BINARY="./sshm-${OS}-${ARCH}"
|
||||||
if [ ! -f "$EXTRACTED_BINARY" ]; then
|
if [ ! -f "$EXTRACTED_BINARY" ]; then
|
||||||
printf "${RED}Could not find extracted binary: $EXTRACTED_BINARY${NC}\n"
|
printf "${RED}Could not find extracted binary: $EXTRACTED_BINARY${NC}\n"
|
||||||
exit 1
|
exit 1
|
||||||
@ -204,24 +158,18 @@ checkExisting() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
main() {
|
main() {
|
||||||
# Check for help argument
|
|
||||||
if [ "$1" = "-h" ] || [ "$1" = "--help" ] || [ "$1" = "help" ]; then
|
|
||||||
usage
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
printf "${PURPLE}Installing SSHM - SSH Connection Manager${NC}\n\n"
|
printf "${PURPLE}Installing SSHM - SSH Connection Manager${NC}\n\n"
|
||||||
|
|
||||||
|
# Check if already installed
|
||||||
|
checkExisting
|
||||||
|
|
||||||
# Set up system detection
|
# Set up system detection
|
||||||
setSystem
|
setSystem
|
||||||
printf "${GREEN}Detected system: $OS ($ARCH)${NC}\n"
|
printf "${GREEN}Detected system: $OS ($ARCH)${NC}\n"
|
||||||
|
|
||||||
# Get and validate version FIRST (this can fail early)
|
# Get latest version
|
||||||
getLatestVersion
|
getLatestVersion
|
||||||
|
|
||||||
# Check if already installed (this might prompt user)
|
|
||||||
checkExisting
|
|
||||||
|
|
||||||
# Download and install
|
# Download and install
|
||||||
downloadBinary
|
downloadBinary
|
||||||
install
|
install
|
||||||
|
@ -80,11 +80,7 @@ if ($LocalBinary -ne "") {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# Download binary
|
# Download binary
|
||||||
# Map architecture to match GoReleaser format
|
$fileName = "sshm-windows-$arch.zip"
|
||||||
$goreleaserArch = if ($arch -eq "amd64") { "x86_64" } else { "i386" }
|
|
||||||
|
|
||||||
# GoReleaser format: sshm_Windows_x86_64.zip
|
|
||||||
$fileName = "sshm_Windows_$goreleaserArch.zip"
|
|
||||||
$downloadUrl = "https://github.com/Gu1llaum-3/sshm/releases/download/$latestVersion/$fileName"
|
$downloadUrl = "https://github.com/Gu1llaum-3/sshm/releases/download/$latestVersion/$fileName"
|
||||||
$tempFile = "$env:TEMP\$fileName"
|
$tempFile = "$env:TEMP\$fileName"
|
||||||
|
|
||||||
@ -105,8 +101,7 @@ if ($LocalBinary -ne "") {
|
|||||||
Write-Info "Extracting..."
|
Write-Info "Extracting..."
|
||||||
try {
|
try {
|
||||||
Expand-Archive -Path $tempFile -DestinationPath $env:TEMP -Force
|
Expand-Archive -Path $tempFile -DestinationPath $env:TEMP -Force
|
||||||
# GoReleaser extracts the binary as just "sshm.exe", not with platform suffix
|
$extractedBinary = "$env:TEMP\sshm-windows-$arch.exe"
|
||||||
$extractedBinary = "$env:TEMP\sshm.exe"
|
|
||||||
$targetPath = "$InstallDir\sshm.exe"
|
$targetPath = "$InstallDir\sshm.exe"
|
||||||
|
|
||||||
Move-Item -Path $extractedBinary -Destination $targetPath -Force
|
Move-Item -Path $extractedBinary -Destination $targetPath -Force
|
||||||
|
@ -1,146 +0,0 @@
|
|||||||
package config
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
)
|
|
||||||
|
|
||||||
// KeyBindings represents configurable key bindings for the application
|
|
||||||
type KeyBindings struct {
|
|
||||||
// Quit keys - keys that will quit the application
|
|
||||||
QuitKeys []string `json:"quit_keys"`
|
|
||||||
|
|
||||||
// DisableEscQuit - if true, ESC key won't quit the application (useful for vim users)
|
|
||||||
DisableEscQuit bool `json:"disable_esc_quit"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// AppConfig represents the main application configuration
|
|
||||||
type AppConfig struct {
|
|
||||||
KeyBindings KeyBindings `json:"key_bindings"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetDefaultKeyBindings returns the default key bindings configuration
|
|
||||||
func GetDefaultKeyBindings() KeyBindings {
|
|
||||||
return KeyBindings{
|
|
||||||
QuitKeys: []string{"q", "ctrl+c"}, // Default keeps current behavior minus ESC
|
|
||||||
DisableEscQuit: false, // Default to false for backward compatibility
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetDefaultAppConfig returns the default application configuration
|
|
||||||
func GetDefaultAppConfig() AppConfig {
|
|
||||||
return AppConfig{
|
|
||||||
KeyBindings: GetDefaultKeyBindings(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetAppConfigPath returns the path to the application config file
|
|
||||||
func GetAppConfigPath() (string, error) {
|
|
||||||
configDir, err := GetSSHMConfigDir()
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
return filepath.Join(configDir, "config.json"), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoadAppConfig loads the application configuration from file
|
|
||||||
// If the file doesn't exist, it returns the default configuration
|
|
||||||
func LoadAppConfig() (*AppConfig, error) {
|
|
||||||
configPath, err := GetAppConfigPath()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// If config file doesn't exist, return default config and create the file
|
|
||||||
if _, err := os.Stat(configPath); os.IsNotExist(err) {
|
|
||||||
defaultConfig := GetDefaultAppConfig()
|
|
||||||
|
|
||||||
// Create config directory if it doesn't exist
|
|
||||||
configDir := filepath.Dir(configPath)
|
|
||||||
if err := os.MkdirAll(configDir, 0755); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save default config to file
|
|
||||||
if err := SaveAppConfig(&defaultConfig); err != nil {
|
|
||||||
// If we can't save, just return the default config without erroring
|
|
||||||
// This allows the app to work even if config file can't be created
|
|
||||||
return &defaultConfig, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return &defaultConfig, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read existing config file
|
|
||||||
data, err := os.ReadFile(configPath)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var config AppConfig
|
|
||||||
if err := json.Unmarshal(data, &config); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validate and fill in missing fields with defaults
|
|
||||||
config = mergeWithDefaults(config)
|
|
||||||
|
|
||||||
return &config, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// SaveAppConfig saves the application configuration to file
|
|
||||||
func SaveAppConfig(config *AppConfig) error {
|
|
||||||
if config == nil {
|
|
||||||
return errors.New("config cannot be nil")
|
|
||||||
}
|
|
||||||
|
|
||||||
configPath, err := GetAppConfigPath()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create config directory if it doesn't exist
|
|
||||||
configDir := filepath.Dir(configPath)
|
|
||||||
if err := os.MkdirAll(configDir, 0755); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
data, err := json.MarshalIndent(config, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return os.WriteFile(configPath, data, 0644)
|
|
||||||
}
|
|
||||||
|
|
||||||
// mergeWithDefaults ensures all required fields are set with defaults if missing
|
|
||||||
func mergeWithDefaults(config AppConfig) AppConfig {
|
|
||||||
defaults := GetDefaultAppConfig()
|
|
||||||
|
|
||||||
// If QuitKeys is empty, use defaults
|
|
||||||
if len(config.KeyBindings.QuitKeys) == 0 {
|
|
||||||
config.KeyBindings.QuitKeys = defaults.KeyBindings.QuitKeys
|
|
||||||
}
|
|
||||||
|
|
||||||
return config
|
|
||||||
}
|
|
||||||
|
|
||||||
// ShouldQuitOnKey checks if the given key should trigger quit based on configuration
|
|
||||||
func (kb *KeyBindings) ShouldQuitOnKey(key string) bool {
|
|
||||||
// Special handling for ESC key
|
|
||||||
if key == "esc" {
|
|
||||||
return !kb.DisableEscQuit
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if key is in the quit keys list
|
|
||||||
for _, quitKey := range kb.QuitKeys {
|
|
||||||
if quitKey == key {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
|
@ -1,181 +0,0 @@
|
|||||||
package config
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestDefaultKeyBindings(t *testing.T) {
|
|
||||||
kb := GetDefaultKeyBindings()
|
|
||||||
|
|
||||||
// Test default configuration
|
|
||||||
if kb.DisableEscQuit {
|
|
||||||
t.Error("Default configuration should allow ESC to quit (backward compatibility)")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test default quit keys
|
|
||||||
expectedQuitKeys := []string{"q", "ctrl+c"}
|
|
||||||
if len(kb.QuitKeys) != len(expectedQuitKeys) {
|
|
||||||
t.Errorf("Expected %d quit keys, got %d", len(expectedQuitKeys), len(kb.QuitKeys))
|
|
||||||
}
|
|
||||||
|
|
||||||
for i, expected := range expectedQuitKeys {
|
|
||||||
if i >= len(kb.QuitKeys) || kb.QuitKeys[i] != expected {
|
|
||||||
t.Errorf("Expected quit key %s, got %s", expected, kb.QuitKeys[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestShouldQuitOnKey(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
name string
|
|
||||||
keyBindings KeyBindings
|
|
||||||
key string
|
|
||||||
expectedResult bool
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "Default config - ESC should quit",
|
|
||||||
keyBindings: KeyBindings{
|
|
||||||
QuitKeys: []string{"q", "ctrl+c"},
|
|
||||||
DisableEscQuit: false,
|
|
||||||
},
|
|
||||||
key: "esc",
|
|
||||||
expectedResult: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Disabled ESC quit - ESC should not quit",
|
|
||||||
keyBindings: KeyBindings{
|
|
||||||
QuitKeys: []string{"q", "ctrl+c"},
|
|
||||||
DisableEscQuit: true,
|
|
||||||
},
|
|
||||||
key: "esc",
|
|
||||||
expectedResult: false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Q key should quit",
|
|
||||||
keyBindings: KeyBindings{
|
|
||||||
QuitKeys: []string{"q", "ctrl+c"},
|
|
||||||
DisableEscQuit: true,
|
|
||||||
},
|
|
||||||
key: "q",
|
|
||||||
expectedResult: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Ctrl+C should quit",
|
|
||||||
keyBindings: KeyBindings{
|
|
||||||
QuitKeys: []string{"q", "ctrl+c"},
|
|
||||||
DisableEscQuit: true,
|
|
||||||
},
|
|
||||||
key: "ctrl+c",
|
|
||||||
expectedResult: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Other keys should not quit",
|
|
||||||
keyBindings: KeyBindings{
|
|
||||||
QuitKeys: []string{"q", "ctrl+c"},
|
|
||||||
DisableEscQuit: true,
|
|
||||||
},
|
|
||||||
key: "enter",
|
|
||||||
expectedResult: false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tt := range tests {
|
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
|
||||||
result := tt.keyBindings.ShouldQuitOnKey(tt.key)
|
|
||||||
if result != tt.expectedResult {
|
|
||||||
t.Errorf("ShouldQuitOnKey(%q) = %v, expected %v", tt.key, result, tt.expectedResult)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAppConfigBasics(t *testing.T) {
|
|
||||||
// Test default config creation
|
|
||||||
defaultConfig := GetDefaultAppConfig()
|
|
||||||
|
|
||||||
if defaultConfig.KeyBindings.DisableEscQuit {
|
|
||||||
t.Error("Default configuration should allow ESC to quit")
|
|
||||||
}
|
|
||||||
|
|
||||||
expectedQuitKeys := []string{"q", "ctrl+c"}
|
|
||||||
if len(defaultConfig.KeyBindings.QuitKeys) != len(expectedQuitKeys) {
|
|
||||||
t.Errorf("Expected %d quit keys, got %d", len(expectedQuitKeys), len(defaultConfig.KeyBindings.QuitKeys))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMergeWithDefaults(t *testing.T) {
|
|
||||||
// Test config with missing QuitKeys
|
|
||||||
incompleteConfig := AppConfig{
|
|
||||||
KeyBindings: KeyBindings{
|
|
||||||
DisableEscQuit: true,
|
|
||||||
// QuitKeys is missing
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
mergedConfig := mergeWithDefaults(incompleteConfig)
|
|
||||||
|
|
||||||
// Should preserve DisableEscQuit
|
|
||||||
if !mergedConfig.KeyBindings.DisableEscQuit {
|
|
||||||
t.Error("Should preserve DisableEscQuit as true")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Should fill in default QuitKeys
|
|
||||||
expectedQuitKeys := []string{"q", "ctrl+c"}
|
|
||||||
if len(mergedConfig.KeyBindings.QuitKeys) != len(expectedQuitKeys) {
|
|
||||||
t.Errorf("Expected %d quit keys, got %d", len(expectedQuitKeys), len(mergedConfig.KeyBindings.QuitKeys))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSaveAndLoadAppConfigIntegration(t *testing.T) {
|
|
||||||
// Create a temporary directory for testing
|
|
||||||
tempDir, err := os.MkdirTemp("", "sshm_test")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create temp directory: %v", err)
|
|
||||||
}
|
|
||||||
defer os.RemoveAll(tempDir)
|
|
||||||
|
|
||||||
// Create a custom config file directly in temp directory
|
|
||||||
configPath := filepath.Join(tempDir, "config.json")
|
|
||||||
|
|
||||||
customConfig := AppConfig{
|
|
||||||
KeyBindings: KeyBindings{
|
|
||||||
QuitKeys: []string{"q"},
|
|
||||||
DisableEscQuit: true,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save config directly to file
|
|
||||||
data, err := json.MarshalIndent(customConfig, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to marshal config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
err = os.WriteFile(configPath, data, 0644)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to write config file: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read and unmarshal config
|
|
||||||
readData, err := os.ReadFile(configPath)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to read config file: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var loadedConfig AppConfig
|
|
||||||
err = json.Unmarshal(readData, &loadedConfig)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to unmarshal config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify the loaded config matches what we saved
|
|
||||||
if !loadedConfig.KeyBindings.DisableEscQuit {
|
|
||||||
t.Error("DisableEscQuit should be true")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(loadedConfig.KeyBindings.QuitKeys) != 1 || loadedConfig.KeyBindings.QuitKeys[0] != "q" {
|
|
||||||
t.Errorf("Expected quit keys to be ['q'], got %v", loadedConfig.KeyBindings.QuitKeys)
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
@ -987,710 +987,3 @@ func TestMoveHostToFile(t *testing.T) {
|
|||||||
// Test that the component functions work for the move operation
|
// Test that the component functions work for the move operation
|
||||||
t.Log("MoveHostToFile() error handling works correctly")
|
t.Log("MoveHostToFile() error handling works correctly")
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestParseSSHConfigWithMultipleHostsOnSameLine(t *testing.T) {
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
|
|
||||||
configFile := filepath.Join(tempDir, "config")
|
|
||||||
configContent := `# Test multiple hosts on same line
|
|
||||||
Host local1 local2
|
|
||||||
HostName ::1
|
|
||||||
User myuser
|
|
||||||
|
|
||||||
Host root-server
|
|
||||||
User root
|
|
||||||
HostName root.example.com
|
|
||||||
|
|
||||||
Host web1 web2 web3
|
|
||||||
HostName ::1
|
|
||||||
User webuser
|
|
||||||
Port 8080
|
|
||||||
|
|
||||||
Host single-host
|
|
||||||
HostName single.example.com
|
|
||||||
User singleuser
|
|
||||||
`
|
|
||||||
|
|
||||||
err := os.WriteFile(configFile, []byte(configContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
hosts, err := ParseSSHConfigFile(configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ParseSSHConfigFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Should get 7 hosts: local1, local2, root-server, web1, web2, web3, single-host
|
|
||||||
expectedHosts := map[string]struct{}{
|
|
||||||
"local1": {},
|
|
||||||
"local2": {},
|
|
||||||
"root-server": {},
|
|
||||||
"web1": {},
|
|
||||||
"web2": {},
|
|
||||||
"web3": {},
|
|
||||||
"single-host": {},
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hosts) != len(expectedHosts) {
|
|
||||||
t.Errorf("Expected %d hosts, got %d", len(expectedHosts), len(hosts))
|
|
||||||
for _, host := range hosts {
|
|
||||||
t.Logf("Found host: %s", host.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hostMap := make(map[string]SSHHost)
|
|
||||||
for _, host := range hosts {
|
|
||||||
hostMap[host.Name] = host
|
|
||||||
}
|
|
||||||
|
|
||||||
for expectedHostName := range expectedHosts {
|
|
||||||
if _, found := hostMap[expectedHostName]; !found {
|
|
||||||
t.Errorf("Expected host %s not found", expectedHostName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify properties based on host name
|
|
||||||
if host, found := hostMap["local1"]; found {
|
|
||||||
if host.Hostname != "::1" || host.User != "myuser" {
|
|
||||||
t.Errorf("local1 properties incorrect: hostname=%s, user=%s", host.Hostname, host.User)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["local2"]; found {
|
|
||||||
if host.Hostname != "::1" || host.User != "myuser" {
|
|
||||||
t.Errorf("local2 properties incorrect: hostname=%s, user=%s", host.Hostname, host.User)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["web1"]; found {
|
|
||||||
if host.Hostname != "::1" || host.User != "webuser" || host.Port != "8080" {
|
|
||||||
t.Errorf("web1 properties incorrect: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["web2"]; found {
|
|
||||||
if host.Hostname != "::1" || host.User != "webuser" || host.Port != "8080" {
|
|
||||||
t.Errorf("web2 properties incorrect: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["web3"]; found {
|
|
||||||
if host.Hostname != "::1" || host.User != "webuser" || host.Port != "8080" {
|
|
||||||
t.Errorf("web3 properties incorrect: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["root-server"]; found {
|
|
||||||
if host.User != "root" || host.Hostname != "root.example.com" {
|
|
||||||
t.Errorf("root-server properties incorrect: user=%s, hostname=%s", host.User, host.Hostname)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestUpdateSSHHostInFileWithMultiHost(t *testing.T) {
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
|
|
||||||
configFile := filepath.Join(tempDir, "config")
|
|
||||||
configContent := `# Test config with multi-host
|
|
||||||
Host web1 web2 web3
|
|
||||||
HostName webserver.example.com
|
|
||||||
User webuser
|
|
||||||
Port 2222
|
|
||||||
|
|
||||||
Host database
|
|
||||||
HostName db.example.com
|
|
||||||
User dbuser
|
|
||||||
`
|
|
||||||
|
|
||||||
err := os.WriteFile(configFile, []byte(configContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update web2 in the multi-host line
|
|
||||||
newHost := SSHHost{
|
|
||||||
Name: "web2-updated",
|
|
||||||
Hostname: "newweb.example.com",
|
|
||||||
User: "newuser",
|
|
||||||
Port: "22",
|
|
||||||
}
|
|
||||||
|
|
||||||
err = UpdateSSHHostInFile("web2", newHost, configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("UpdateSSHHostInFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse the updated config
|
|
||||||
hosts, err := ParseSSHConfigFile(configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ParseSSHConfigFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Should have: web1, web3, web2-updated, database
|
|
||||||
expectedHosts := []string{"web1", "web3", "web2-updated", "database"}
|
|
||||||
|
|
||||||
hostMap := make(map[string]SSHHost)
|
|
||||||
for _, host := range hosts {
|
|
||||||
hostMap[host.Name] = host
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hosts) != len(expectedHosts) {
|
|
||||||
t.Errorf("Expected %d hosts, got %d", len(expectedHosts), len(hosts))
|
|
||||||
for _, host := range hosts {
|
|
||||||
t.Logf("Found host: %s", host.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, expectedHostName := range expectedHosts {
|
|
||||||
if _, found := hostMap[expectedHostName]; !found {
|
|
||||||
t.Errorf("Expected host %s not found", expectedHostName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify web1 and web3 still have original properties
|
|
||||||
if host, found := hostMap["web1"]; found {
|
|
||||||
if host.Hostname != "webserver.example.com" || host.User != "webuser" || host.Port != "2222" {
|
|
||||||
t.Errorf("web1 properties changed: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["web3"]; found {
|
|
||||||
if host.Hostname != "webserver.example.com" || host.User != "webuser" || host.Port != "2222" {
|
|
||||||
t.Errorf("web3 properties changed: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify web2-updated has new properties
|
|
||||||
if host, found := hostMap["web2-updated"]; found {
|
|
||||||
if host.Hostname != "newweb.example.com" || host.User != "newuser" || host.Port != "22" {
|
|
||||||
t.Errorf("web2-updated properties incorrect: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify database is unchanged
|
|
||||||
if host, found := hostMap["database"]; found {
|
|
||||||
if host.Hostname != "db.example.com" || host.User != "dbuser" {
|
|
||||||
t.Errorf("database properties changed: hostname=%s, user=%s", host.Hostname, host.User)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestIsPartOfMultiHostDeclaration(t *testing.T) {
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
|
|
||||||
configFile := filepath.Join(tempDir, "config")
|
|
||||||
configContent := `Host single
|
|
||||||
HostName single.example.com
|
|
||||||
|
|
||||||
Host multi1 multi2 multi3
|
|
||||||
HostName multi.example.com
|
|
||||||
|
|
||||||
Host another
|
|
||||||
HostName another.example.com
|
|
||||||
`
|
|
||||||
|
|
||||||
err := os.WriteFile(configFile, []byte(configContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
tests := []struct {
|
|
||||||
hostName string
|
|
||||||
expectedMulti bool
|
|
||||||
expectedHosts []string
|
|
||||||
}{
|
|
||||||
{"single", false, []string{"single"}},
|
|
||||||
{"multi1", true, []string{"multi1", "multi2", "multi3"}},
|
|
||||||
{"multi2", true, []string{"multi1", "multi2", "multi3"}},
|
|
||||||
{"multi3", true, []string{"multi1", "multi2", "multi3"}},
|
|
||||||
{"another", false, []string{"another"}},
|
|
||||||
{"nonexistent", false, nil},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tt := range tests {
|
|
||||||
t.Run(tt.hostName, func(t *testing.T) {
|
|
||||||
isMulti, hostNames, err := IsPartOfMultiHostDeclaration(tt.hostName, configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("IsPartOfMultiHostDeclaration() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if isMulti != tt.expectedMulti {
|
|
||||||
t.Errorf("Expected isMulti=%v, got %v", tt.expectedMulti, isMulti)
|
|
||||||
}
|
|
||||||
|
|
||||||
if tt.expectedHosts == nil && hostNames != nil {
|
|
||||||
t.Errorf("Expected hostNames to be nil, got %v", hostNames)
|
|
||||||
} else if tt.expectedHosts != nil {
|
|
||||||
if len(hostNames) != len(tt.expectedHosts) {
|
|
||||||
t.Errorf("Expected %d hostNames, got %d", len(tt.expectedHosts), len(hostNames))
|
|
||||||
} else {
|
|
||||||
for i, expectedHost := range tt.expectedHosts {
|
|
||||||
if i < len(hostNames) && hostNames[i] != expectedHost {
|
|
||||||
t.Errorf("Expected hostNames[%d]=%s, got %s", i, expectedHost, hostNames[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestDeleteSSHHostFromFileWithMultiHost(t *testing.T) {
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
|
|
||||||
configFile := filepath.Join(tempDir, "config")
|
|
||||||
configContent := `# Test config with multi-host deletion
|
|
||||||
Host web1 web2 web3
|
|
||||||
HostName webserver.example.com
|
|
||||||
User webuser
|
|
||||||
Port 2222
|
|
||||||
|
|
||||||
Host database
|
|
||||||
HostName db.example.com
|
|
||||||
User dbuser
|
|
||||||
|
|
||||||
# Tags: production, critical
|
|
||||||
Host app1 app2
|
|
||||||
HostName appserver.example.com
|
|
||||||
User appuser
|
|
||||||
`
|
|
||||||
|
|
||||||
err := os.WriteFile(configFile, []byte(configContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test 1: Delete one host from multi-host block (should keep others)
|
|
||||||
err = DeleteSSHHostFromFile("web2", configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("DeleteSSHHostFromFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse the updated config
|
|
||||||
hosts, err := ParseSSHConfigFile(configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ParseSSHConfigFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Should have: web1, web3, database, app1, app2 (web2 removed)
|
|
||||||
expectedHosts := []string{"web1", "web3", "database", "app1", "app2"}
|
|
||||||
|
|
||||||
hostMap := make(map[string]SSHHost)
|
|
||||||
for _, host := range hosts {
|
|
||||||
hostMap[host.Name] = host
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hosts) != len(expectedHosts) {
|
|
||||||
t.Errorf("Expected %d hosts, got %d", len(expectedHosts), len(hosts))
|
|
||||||
for _, host := range hosts {
|
|
||||||
t.Logf("Found host: %s", host.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, expectedHostName := range expectedHosts {
|
|
||||||
if _, found := hostMap[expectedHostName]; !found {
|
|
||||||
t.Errorf("Expected host %s not found", expectedHostName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify web2 is not present
|
|
||||||
if _, found := hostMap["web2"]; found {
|
|
||||||
t.Error("web2 should have been deleted")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify web1 and web3 still have original properties
|
|
||||||
if host, found := hostMap["web1"]; found {
|
|
||||||
if host.Hostname != "webserver.example.com" || host.User != "webuser" || host.Port != "2222" {
|
|
||||||
t.Errorf("web1 properties incorrect: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if host, found := hostMap["web3"]; found {
|
|
||||||
if host.Hostname != "webserver.example.com" || host.User != "webuser" || host.Port != "2222" {
|
|
||||||
t.Errorf("web3 properties incorrect: hostname=%s, user=%s, port=%s", host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test 2: Delete one host from multi-host block with tags
|
|
||||||
err = DeleteSSHHostFromFile("app1", configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("DeleteSSHHostFromFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse again
|
|
||||||
hosts, err = ParseSSHConfigFile(configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ParseSSHConfigFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Should have: web1, web3, database, app2 (app1 removed)
|
|
||||||
expectedHosts = []string{"web1", "web3", "database", "app2"}
|
|
||||||
|
|
||||||
hostMap = make(map[string]SSHHost)
|
|
||||||
for _, host := range hosts {
|
|
||||||
hostMap[host.Name] = host
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hosts) != len(expectedHosts) {
|
|
||||||
t.Errorf("Expected %d hosts, got %d", len(expectedHosts), len(hosts))
|
|
||||||
for _, host := range hosts {
|
|
||||||
t.Logf("Found host: %s", host.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify app2 still has tags
|
|
||||||
if host, found := hostMap["app2"]; found {
|
|
||||||
if !contains(host.Tags, "production") || !contains(host.Tags, "critical") {
|
|
||||||
t.Errorf("app2 tags incorrect: %v", host.Tags)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestUpdateMultiHostBlock(t *testing.T) {
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
|
|
||||||
configFile := filepath.Join(tempDir, "config")
|
|
||||||
configContent := `# Test config for multi-host block update
|
|
||||||
Host server1 server2 server3
|
|
||||||
HostName cluster.example.com
|
|
||||||
User clusteruser
|
|
||||||
Port 2222
|
|
||||||
|
|
||||||
Host single
|
|
||||||
HostName single.example.com
|
|
||||||
User singleuser
|
|
||||||
`
|
|
||||||
|
|
||||||
err := os.WriteFile(configFile, []byte(configContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the multi-host block
|
|
||||||
originalHosts := []string{"server1", "server2", "server3"}
|
|
||||||
newHosts := []string{"server1", "server4", "server5"} // Remove server2, server3 and add server4, server5
|
|
||||||
commonProperties := SSHHost{
|
|
||||||
Hostname: "newcluster.example.com",
|
|
||||||
User: "newuser",
|
|
||||||
Port: "22",
|
|
||||||
Tags: []string{"updated", "cluster"},
|
|
||||||
}
|
|
||||||
|
|
||||||
err = UpdateMultiHostBlock(originalHosts, newHosts, commonProperties, configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("UpdateMultiHostBlock() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse the updated config
|
|
||||||
hosts, err := ParseSSHConfigFile(configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ParseSSHConfigFile() error = %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Should have: server1, server4, server5, single
|
|
||||||
expectedHosts := []string{"server1", "server4", "server5", "single"}
|
|
||||||
|
|
||||||
hostMap := make(map[string]SSHHost)
|
|
||||||
for _, host := range hosts {
|
|
||||||
hostMap[host.Name] = host
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hosts) != len(expectedHosts) {
|
|
||||||
t.Errorf("Expected %d hosts, got %d", len(expectedHosts), len(hosts))
|
|
||||||
for _, host := range hosts {
|
|
||||||
t.Logf("Found host: %s", host.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify new hosts have updated properties
|
|
||||||
for _, hostName := range []string{"server1", "server4", "server5"} {
|
|
||||||
if host, found := hostMap[hostName]; found {
|
|
||||||
if host.Hostname != "newcluster.example.com" || host.User != "newuser" || host.Port != "22" {
|
|
||||||
t.Errorf("%s properties incorrect: hostname=%s, user=%s, port=%s",
|
|
||||||
hostName, host.Hostname, host.User, host.Port)
|
|
||||||
}
|
|
||||||
if !contains(host.Tags, "updated") || !contains(host.Tags, "cluster") {
|
|
||||||
t.Errorf("%s tags incorrect: %v", hostName, host.Tags)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
t.Errorf("Expected host %s not found", hostName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify single host is unchanged
|
|
||||||
if host, found := hostMap["single"]; found {
|
|
||||||
if host.Hostname != "single.example.com" || host.User != "singleuser" {
|
|
||||||
t.Errorf("single host properties changed: hostname=%s, user=%s", host.Hostname, host.User)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify old hosts are gone
|
|
||||||
for _, oldHost := range []string{"server2", "server3"} {
|
|
||||||
if _, found := hostMap[oldHost]; found {
|
|
||||||
t.Errorf("Old host %s should have been removed", oldHost)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper function to check if slice contains a string
|
|
||||||
func contains(slice []string, item string) bool {
|
|
||||||
for _, s := range slice {
|
|
||||||
if s == item {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper function to create temporary config files for testing
|
|
||||||
func createTempConfigFile(content string) (string, error) {
|
|
||||||
tempFile, err := os.CreateTemp("", "ssh_config_test_*.conf")
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
defer tempFile.Close()
|
|
||||||
|
|
||||||
_, err = tempFile.WriteString(content)
|
|
||||||
if err != nil {
|
|
||||||
os.Remove(tempFile.Name())
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
return tempFile.Name(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFormatSSHConfigValue(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
name string
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "simple path without spaces",
|
|
||||||
input: "/home/user/.ssh/id_rsa",
|
|
||||||
expected: "/home/user/.ssh/id_rsa",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "path with spaces",
|
|
||||||
input: "/home/user/My Documents/ssh key",
|
|
||||||
expected: "\"/home/user/My Documents/ssh key\"",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Windows path with spaces",
|
|
||||||
input: `G:\My Drive\7 - Tech\9 - SSH Keys\Server_WF.opk`,
|
|
||||||
expected: `"G:\My Drive\7 - Tech\9 - SSH Keys\Server_WF.opk"`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "path with quotes but no spaces",
|
|
||||||
input: `/home/user/key"with"quotes`,
|
|
||||||
expected: `/home/user/key"with"quotes`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "path with spaces and quotes",
|
|
||||||
input: `/home/user/key "with" quotes`,
|
|
||||||
expected: `"/home/user/key "with" quotes"`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "empty path",
|
|
||||||
input: "",
|
|
||||||
expected: "",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "path with single space at end",
|
|
||||||
input: "/home/user/key ",
|
|
||||||
expected: "\"/home/user/key \"",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tt := range tests {
|
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
|
||||||
result := formatSSHConfigValue(tt.input)
|
|
||||||
if result != tt.expected {
|
|
||||||
t.Errorf("formatSSHConfigValue(%q) = %q, want %q", tt.input, result, tt.expected)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAddSSHHostWithSpacesInPath(t *testing.T) {
|
|
||||||
// Create temporary config file
|
|
||||||
configFile, err := createTempConfigFile(`Host existing
|
|
||||||
HostName existing.com
|
|
||||||
`)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config file: %v", err)
|
|
||||||
}
|
|
||||||
defer os.Remove(configFile)
|
|
||||||
|
|
||||||
// Test adding host with path containing spaces
|
|
||||||
host := SSHHost{
|
|
||||||
Name: "test-spaces",
|
|
||||||
Hostname: "test.com",
|
|
||||||
User: "testuser",
|
|
||||||
Identity: "/path/with spaces/key file",
|
|
||||||
}
|
|
||||||
|
|
||||||
err = AddSSHHostToFile(host, configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("AddSSHHostToFile failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read the file and verify quotes are added
|
|
||||||
content, err := os.ReadFile(configFile)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to read config file: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
contentStr := string(content)
|
|
||||||
expectedIdentityLine := ` IdentityFile "/path/with spaces/key file"`
|
|
||||||
if !strings.Contains(contentStr, expectedIdentityLine) {
|
|
||||||
t.Errorf("Expected identity file line with quotes not found.\nContent:\n%s\nExpected line: %s", contentStr, expectedIdentityLine)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestIsNonSSHConfigFile(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
fileName string
|
|
||||||
expected bool
|
|
||||||
}{
|
|
||||||
// Should be excluded
|
|
||||||
{"README", true},
|
|
||||||
{"README.txt", true},
|
|
||||||
{"README.md", true},
|
|
||||||
{"script.sh", true},
|
|
||||||
{"data.json", true},
|
|
||||||
{"notes.txt", true},
|
|
||||||
{".gitignore", true},
|
|
||||||
{"backup.bak", true},
|
|
||||||
{"old.orig", true},
|
|
||||||
{"log.log", true},
|
|
||||||
{"temp.tmp", true},
|
|
||||||
{"archive.zip", true},
|
|
||||||
{"image.jpg", true},
|
|
||||||
{"python.py", true},
|
|
||||||
{"golang.go", true},
|
|
||||||
{"config.yaml", true},
|
|
||||||
{"config.yml", true},
|
|
||||||
{"config.toml", true},
|
|
||||||
|
|
||||||
// Should NOT be excluded (valid SSH config files)
|
|
||||||
{"config", false},
|
|
||||||
{"servers.conf", false},
|
|
||||||
{"production", false},
|
|
||||||
{"staging", false},
|
|
||||||
{"hosts", false},
|
|
||||||
{"ssh_config", false},
|
|
||||||
{"work-servers", false},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, test := range tests {
|
|
||||||
// Create a temporary file for content testing
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
filePath := filepath.Join(tempDir, test.fileName)
|
|
||||||
|
|
||||||
// Write appropriate content based on expected result
|
|
||||||
var content string
|
|
||||||
if test.expected {
|
|
||||||
// Write non-SSH content for files that should be excluded
|
|
||||||
content = "# This is not an SSH config file\nSome random content"
|
|
||||||
} else {
|
|
||||||
// Write SSH-like content for files that should be included
|
|
||||||
content = "Host example\n HostName example.com\n User testuser"
|
|
||||||
}
|
|
||||||
|
|
||||||
err := os.WriteFile(filePath, []byte(content), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create test file %s: %v", test.fileName, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
result := isNonSSHConfigFile(filePath)
|
|
||||||
if result != test.expected {
|
|
||||||
t.Errorf("isNonSSHConfigFile(%q) = %v, want %v", test.fileName, result, test.expected)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestQuickHostExists(t *testing.T) {
|
|
||||||
// Create temporary directory for test files
|
|
||||||
tempDir := t.TempDir()
|
|
||||||
|
|
||||||
// Create main config file
|
|
||||||
mainConfig := filepath.Join(tempDir, "config")
|
|
||||||
mainConfigContent := `Host main-host
|
|
||||||
HostName example.com
|
|
||||||
|
|
||||||
Include config.d/*
|
|
||||||
|
|
||||||
Host another-host
|
|
||||||
HostName another.example.com
|
|
||||||
`
|
|
||||||
|
|
||||||
err := os.WriteFile(mainConfig, []byte(mainConfigContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create main config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create config.d directory
|
|
||||||
configDir := filepath.Join(tempDir, "config.d")
|
|
||||||
err = os.MkdirAll(configDir, 0700)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create config.d: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create valid SSH config file in config.d
|
|
||||||
validConfig := filepath.Join(configDir, "servers.conf")
|
|
||||||
validConfigContent := `Host included-host
|
|
||||||
HostName included.example.com
|
|
||||||
User includeduser
|
|
||||||
|
|
||||||
Host production-server
|
|
||||||
HostName prod.example.com
|
|
||||||
User produser
|
|
||||||
`
|
|
||||||
|
|
||||||
err = os.WriteFile(validConfig, []byte(validConfigContent), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create valid config: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create files that should be excluded (README, etc.)
|
|
||||||
excludedFiles := map[string]string{
|
|
||||||
"README": "# This is a README file\nDocumentation goes here",
|
|
||||||
"README.md": "# SSH Configuration\nThis directory contains...",
|
|
||||||
"script.sh": "#!/bin/bash\necho 'hello world'",
|
|
||||||
"data.json": `{"key": "value"}`,
|
|
||||||
}
|
|
||||||
|
|
||||||
for fileName, content := range excludedFiles {
|
|
||||||
filePath := filepath.Join(configDir, fileName)
|
|
||||||
err = os.WriteFile(filePath, []byte(content), 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Failed to create %s: %v", fileName, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test hosts that should be found
|
|
||||||
existingHosts := []string{"main-host", "another-host", "included-host", "production-server"}
|
|
||||||
for _, hostName := range existingHosts {
|
|
||||||
found, err := QuickHostExistsInFile(hostName, mainConfig)
|
|
||||||
if err != nil {
|
|
||||||
t.Errorf("QuickHostExistsInFile(%q) error = %v", hostName, err)
|
|
||||||
}
|
|
||||||
if !found {
|
|
||||||
t.Errorf("QuickHostExistsInFile(%q) = false, want true", hostName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test hosts that should NOT be found
|
|
||||||
nonExistingHosts := []string{"nonexistent-host", "fake-server", "unknown"}
|
|
||||||
for _, hostName := range nonExistingHosts {
|
|
||||||
found, err := QuickHostExistsInFile(hostName, mainConfig)
|
|
||||||
if err != nil {
|
|
||||||
t.Errorf("QuickHostExistsInFile(%q) error = %v", hostName, err)
|
|
||||||
}
|
|
||||||
if found {
|
|
||||||
t.Errorf("QuickHostExistsInFile(%q) = true, want false", hostName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -1,14 +1,12 @@
|
|||||||
package ui
|
package ui
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"os"
|
"os"
|
||||||
"os/user"
|
"os/user"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/Gu1llaum-3/sshm/internal/config"
|
"github.com/Gu1llaum-3/sshm/internal/config"
|
||||||
"github.com/Gu1llaum-3/sshm/internal/validation"
|
"github.com/Gu1llaum-3/sshm/internal/validation"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/charmbracelet/bubbles/textinput"
|
"github.com/charmbracelet/bubbles/textinput"
|
||||||
tea "github.com/charmbracelet/bubbletea"
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
@ -17,7 +15,6 @@ import (
|
|||||||
type addFormModel struct {
|
type addFormModel struct {
|
||||||
inputs []textinput.Model
|
inputs []textinput.Model
|
||||||
focused int
|
focused int
|
||||||
currentTab int // 0 = General, 1 = Advanced
|
|
||||||
err string
|
err string
|
||||||
styles Styles
|
styles Styles
|
||||||
success bool
|
success bool
|
||||||
@ -49,7 +46,7 @@ func NewAddForm(hostname string, styles Styles, width, height int, configFile st
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inputs := make([]textinput.Model, 10) // Increased from 9 to 10 for RequestTTY
|
inputs := make([]textinput.Model, 8)
|
||||||
|
|
||||||
// Name input
|
// Name input
|
||||||
inputs[nameInput] = textinput.New()
|
inputs[nameInput] = textinput.New()
|
||||||
@ -103,22 +100,9 @@ func NewAddForm(hostname string, styles Styles, width, height int, configFile st
|
|||||||
inputs[tagsInput].CharLimit = 200
|
inputs[tagsInput].CharLimit = 200
|
||||||
inputs[tagsInput].Width = 50
|
inputs[tagsInput].Width = 50
|
||||||
|
|
||||||
// Remote Command input
|
|
||||||
inputs[remoteCommandInput] = textinput.New()
|
|
||||||
inputs[remoteCommandInput].Placeholder = "ls -la, htop, bash"
|
|
||||||
inputs[remoteCommandInput].CharLimit = 300
|
|
||||||
inputs[remoteCommandInput].Width = 70
|
|
||||||
|
|
||||||
// RequestTTY input
|
|
||||||
inputs[requestTTYInput] = textinput.New()
|
|
||||||
inputs[requestTTYInput].Placeholder = "yes, no, force, auto"
|
|
||||||
inputs[requestTTYInput].CharLimit = 10
|
|
||||||
inputs[requestTTYInput].Width = 30
|
|
||||||
|
|
||||||
return &addFormModel{
|
return &addFormModel{
|
||||||
inputs: inputs,
|
inputs: inputs,
|
||||||
focused: nameInput,
|
focused: nameInput,
|
||||||
currentTab: tabGeneral, // Start on General tab
|
|
||||||
styles: styles,
|
styles: styles,
|
||||||
width: width,
|
width: width,
|
||||||
height: height,
|
height: height,
|
||||||
@ -126,11 +110,6 @@ func NewAddForm(hostname string, styles Styles, width, height int, configFile st
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
|
||||||
tabGeneral = iota
|
|
||||||
tabAdvanced
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
nameInput = iota
|
nameInput = iota
|
||||||
hostnameInput
|
hostnameInput
|
||||||
@ -138,11 +117,8 @@ const (
|
|||||||
portInput
|
portInput
|
||||||
identityInput
|
identityInput
|
||||||
proxyJumpInput
|
proxyJumpInput
|
||||||
tagsInput
|
|
||||||
// Advanced tab inputs
|
|
||||||
optionsInput
|
optionsInput
|
||||||
remoteCommandInput
|
tagsInput
|
||||||
requestTTYInput
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Messages for communication with parent model
|
// Messages for communication with parent model
|
||||||
@ -172,24 +148,40 @@ func (m *addFormModel) Update(msg tea.Msg) (*addFormModel, tea.Cmd) {
|
|||||||
case "ctrl+c", "esc":
|
case "ctrl+c", "esc":
|
||||||
return m, func() tea.Msg { return addFormCancelMsg{} }
|
return m, func() tea.Msg { return addFormCancelMsg{} }
|
||||||
|
|
||||||
case "ctrl+s":
|
case "ctrl+enter":
|
||||||
// Allow submission from any field with Ctrl+S (Save)
|
// Allow submission from any field with Ctrl+Enter
|
||||||
return m, m.submitForm()
|
return m, m.submitForm()
|
||||||
|
|
||||||
case "ctrl+j":
|
|
||||||
// Switch to next tab
|
|
||||||
m.currentTab = (m.currentTab + 1) % 2
|
|
||||||
m.focused = m.getFirstInputForTab(m.currentTab)
|
|
||||||
return m, m.updateFocus()
|
|
||||||
|
|
||||||
case "ctrl+k":
|
|
||||||
// Switch to previous tab
|
|
||||||
m.currentTab = (m.currentTab - 1 + 2) % 2
|
|
||||||
m.focused = m.getFirstInputForTab(m.currentTab)
|
|
||||||
return m, m.updateFocus()
|
|
||||||
|
|
||||||
case "tab", "shift+tab", "enter", "up", "down":
|
case "tab", "shift+tab", "enter", "up", "down":
|
||||||
return m, m.handleNavigation(msg.String())
|
s := msg.String()
|
||||||
|
|
||||||
|
// Handle form submission
|
||||||
|
if s == "enter" && m.focused == len(m.inputs)-1 {
|
||||||
|
return m, m.submitForm()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cycle inputs
|
||||||
|
if s == "up" || s == "shift+tab" {
|
||||||
|
m.focused--
|
||||||
|
} else {
|
||||||
|
m.focused++
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.focused > len(m.inputs)-1 {
|
||||||
|
m.focused = 0
|
||||||
|
} else if m.focused < 0 {
|
||||||
|
m.focused = len(m.inputs) - 1
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range m.inputs {
|
||||||
|
if i == m.focused {
|
||||||
|
cmds = append(cmds, m.inputs[i].Focus())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
m.inputs[i].Blur()
|
||||||
|
}
|
||||||
|
|
||||||
|
return m, tea.Batch(cmds...)
|
||||||
}
|
}
|
||||||
|
|
||||||
case addFormSubmitMsg:
|
case addFormSubmitMsg:
|
||||||
@ -213,104 +205,32 @@ func (m *addFormModel) Update(msg tea.Msg) (*addFormModel, tea.Cmd) {
|
|||||||
return m, tea.Batch(cmds...)
|
return m, tea.Batch(cmds...)
|
||||||
}
|
}
|
||||||
|
|
||||||
// getFirstInputForTab returns the first input index for a given tab
|
|
||||||
func (m *addFormModel) getFirstInputForTab(tab int) int {
|
|
||||||
switch tab {
|
|
||||||
case tabGeneral:
|
|
||||||
return nameInput
|
|
||||||
case tabAdvanced:
|
|
||||||
return optionsInput
|
|
||||||
default:
|
|
||||||
return nameInput
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// getInputsForCurrentTab returns the input indices for the current tab
|
|
||||||
func (m *addFormModel) getInputsForCurrentTab() []int {
|
|
||||||
switch m.currentTab {
|
|
||||||
case tabGeneral:
|
|
||||||
return []int{nameInput, hostnameInput, userInput, portInput, identityInput, proxyJumpInput, tagsInput}
|
|
||||||
case tabAdvanced:
|
|
||||||
return []int{optionsInput, remoteCommandInput, requestTTYInput}
|
|
||||||
default:
|
|
||||||
return []int{nameInput, hostnameInput, userInput, portInput, identityInput, proxyJumpInput, tagsInput}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// updateFocus updates focus for inputs
|
|
||||||
func (m *addFormModel) updateFocus() tea.Cmd {
|
|
||||||
var cmds []tea.Cmd
|
|
||||||
for i := range m.inputs {
|
|
||||||
if i == m.focused {
|
|
||||||
cmds = append(cmds, m.inputs[i].Focus())
|
|
||||||
} else {
|
|
||||||
m.inputs[i].Blur()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return tea.Batch(cmds...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// handleNavigation handles tab/arrow navigation within the current tab
|
|
||||||
func (m *addFormModel) handleNavigation(key string) tea.Cmd {
|
|
||||||
currentTabInputs := m.getInputsForCurrentTab()
|
|
||||||
|
|
||||||
// Find current position within the tab
|
|
||||||
currentPos := 0
|
|
||||||
for i, input := range currentTabInputs {
|
|
||||||
if input == m.focused {
|
|
||||||
currentPos = i
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle form submission on last field of Advanced tab
|
|
||||||
if key == "enter" && m.currentTab == tabAdvanced && currentPos == len(currentTabInputs)-1 {
|
|
||||||
return m.submitForm()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Navigate within current tab
|
|
||||||
if key == "up" || key == "shift+tab" {
|
|
||||||
currentPos--
|
|
||||||
} else {
|
|
||||||
currentPos++
|
|
||||||
}
|
|
||||||
|
|
||||||
// Wrap around within current tab
|
|
||||||
if currentPos >= len(currentTabInputs) {
|
|
||||||
currentPos = 0
|
|
||||||
} else if currentPos < 0 {
|
|
||||||
currentPos = len(currentTabInputs) - 1
|
|
||||||
}
|
|
||||||
|
|
||||||
m.focused = currentTabInputs[currentPos]
|
|
||||||
return m.updateFocus()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *addFormModel) View() string {
|
func (m *addFormModel) View() string {
|
||||||
if m.success {
|
if m.success {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if terminal height is sufficient
|
|
||||||
if !m.isHeightSufficient() {
|
|
||||||
return m.renderHeightWarning()
|
|
||||||
}
|
|
||||||
|
|
||||||
var b strings.Builder
|
var b strings.Builder
|
||||||
|
|
||||||
b.WriteString(m.styles.FormTitle.Render("Add SSH Host Configuration"))
|
b.WriteString(m.styles.FormTitle.Render("Add SSH Host Configuration"))
|
||||||
b.WriteString("\n\n")
|
b.WriteString("\n\n")
|
||||||
|
|
||||||
// Render tabs
|
fields := []string{
|
||||||
b.WriteString(m.renderTabs())
|
"Host Name *",
|
||||||
b.WriteString("\n\n")
|
"Hostname/IP *",
|
||||||
|
"User",
|
||||||
|
"Port",
|
||||||
|
"Identity File",
|
||||||
|
"ProxyJump",
|
||||||
|
"SSH Options",
|
||||||
|
"Tags (comma-separated)",
|
||||||
|
}
|
||||||
|
|
||||||
// Render current tab content
|
for i, field := range fields {
|
||||||
switch m.currentTab {
|
b.WriteString(m.styles.FormField.Render(field))
|
||||||
case tabGeneral:
|
b.WriteString("\n")
|
||||||
b.WriteString(m.renderGeneralTab())
|
b.WriteString(m.inputs[i].View())
|
||||||
case tabAdvanced:
|
b.WriteString("\n\n")
|
||||||
b.WriteString(m.renderAdvancedTab())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if m.err != "" {
|
if m.err != "" {
|
||||||
@ -318,133 +238,13 @@ func (m *addFormModel) View() string {
|
|||||||
b.WriteString("\n\n")
|
b.WriteString("\n\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Help text
|
b.WriteString(m.styles.FormHelp.Render("Tab/Shift+Tab: navigate • Enter on last field: submit • Ctrl+Enter: submit • Ctrl+C/Esc: cancel"))
|
||||||
b.WriteString(m.styles.FormHelp.Render("Tab/Shift+Tab: navigate • Ctrl+J/K: switch tabs"))
|
|
||||||
b.WriteString("\n")
|
|
||||||
b.WriteString(m.styles.FormHelp.Render("Enter on last field: submit • Ctrl+S: save • Ctrl+C/Esc: cancel"))
|
|
||||||
b.WriteString("\n")
|
b.WriteString("\n")
|
||||||
b.WriteString(m.styles.FormHelp.Render("* Required fields"))
|
b.WriteString(m.styles.FormHelp.Render("* Required fields"))
|
||||||
|
|
||||||
return b.String()
|
return b.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
// getMinimumHeight calculates the minimum height needed to display the form
|
|
||||||
func (m *addFormModel) getMinimumHeight() int {
|
|
||||||
// Title: 1 line + 2 newlines = 3
|
|
||||||
titleLines := 3
|
|
||||||
// Tabs: 1 line + 2 newlines = 3
|
|
||||||
tabLines := 3
|
|
||||||
// Fields in current tab
|
|
||||||
var fieldsCount int
|
|
||||||
if m.currentTab == tabGeneral {
|
|
||||||
fieldsCount = 7 // 7 fields in general tab
|
|
||||||
} else {
|
|
||||||
fieldsCount = 3 // 3 fields in advanced tab
|
|
||||||
}
|
|
||||||
// Each field: label (1) + input (1) + spacing (2) = 4 lines per field, but let's be more conservative
|
|
||||||
fieldsLines := fieldsCount * 3 // Reduced from 4 to 3
|
|
||||||
// Help text: 3 lines
|
|
||||||
helpLines := 3
|
|
||||||
// Error message space when needed: 2 lines
|
|
||||||
errorLines := 0 // Only count when there's actually an error
|
|
||||||
if m.err != "" {
|
|
||||||
errorLines = 2
|
|
||||||
}
|
|
||||||
|
|
||||||
return titleLines + tabLines + fieldsLines + helpLines + errorLines + 1 // +1 minimal safety margin
|
|
||||||
}
|
|
||||||
|
|
||||||
// isHeightSufficient checks if the current terminal height is sufficient
|
|
||||||
func (m *addFormModel) isHeightSufficient() bool {
|
|
||||||
return m.height >= m.getMinimumHeight()
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderHeightWarning renders a warning message when height is insufficient
|
|
||||||
func (m *addFormModel) renderHeightWarning() string {
|
|
||||||
required := m.getMinimumHeight()
|
|
||||||
current := m.height
|
|
||||||
|
|
||||||
warning := m.styles.ErrorText.Render("⚠️ Terminal height is too small!")
|
|
||||||
details := m.styles.FormField.Render(fmt.Sprintf("Current: %d lines, Required: %d lines", current, required))
|
|
||||||
instruction := m.styles.FormHelp.Render("Please resize your terminal window and try again.")
|
|
||||||
instruction2 := m.styles.FormHelp.Render("Press Ctrl+C to cancel or resize terminal window.")
|
|
||||||
|
|
||||||
return warning + "\n\n" + details + "\n\n" + instruction + "\n" + instruction2
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderTabs renders the tab headers
|
|
||||||
func (m *addFormModel) renderTabs() string {
|
|
||||||
var generalTab, advancedTab string
|
|
||||||
|
|
||||||
if m.currentTab == tabGeneral {
|
|
||||||
generalTab = m.styles.FocusedLabel.Render("[ General ]")
|
|
||||||
advancedTab = m.styles.FormField.Render(" Advanced ")
|
|
||||||
} else {
|
|
||||||
generalTab = m.styles.FormField.Render(" General ")
|
|
||||||
advancedTab = m.styles.FocusedLabel.Render("[ Advanced ]")
|
|
||||||
}
|
|
||||||
|
|
||||||
return generalTab + " " + advancedTab
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderGeneralTab renders the general tab content
|
|
||||||
func (m *addFormModel) renderGeneralTab() string {
|
|
||||||
var b strings.Builder
|
|
||||||
|
|
||||||
fields := []struct {
|
|
||||||
index int
|
|
||||||
label string
|
|
||||||
}{
|
|
||||||
{nameInput, "Host Name *"},
|
|
||||||
{hostnameInput, "Hostname/IP *"},
|
|
||||||
{userInput, "User"},
|
|
||||||
{portInput, "Port"},
|
|
||||||
{identityInput, "Identity File"},
|
|
||||||
{proxyJumpInput, "ProxyJump"},
|
|
||||||
{tagsInput, "Tags (comma-separated)"},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, field := range fields {
|
|
||||||
fieldStyle := m.styles.FormField
|
|
||||||
if m.focused == field.index {
|
|
||||||
fieldStyle = m.styles.FocusedLabel
|
|
||||||
}
|
|
||||||
b.WriteString(fieldStyle.Render(field.label))
|
|
||||||
b.WriteString("\n")
|
|
||||||
b.WriteString(m.inputs[field.index].View())
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderAdvancedTab renders the advanced tab content
|
|
||||||
func (m *addFormModel) renderAdvancedTab() string {
|
|
||||||
var b strings.Builder
|
|
||||||
|
|
||||||
fields := []struct {
|
|
||||||
index int
|
|
||||||
label string
|
|
||||||
}{
|
|
||||||
{optionsInput, "SSH Options"},
|
|
||||||
{remoteCommandInput, "Remote Command"},
|
|
||||||
{requestTTYInput, "Request TTY"},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, field := range fields {
|
|
||||||
fieldStyle := m.styles.FormField
|
|
||||||
if m.focused == field.index {
|
|
||||||
fieldStyle = m.styles.FocusedLabel
|
|
||||||
}
|
|
||||||
b.WriteString(fieldStyle.Render(field.label))
|
|
||||||
b.WriteString("\n")
|
|
||||||
b.WriteString(m.inputs[field.index].View())
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Standalone wrapper for add form
|
// Standalone wrapper for add form
|
||||||
type standaloneAddForm struct {
|
type standaloneAddForm struct {
|
||||||
*addFormModel
|
*addFormModel
|
||||||
@ -490,8 +290,6 @@ func (m *addFormModel) submitForm() tea.Cmd {
|
|||||||
identity := strings.TrimSpace(m.inputs[identityInput].Value())
|
identity := strings.TrimSpace(m.inputs[identityInput].Value())
|
||||||
proxyJump := strings.TrimSpace(m.inputs[proxyJumpInput].Value())
|
proxyJump := strings.TrimSpace(m.inputs[proxyJumpInput].Value())
|
||||||
options := strings.TrimSpace(m.inputs[optionsInput].Value())
|
options := strings.TrimSpace(m.inputs[optionsInput].Value())
|
||||||
remoteCommand := strings.TrimSpace(m.inputs[remoteCommandInput].Value())
|
|
||||||
requestTTY := strings.TrimSpace(m.inputs[requestTTYInput].Value())
|
|
||||||
|
|
||||||
// Set defaults
|
// Set defaults
|
||||||
if user == "" {
|
if user == "" {
|
||||||
@ -520,16 +318,14 @@ func (m *addFormModel) submitForm() tea.Cmd {
|
|||||||
|
|
||||||
// Create host configuration
|
// Create host configuration
|
||||||
host := config.SSHHost{
|
host := config.SSHHost{
|
||||||
Name: name,
|
Name: name,
|
||||||
Hostname: hostname,
|
Hostname: hostname,
|
||||||
User: user,
|
User: user,
|
||||||
Port: port,
|
Port: port,
|
||||||
Identity: identity,
|
Identity: identity,
|
||||||
ProxyJump: proxyJump,
|
ProxyJump: proxyJump,
|
||||||
Options: config.ParseSSHOptionsFromCommand(options),
|
Options: config.ParseSSHOptionsFromCommand(options),
|
||||||
RemoteCommand: remoteCommand,
|
Tags: tags,
|
||||||
RequestTTY: requestTTY,
|
|
||||||
Tags: tags,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add to config
|
// Add to config
|
||||||
|
@ -1,46 +1,29 @@
|
|||||||
package ui
|
package ui
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/Gu1llaum-3/sshm/internal/config"
|
"github.com/Gu1llaum-3/sshm/internal/config"
|
||||||
"github.com/Gu1llaum-3/sshm/internal/validation"
|
"github.com/Gu1llaum-3/sshm/internal/validation"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/charmbracelet/bubbles/textinput"
|
"github.com/charmbracelet/bubbles/textinput"
|
||||||
tea "github.com/charmbracelet/bubbletea"
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
|
"github.com/charmbracelet/lipgloss"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
|
||||||
focusAreaHosts = iota
|
|
||||||
focusAreaProperties
|
|
||||||
)
|
|
||||||
|
|
||||||
type editFormSubmitMsg struct {
|
|
||||||
hostname string
|
|
||||||
err error
|
|
||||||
}
|
|
||||||
|
|
||||||
type editFormCancelMsg struct{}
|
|
||||||
|
|
||||||
type editFormModel struct {
|
type editFormModel struct {
|
||||||
hostInputs []textinput.Model // Support for multiple hosts
|
inputs []textinput.Model
|
||||||
inputs []textinput.Model
|
focused int
|
||||||
focusArea int // 0=hosts, 1=properties
|
err string
|
||||||
focused int
|
success bool
|
||||||
currentTab int // 0=General, 1=Advanced (only applies when focusArea == focusAreaProperties)
|
styles Styles
|
||||||
err string
|
originalName string
|
||||||
styles Styles
|
host *config.SSHHost // Store the original host with SourceFile
|
||||||
originalName string
|
width int
|
||||||
originalHosts []string // Store original host names for multi-host detection
|
height int
|
||||||
host *config.SSHHost // Store the original host with SourceFile
|
configFile string
|
||||||
configFile string // Configuration file path passed by user
|
|
||||||
actualConfigFile string // Actual config file to use (either configFile or host.SourceFile)
|
|
||||||
width int
|
|
||||||
height int
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewEditForm creates a new edit form model that supports both single and multi-host editing
|
// NewEditForm creates a new edit form model
|
||||||
func NewEditForm(hostName string, styles Styles, width, height int, configFile string) (*editFormModel, error) {
|
func NewEditForm(hostName string, styles Styles, width, height int, configFile string) (*editFormModel, error) {
|
||||||
// Get the existing host configuration
|
// Get the existing host configuration
|
||||||
var host *config.SSHHost
|
var host *config.SSHHost
|
||||||
@ -56,482 +39,207 @@ func NewEditForm(hostName string, styles Styles, width, height int, configFile s
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if this host is part of a multi-host declaration
|
inputs := make([]textinput.Model, 8)
|
||||||
var actualConfigFile string
|
|
||||||
var hostNames []string
|
|
||||||
var isMulti bool
|
|
||||||
|
|
||||||
if configFile != "" {
|
// Name input
|
||||||
actualConfigFile = configFile
|
inputs[nameInput] = textinput.New()
|
||||||
} else {
|
inputs[nameInput].Placeholder = "server-name"
|
||||||
actualConfigFile = host.SourceFile
|
inputs[nameInput].Focus()
|
||||||
}
|
inputs[nameInput].CharLimit = 50
|
||||||
|
inputs[nameInput].Width = 30
|
||||||
if actualConfigFile != "" {
|
inputs[nameInput].SetValue(host.Name)
|
||||||
isMulti, hostNames, err = config.IsPartOfMultiHostDeclaration(hostName, actualConfigFile)
|
|
||||||
if err != nil {
|
|
||||||
// If we can't determine multi-host status, treat as single host
|
|
||||||
isMulti = false
|
|
||||||
hostNames = []string{hostName}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !isMulti {
|
|
||||||
hostNames = []string{hostName}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create host inputs
|
|
||||||
hostInputs := make([]textinput.Model, len(hostNames))
|
|
||||||
for i, name := range hostNames {
|
|
||||||
hostInputs[i] = textinput.New()
|
|
||||||
hostInputs[i].Placeholder = "host-name"
|
|
||||||
hostInputs[i].SetValue(name)
|
|
||||||
if i == 0 {
|
|
||||||
hostInputs[i].Focus()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
inputs := make([]textinput.Model, 9) // Increased from 8 to 9 for RequestTTY
|
|
||||||
|
|
||||||
// Hostname input
|
// Hostname input
|
||||||
inputs[0] = textinput.New()
|
inputs[hostnameInput] = textinput.New()
|
||||||
inputs[0].Placeholder = "192.168.1.100 or example.com"
|
inputs[hostnameInput].Placeholder = "192.168.1.100 or example.com"
|
||||||
inputs[0].CharLimit = 100
|
inputs[hostnameInput].CharLimit = 100
|
||||||
inputs[0].Width = 30
|
inputs[hostnameInput].Width = 30
|
||||||
inputs[0].SetValue(host.Hostname)
|
inputs[hostnameInput].SetValue(host.Hostname)
|
||||||
|
|
||||||
// User input
|
// User input
|
||||||
inputs[1] = textinput.New()
|
inputs[userInput] = textinput.New()
|
||||||
inputs[1].Placeholder = "root"
|
inputs[userInput].Placeholder = "root"
|
||||||
inputs[1].CharLimit = 50
|
inputs[userInput].CharLimit = 50
|
||||||
inputs[1].Width = 30
|
inputs[userInput].Width = 30
|
||||||
inputs[1].SetValue(host.User)
|
inputs[userInput].SetValue(host.User)
|
||||||
|
|
||||||
// Port input
|
// Port input
|
||||||
inputs[2] = textinput.New()
|
inputs[portInput] = textinput.New()
|
||||||
inputs[2].Placeholder = "22"
|
inputs[portInput].Placeholder = "22"
|
||||||
inputs[2].CharLimit = 5
|
inputs[portInput].CharLimit = 5
|
||||||
inputs[2].Width = 30
|
inputs[portInput].Width = 30
|
||||||
inputs[2].SetValue(host.Port)
|
inputs[portInput].SetValue(host.Port)
|
||||||
|
|
||||||
// Identity input
|
// Identity input
|
||||||
inputs[3] = textinput.New()
|
inputs[identityInput] = textinput.New()
|
||||||
inputs[3].Placeholder = "~/.ssh/id_rsa"
|
inputs[identityInput].Placeholder = "~/.ssh/id_rsa"
|
||||||
inputs[3].CharLimit = 200
|
inputs[identityInput].CharLimit = 200
|
||||||
inputs[3].Width = 50
|
inputs[identityInput].Width = 50
|
||||||
inputs[3].SetValue(host.Identity)
|
inputs[identityInput].SetValue(host.Identity)
|
||||||
|
|
||||||
// ProxyJump input
|
// ProxyJump input
|
||||||
inputs[4] = textinput.New()
|
inputs[proxyJumpInput] = textinput.New()
|
||||||
inputs[4].Placeholder = "jump-server"
|
inputs[proxyJumpInput].Placeholder = "user@jump-host:port or existing-host-name"
|
||||||
inputs[4].CharLimit = 100
|
inputs[proxyJumpInput].CharLimit = 200
|
||||||
inputs[4].Width = 30
|
inputs[proxyJumpInput].Width = 50
|
||||||
inputs[4].SetValue(host.ProxyJump)
|
inputs[proxyJumpInput].SetValue(host.ProxyJump)
|
||||||
|
|
||||||
// Options input
|
// SSH Options input
|
||||||
inputs[5] = textinput.New()
|
inputs[optionsInput] = textinput.New()
|
||||||
inputs[5].Placeholder = "-o StrictHostKeyChecking=no"
|
inputs[optionsInput].Placeholder = "-o Compression=yes -o ServerAliveInterval=60"
|
||||||
inputs[5].CharLimit = 200
|
inputs[optionsInput].CharLimit = 500
|
||||||
inputs[5].Width = 50
|
inputs[optionsInput].Width = 70
|
||||||
if host.Options != "" {
|
inputs[optionsInput].SetValue(config.FormatSSHOptionsForCommand(host.Options))
|
||||||
inputs[5].SetValue(config.FormatSSHOptionsForCommand(host.Options))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tags input
|
// Tags input
|
||||||
inputs[6] = textinput.New()
|
inputs[tagsInput] = textinput.New()
|
||||||
inputs[6].Placeholder = "production, web, database"
|
inputs[tagsInput].Placeholder = "production, web, database"
|
||||||
inputs[6].CharLimit = 200
|
inputs[tagsInput].CharLimit = 200
|
||||||
inputs[6].Width = 50
|
inputs[tagsInput].Width = 50
|
||||||
if len(host.Tags) > 0 {
|
if len(host.Tags) > 0 {
|
||||||
inputs[6].SetValue(strings.Join(host.Tags, ", "))
|
inputs[tagsInput].SetValue(strings.Join(host.Tags, ", "))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remote Command input
|
|
||||||
inputs[7] = textinput.New()
|
|
||||||
inputs[7].Placeholder = "ls -la, htop, bash"
|
|
||||||
inputs[7].CharLimit = 300
|
|
||||||
inputs[7].Width = 70
|
|
||||||
inputs[7].SetValue(host.RemoteCommand)
|
|
||||||
|
|
||||||
// RequestTTY input
|
|
||||||
inputs[8] = textinput.New()
|
|
||||||
inputs[8].Placeholder = "yes, no, force, auto"
|
|
||||||
inputs[8].CharLimit = 10
|
|
||||||
inputs[8].Width = 30
|
|
||||||
inputs[8].SetValue(host.RequestTTY)
|
|
||||||
|
|
||||||
return &editFormModel{
|
return &editFormModel{
|
||||||
hostInputs: hostInputs,
|
inputs: inputs,
|
||||||
inputs: inputs,
|
focused: nameInput,
|
||||||
focusArea: focusAreaHosts, // Start with hosts focused for multi-host editing
|
originalName: hostName,
|
||||||
focused: 0,
|
host: host,
|
||||||
currentTab: 0, // Start on General tab
|
configFile: configFile,
|
||||||
originalName: hostName,
|
styles: styles,
|
||||||
originalHosts: hostNames,
|
width: width,
|
||||||
host: host,
|
height: height,
|
||||||
configFile: configFile,
|
|
||||||
actualConfigFile: actualConfigFile,
|
|
||||||
styles: styles,
|
|
||||||
width: width,
|
|
||||||
height: height,
|
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Messages for communication with parent model
|
||||||
|
type editFormSubmitMsg struct {
|
||||||
|
hostname string
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
type editFormCancelMsg struct{}
|
||||||
|
|
||||||
func (m *editFormModel) Init() tea.Cmd {
|
func (m *editFormModel) Init() tea.Cmd {
|
||||||
return textinput.Blink
|
return textinput.Blink
|
||||||
}
|
}
|
||||||
|
|
||||||
// addHostInput adds a new empty host input
|
func (m *editFormModel) Update(msg tea.Msg) (*editFormModel, tea.Cmd) {
|
||||||
func (m *editFormModel) addHostInput() tea.Cmd {
|
|
||||||
newInput := textinput.New()
|
|
||||||
newInput.Placeholder = "host-name"
|
|
||||||
newInput.Focus()
|
|
||||||
|
|
||||||
// Unfocus current input regardless of which area we're in
|
|
||||||
if m.focusArea == focusAreaHosts && m.focused < len(m.hostInputs) {
|
|
||||||
m.hostInputs[m.focused].Blur()
|
|
||||||
} else if m.focusArea == focusAreaProperties && m.focused < len(m.inputs) {
|
|
||||||
m.inputs[m.focused].Blur()
|
|
||||||
}
|
|
||||||
|
|
||||||
m.hostInputs = append(m.hostInputs, newInput)
|
|
||||||
|
|
||||||
// Move focus to the new host input
|
|
||||||
m.focusArea = focusAreaHosts
|
|
||||||
m.focused = len(m.hostInputs) - 1
|
|
||||||
|
|
||||||
return textinput.Blink
|
|
||||||
}
|
|
||||||
|
|
||||||
// deleteHostInput removes the currently focused host input
|
|
||||||
func (m *editFormModel) deleteHostInput() tea.Cmd {
|
|
||||||
if len(m.hostInputs) <= 1 || m.focusArea != focusAreaHosts {
|
|
||||||
return nil // Can't delete if only one host or not in host area
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the focused host input
|
|
||||||
m.hostInputs = append(m.hostInputs[:m.focused], m.hostInputs[m.focused+1:]...)
|
|
||||||
|
|
||||||
// Adjust focus
|
|
||||||
if m.focused >= len(m.hostInputs) {
|
|
||||||
m.focused = len(m.hostInputs) - 1
|
|
||||||
}
|
|
||||||
|
|
||||||
// Focus the new current input
|
|
||||||
if len(m.hostInputs) > 0 {
|
|
||||||
m.hostInputs[m.focused].Focus()
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// updateFocus updates the focus state based on current area and index
|
|
||||||
func (m *editFormModel) updateFocus() tea.Cmd {
|
|
||||||
// Blur all inputs first
|
|
||||||
for i := range m.hostInputs {
|
|
||||||
m.hostInputs[i].Blur()
|
|
||||||
}
|
|
||||||
for i := range m.inputs {
|
|
||||||
m.inputs[i].Blur()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Focus the appropriate input
|
|
||||||
if m.focusArea == focusAreaHosts {
|
|
||||||
if m.focused < len(m.hostInputs) {
|
|
||||||
m.hostInputs[m.focused].Focus()
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if m.focused < len(m.inputs) {
|
|
||||||
m.inputs[m.focused].Focus()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return textinput.Blink
|
|
||||||
}
|
|
||||||
|
|
||||||
// getPropertiesForCurrentTab returns the property input indices for the current tab
|
|
||||||
func (m *editFormModel) getPropertiesForCurrentTab() []int {
|
|
||||||
switch m.currentTab {
|
|
||||||
case 0: // General
|
|
||||||
return []int{0, 1, 2, 3, 4, 6} // hostname, user, port, identity, proxyjump, tags
|
|
||||||
case 1: // Advanced
|
|
||||||
return []int{5, 7, 8} // options, remotecommand, requesttty
|
|
||||||
default:
|
|
||||||
return []int{0, 1, 2, 3, 4, 6}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// getFirstPropertyForTab returns the first property index for a given tab
|
|
||||||
func (m *editFormModel) getFirstPropertyForTab(tab int) int {
|
|
||||||
properties := []int{0, 1, 2, 3, 4, 6} // General tab
|
|
||||||
if tab == 1 {
|
|
||||||
properties = []int{5, 7, 8} // Advanced tab
|
|
||||||
}
|
|
||||||
if len(properties) > 0 {
|
|
||||||
return properties[0]
|
|
||||||
}
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// handleEditNavigation handles navigation in the edit form with tab support
|
|
||||||
func (m *editFormModel) handleEditNavigation(key string) tea.Cmd {
|
|
||||||
if m.focusArea == focusAreaHosts {
|
|
||||||
// Navigate in hosts area
|
|
||||||
if key == "up" || key == "shift+tab" {
|
|
||||||
m.focused--
|
|
||||||
} else {
|
|
||||||
m.focused++
|
|
||||||
}
|
|
||||||
|
|
||||||
if m.focused >= len(m.hostInputs) {
|
|
||||||
// Move to properties area, keep current tab
|
|
||||||
m.focusArea = focusAreaProperties
|
|
||||||
// Keep the current tab instead of forcing it to 0
|
|
||||||
m.focused = m.getFirstPropertyForTab(m.currentTab)
|
|
||||||
} else if m.focused < 0 {
|
|
||||||
m.focused = len(m.hostInputs) - 1
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Navigate in properties area within current tab
|
|
||||||
currentTabProperties := m.getPropertiesForCurrentTab()
|
|
||||||
|
|
||||||
// Find current position within the tab
|
|
||||||
currentPos := 0
|
|
||||||
for i, prop := range currentTabProperties {
|
|
||||||
if prop == m.focused {
|
|
||||||
currentPos = i
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle form submission on last field of Advanced tab
|
|
||||||
if key == "enter" && m.currentTab == 1 && currentPos == len(currentTabProperties)-1 {
|
|
||||||
return m.submitEditForm()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Navigate within current tab
|
|
||||||
if key == "up" || key == "shift+tab" {
|
|
||||||
currentPos--
|
|
||||||
} else {
|
|
||||||
currentPos++
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle transitions between areas and tabs
|
|
||||||
if currentPos >= len(currentTabProperties) {
|
|
||||||
// Move to next area/tab
|
|
||||||
if m.currentTab == 0 {
|
|
||||||
// Move to advanced tab
|
|
||||||
m.currentTab = 1
|
|
||||||
m.focused = m.getFirstPropertyForTab(1)
|
|
||||||
} else {
|
|
||||||
// Move back to hosts area
|
|
||||||
m.focusArea = focusAreaHosts
|
|
||||||
m.focused = 0
|
|
||||||
}
|
|
||||||
} else if currentPos < 0 {
|
|
||||||
// Move to previous area/tab
|
|
||||||
if m.currentTab == 1 {
|
|
||||||
// Move to general tab
|
|
||||||
m.currentTab = 0
|
|
||||||
properties := m.getPropertiesForCurrentTab()
|
|
||||||
m.focused = properties[len(properties)-1]
|
|
||||||
} else {
|
|
||||||
// Move to hosts area
|
|
||||||
m.focusArea = focusAreaHosts
|
|
||||||
m.focused = len(m.hostInputs) - 1
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
m.focused = currentTabProperties[currentPos]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return m.updateFocus()
|
|
||||||
}
|
|
||||||
|
|
||||||
// getMinimumHeight calculates the minimum height needed to display the edit form
|
|
||||||
func (m *editFormModel) getMinimumHeight() int {
|
|
||||||
// Title: 1 line + 2 newlines = 3
|
|
||||||
titleLines := 3
|
|
||||||
// Config file info: 1 line + 2 newlines = 3
|
|
||||||
configLines := 3
|
|
||||||
// Host Names section: title (1) + spacing (2) = 3
|
|
||||||
hostSectionLines := 3
|
|
||||||
// Host inputs: number of hosts * 3 lines each (reduced from 4)
|
|
||||||
hostLines := len(m.hostInputs) * 3
|
|
||||||
// Properties section: title (1) + spacing (2) = 3
|
|
||||||
propertiesSectionLines := 3
|
|
||||||
// Tabs: 1 line + 2 newlines = 3
|
|
||||||
tabLines := 3
|
|
||||||
// Fields in current tab
|
|
||||||
var fieldsCount int
|
|
||||||
if m.currentTab == 0 {
|
|
||||||
fieldsCount = 6 // 6 fields in general tab
|
|
||||||
} else {
|
|
||||||
fieldsCount = 3 // 3 fields in advanced tab
|
|
||||||
}
|
|
||||||
// Each field: reduced from 4 to 3 lines per field
|
|
||||||
fieldsLines := fieldsCount * 3
|
|
||||||
// Help text: 3 lines
|
|
||||||
helpLines := 3
|
|
||||||
// Error message space when needed: 2 lines
|
|
||||||
errorLines := 0 // Only count when there's actually an error
|
|
||||||
if m.err != "" {
|
|
||||||
errorLines = 2
|
|
||||||
}
|
|
||||||
|
|
||||||
return titleLines + configLines + hostSectionLines + hostLines + propertiesSectionLines + tabLines + fieldsLines + helpLines + errorLines + 1 // +1 minimal safety margin
|
|
||||||
}
|
|
||||||
|
|
||||||
// isHeightSufficient checks if the current terminal height is sufficient
|
|
||||||
func (m *editFormModel) isHeightSufficient() bool {
|
|
||||||
return m.height >= m.getMinimumHeight()
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderHeightWarning renders a warning message when height is insufficient
|
|
||||||
func (m *editFormModel) renderHeightWarning() string {
|
|
||||||
required := m.getMinimumHeight()
|
|
||||||
current := m.height
|
|
||||||
|
|
||||||
warning := m.styles.ErrorText.Render("⚠️ Terminal height is too small!")
|
|
||||||
details := m.styles.FormField.Render(fmt.Sprintf("Current: %d lines, Required: %d lines", current, required))
|
|
||||||
instruction := m.styles.FormHelp.Render("Please resize your terminal window and try again.")
|
|
||||||
instruction2 := m.styles.FormHelp.Render("Press Ctrl+C to cancel or resize terminal window.")
|
|
||||||
|
|
||||||
return warning + "\n\n" + details + "\n\n" + instruction + "\n" + instruction2
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *editFormModel) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|
||||||
var cmds []tea.Cmd
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
switch msg := msg.(type) {
|
switch msg := msg.(type) {
|
||||||
case tea.WindowSizeMsg:
|
case tea.WindowSizeMsg:
|
||||||
m.width = msg.Width
|
m.width = msg.Width
|
||||||
m.height = msg.Height
|
m.height = msg.Height
|
||||||
|
m.styles = NewStyles(m.width)
|
||||||
|
return m, nil
|
||||||
|
|
||||||
case tea.KeyMsg:
|
case tea.KeyMsg:
|
||||||
switch msg.String() {
|
switch msg.String() {
|
||||||
case "ctrl+c", "esc":
|
case "ctrl+c", "esc":
|
||||||
m.err = ""
|
|
||||||
return m, func() tea.Msg { return editFormCancelMsg{} }
|
return m, func() tea.Msg { return editFormCancelMsg{} }
|
||||||
|
|
||||||
case "ctrl+s":
|
case "ctrl+enter":
|
||||||
// Allow submission from any field with Ctrl+S (Save)
|
// Allow submission from any field with Ctrl+Enter
|
||||||
return m, m.submitEditForm()
|
return m, m.submitEditForm()
|
||||||
|
|
||||||
case "ctrl+j":
|
|
||||||
// Switch to next tab
|
|
||||||
m.currentTab = (m.currentTab + 1) % 2
|
|
||||||
// If we're in hosts area, stay there. If in properties, go to the first field of the new tab
|
|
||||||
if m.focusArea == focusAreaProperties {
|
|
||||||
m.focused = m.getFirstPropertyForTab(m.currentTab)
|
|
||||||
}
|
|
||||||
return m, m.updateFocus()
|
|
||||||
|
|
||||||
case "ctrl+k":
|
|
||||||
// Switch to previous tab
|
|
||||||
m.currentTab = (m.currentTab - 1 + 2) % 2
|
|
||||||
// If we're in hosts area, stay there. If in properties, go to the first field of the new tab
|
|
||||||
if m.focusArea == focusAreaProperties {
|
|
||||||
m.focused = m.getFirstPropertyForTab(m.currentTab)
|
|
||||||
}
|
|
||||||
return m, m.updateFocus()
|
|
||||||
|
|
||||||
case "tab", "shift+tab", "enter", "up", "down":
|
case "tab", "shift+tab", "enter", "up", "down":
|
||||||
return m, m.handleEditNavigation(msg.String())
|
s := msg.String()
|
||||||
|
|
||||||
case "ctrl+a":
|
// Handle form submission
|
||||||
// Add a new host input
|
if s == "enter" && m.focused == len(m.inputs)-1 {
|
||||||
return m, m.addHostInput()
|
return m, m.submitEditForm()
|
||||||
|
|
||||||
case "ctrl+d":
|
|
||||||
// Delete the currently focused host (if more than one exists)
|
|
||||||
if m.focusArea == focusAreaHosts && len(m.hostInputs) > 1 {
|
|
||||||
return m, m.deleteHostInput()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Cycle inputs
|
||||||
|
if s == "up" || s == "shift+tab" {
|
||||||
|
m.focused--
|
||||||
|
} else {
|
||||||
|
m.focused++
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.focused > len(m.inputs)-1 {
|
||||||
|
m.focused = 0
|
||||||
|
} else if m.focused < 0 {
|
||||||
|
m.focused = len(m.inputs) - 1
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range m.inputs {
|
||||||
|
if i == m.focused {
|
||||||
|
cmds = append(cmds, m.inputs[i].Focus())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
m.inputs[i].Blur()
|
||||||
|
}
|
||||||
|
|
||||||
|
return m, tea.Batch(cmds...)
|
||||||
}
|
}
|
||||||
|
|
||||||
case editFormSubmitMsg:
|
case editFormSubmitMsg:
|
||||||
if msg.err != nil {
|
if msg.err != nil {
|
||||||
m.err = msg.err.Error()
|
m.err = msg.err.Error()
|
||||||
} else {
|
} else {
|
||||||
// Success: let the wrapper handle this
|
m.success = true
|
||||||
// In TUI mode, this will be handled by the parent
|
m.err = ""
|
||||||
// In standalone mode, the wrapper will quit
|
// Don't quit here, let parent handle the success
|
||||||
}
|
}
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update host inputs
|
// Update inputs
|
||||||
hostCmd := make([]tea.Cmd, len(m.hostInputs))
|
cmd := make([]tea.Cmd, len(m.inputs))
|
||||||
for i := range m.hostInputs {
|
|
||||||
m.hostInputs[i], hostCmd[i] = m.hostInputs[i].Update(msg)
|
|
||||||
}
|
|
||||||
cmds = append(cmds, hostCmd...)
|
|
||||||
|
|
||||||
// Update property inputs
|
|
||||||
propCmd := make([]tea.Cmd, len(m.inputs))
|
|
||||||
for i := range m.inputs {
|
for i := range m.inputs {
|
||||||
m.inputs[i], propCmd[i] = m.inputs[i].Update(msg)
|
m.inputs[i], cmd[i] = m.inputs[i].Update(msg)
|
||||||
}
|
}
|
||||||
cmds = append(cmds, propCmd...)
|
cmds = append(cmds, cmd...)
|
||||||
|
|
||||||
return m, tea.Batch(cmds...)
|
return m, tea.Batch(cmds...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *editFormModel) View() string {
|
func (m *editFormModel) View() string {
|
||||||
// Check if terminal height is sufficient
|
if m.success {
|
||||||
if !m.isHeightSufficient() {
|
return ""
|
||||||
return m.renderHeightWarning()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var b strings.Builder
|
var b strings.Builder
|
||||||
|
|
||||||
if m.err != "" {
|
b.WriteString(m.styles.FormTitle.Render("Edit SSH Host Configuration"))
|
||||||
b.WriteString(m.styles.Error.Render("Error: " + m.err))
|
b.WriteString("\n")
|
||||||
b.WriteString("\n\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
b.WriteString(m.styles.Header.Render("Edit SSH Host"))
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
|
|
||||||
|
// Show source file information
|
||||||
if m.host != nil && m.host.SourceFile != "" {
|
if m.host != nil && m.host.SourceFile != "" {
|
||||||
labelStyle := m.styles.FormField
|
b.WriteString("\n") // Ligne d'espace avant Config file
|
||||||
pathStyle := m.styles.FormField
|
|
||||||
|
// Style for "Config file:" label in primary color
|
||||||
|
labelStyle := lipgloss.NewStyle().
|
||||||
|
Foreground(lipgloss.Color("#00ADD8")). // Primary color
|
||||||
|
Bold(true)
|
||||||
|
|
||||||
|
// Style for the file path in white
|
||||||
|
pathStyle := lipgloss.NewStyle().
|
||||||
|
Foreground(lipgloss.Color("#FFFFFF"))
|
||||||
|
|
||||||
configInfo := labelStyle.Render("Config file: ") + pathStyle.Render(formatConfigFile(m.host.SourceFile))
|
configInfo := labelStyle.Render("Config file: ") + pathStyle.Render(formatConfigFile(m.host.SourceFile))
|
||||||
b.WriteString(configInfo)
|
b.WriteString(configInfo)
|
||||||
}
|
}
|
||||||
|
|
||||||
b.WriteString("\n\n")
|
b.WriteString("\n\n")
|
||||||
|
|
||||||
// Host Names Section
|
fields := []string{
|
||||||
b.WriteString(m.styles.FormTitle.Render("Host Names"))
|
"Host Name *",
|
||||||
b.WriteString("\n\n")
|
"Hostname/IP *",
|
||||||
|
"User",
|
||||||
for i, hostInput := range m.hostInputs {
|
"Port",
|
||||||
hostStyle := m.styles.FormField
|
"Identity File",
|
||||||
if m.focusArea == focusAreaHosts && m.focused == i {
|
"ProxyJump",
|
||||||
hostStyle = m.styles.FocusedLabel
|
"SSH Options",
|
||||||
}
|
"Tags (comma-separated)",
|
||||||
b.WriteString(hostStyle.Render(fmt.Sprintf("Host Name %d *", i+1)))
|
|
||||||
b.WriteString("\n")
|
|
||||||
b.WriteString(hostInput.View())
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Properties Section
|
for i, field := range fields {
|
||||||
b.WriteString(m.styles.FormTitle.Render("Common Properties"))
|
b.WriteString(m.styles.FormField.Render(field))
|
||||||
b.WriteString("\n\n")
|
b.WriteString("\n")
|
||||||
|
b.WriteString(m.inputs[i].View())
|
||||||
// Render tabs for properties
|
b.WriteString("\n\n")
|
||||||
b.WriteString(m.renderEditTabs())
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
|
|
||||||
// Render current tab content
|
|
||||||
switch m.currentTab {
|
|
||||||
case 0: // General
|
|
||||||
b.WriteString(m.renderEditGeneralTab())
|
|
||||||
case 1: // Advanced
|
|
||||||
b.WriteString(m.renderEditAdvancedTab())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if m.err != "" {
|
if m.err != "" {
|
||||||
@ -539,87 +247,9 @@ func (m *editFormModel) View() string {
|
|||||||
b.WriteString("\n\n")
|
b.WriteString("\n\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Show different help based on number of hosts
|
b.WriteString(m.styles.FormHelp.Render("Tab/Shift+Tab: navigate • Enter on last field: submit • Ctrl+Enter: submit • Ctrl+C/Esc: cancel"))
|
||||||
if len(m.hostInputs) > 1 {
|
b.WriteString("\n")
|
||||||
b.WriteString(m.styles.FormHelp.Render("Tab/↑↓/Enter: navigate • Ctrl+J/K: switch tabs • Ctrl+A: add host • Ctrl+D: delete host"))
|
b.WriteString(m.styles.FormHelp.Render("* Required fields"))
|
||||||
b.WriteString("\n")
|
|
||||||
} else {
|
|
||||||
b.WriteString(m.styles.FormHelp.Render("Tab/↑↓/Enter: navigate • Ctrl+J/K: switch tabs • Ctrl+A: add host"))
|
|
||||||
b.WriteString("\n")
|
|
||||||
}
|
|
||||||
b.WriteString(m.styles.FormHelp.Render("Ctrl+S: save • Ctrl+C/Esc: cancel • * Required fields"))
|
|
||||||
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderEditTabs renders the tab headers for properties
|
|
||||||
func (m *editFormModel) renderEditTabs() string {
|
|
||||||
var generalTab, advancedTab string
|
|
||||||
|
|
||||||
if m.currentTab == 0 {
|
|
||||||
generalTab = m.styles.FocusedLabel.Render("[ General ]")
|
|
||||||
advancedTab = m.styles.FormField.Render(" Advanced ")
|
|
||||||
} else {
|
|
||||||
generalTab = m.styles.FormField.Render(" General ")
|
|
||||||
advancedTab = m.styles.FocusedLabel.Render("[ Advanced ]")
|
|
||||||
}
|
|
||||||
|
|
||||||
return generalTab + " " + advancedTab
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderEditGeneralTab renders the general tab content for properties
|
|
||||||
func (m *editFormModel) renderEditGeneralTab() string {
|
|
||||||
var b strings.Builder
|
|
||||||
|
|
||||||
fields := []struct {
|
|
||||||
index int
|
|
||||||
label string
|
|
||||||
}{
|
|
||||||
{0, "Hostname/IP *"},
|
|
||||||
{1, "User"},
|
|
||||||
{2, "Port"},
|
|
||||||
{3, "Identity File"},
|
|
||||||
{4, "Proxy Jump"},
|
|
||||||
{6, "Tags (comma-separated)"},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, field := range fields {
|
|
||||||
fieldStyle := m.styles.FormField
|
|
||||||
if m.focusArea == focusAreaProperties && m.focused == field.index {
|
|
||||||
fieldStyle = m.styles.FocusedLabel
|
|
||||||
}
|
|
||||||
b.WriteString(fieldStyle.Render(field.label))
|
|
||||||
b.WriteString("\n")
|
|
||||||
b.WriteString(m.inputs[field.index].View())
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// renderEditAdvancedTab renders the advanced tab content for properties
|
|
||||||
func (m *editFormModel) renderEditAdvancedTab() string {
|
|
||||||
var b strings.Builder
|
|
||||||
|
|
||||||
fields := []struct {
|
|
||||||
index int
|
|
||||||
label string
|
|
||||||
}{
|
|
||||||
{5, "SSH Options"},
|
|
||||||
{7, "Remote Command"},
|
|
||||||
{8, "Request TTY"},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, field := range fields {
|
|
||||||
fieldStyle := m.styles.FormField
|
|
||||||
if m.focusArea == focusAreaProperties && m.focused == field.index {
|
|
||||||
fieldStyle = m.styles.FocusedLabel
|
|
||||||
}
|
|
||||||
b.WriteString(fieldStyle.Render(field.label))
|
|
||||||
b.WriteString("\n")
|
|
||||||
b.WriteString(m.inputs[field.index].View())
|
|
||||||
b.WriteString("\n\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
return b.String()
|
return b.String()
|
||||||
}
|
}
|
||||||
@ -634,29 +264,29 @@ func (m standaloneEditForm) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
case editFormSubmitMsg:
|
case editFormSubmitMsg:
|
||||||
if msg.err != nil {
|
if msg.err != nil {
|
||||||
m.editFormModel.err = msg.err.Error()
|
m.editFormModel.err = msg.err.Error()
|
||||||
return m, nil
|
|
||||||
} else {
|
} else {
|
||||||
// Success: quit the program
|
m.editFormModel.success = true
|
||||||
return m, tea.Quit
|
return m, tea.Quit
|
||||||
}
|
}
|
||||||
|
return m, nil
|
||||||
case editFormCancelMsg:
|
case editFormCancelMsg:
|
||||||
return m, tea.Quit
|
return m, tea.Quit
|
||||||
}
|
}
|
||||||
|
|
||||||
newForm, cmd := m.editFormModel.Update(msg)
|
newForm, cmd := m.editFormModel.Update(msg)
|
||||||
m.editFormModel = newForm.(*editFormModel)
|
m.editFormModel = newForm
|
||||||
return m, cmd
|
return m, cmd
|
||||||
}
|
}
|
||||||
|
|
||||||
// RunEditForm runs the edit form as a standalone program
|
// RunEditForm provides backward compatibility for standalone edit form
|
||||||
func RunEditForm(hostName string, configFile string) error {
|
func RunEditForm(hostName string, configFile string) error {
|
||||||
styles := NewStyles(80) // Default width
|
styles := NewStyles(80)
|
||||||
editForm, err := NewEditForm(hostName, styles, 80, 24, configFile)
|
editForm, err := NewEditForm(hostName, styles, 80, 24, configFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
m := standaloneEditForm{editForm}
|
m := standaloneEditForm{editForm}
|
||||||
|
|
||||||
p := tea.NewProgram(m, tea.WithAltScreen())
|
p := tea.NewProgram(m, tea.WithAltScreen())
|
||||||
_, err = p.Run()
|
_, err = p.Run()
|
||||||
return err
|
return err
|
||||||
@ -664,48 +294,28 @@ func RunEditForm(hostName string, configFile string) error {
|
|||||||
|
|
||||||
func (m *editFormModel) submitEditForm() tea.Cmd {
|
func (m *editFormModel) submitEditForm() tea.Cmd {
|
||||||
return func() tea.Msg {
|
return func() tea.Msg {
|
||||||
// Collect host names
|
// Get values
|
||||||
var hostNames []string
|
name := strings.TrimSpace(m.inputs[nameInput].Value())
|
||||||
for _, input := range m.hostInputs {
|
hostname := strings.TrimSpace(m.inputs[hostnameInput].Value())
|
||||||
name := strings.TrimSpace(input.Value())
|
user := strings.TrimSpace(m.inputs[userInput].Value())
|
||||||
if name != "" {
|
port := strings.TrimSpace(m.inputs[portInput].Value())
|
||||||
hostNames = append(hostNames, name)
|
identity := strings.TrimSpace(m.inputs[identityInput].Value())
|
||||||
}
|
proxyJump := strings.TrimSpace(m.inputs[proxyJumpInput].Value())
|
||||||
}
|
options := strings.TrimSpace(m.inputs[optionsInput].Value())
|
||||||
|
|
||||||
if len(hostNames) == 0 {
|
|
||||||
return editFormSubmitMsg{err: fmt.Errorf("at least one host name is required")}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get property values using direct indices
|
|
||||||
hostname := strings.TrimSpace(m.inputs[0].Value()) // hostnameInput
|
|
||||||
user := strings.TrimSpace(m.inputs[1].Value()) // userInput
|
|
||||||
port := strings.TrimSpace(m.inputs[2].Value()) // portInput
|
|
||||||
identity := strings.TrimSpace(m.inputs[3].Value()) // identityInput
|
|
||||||
proxyJump := strings.TrimSpace(m.inputs[4].Value()) // proxyJumpInput
|
|
||||||
options := strings.TrimSpace(m.inputs[5].Value()) // optionsInput
|
|
||||||
remoteCommand := strings.TrimSpace(m.inputs[7].Value()) // remoteCommandInput
|
|
||||||
requestTTY := strings.TrimSpace(m.inputs[8].Value()) // requestTTYInput
|
|
||||||
|
|
||||||
// Set defaults
|
// Set defaults
|
||||||
if port == "" {
|
if port == "" {
|
||||||
port = "22"
|
port = "22"
|
||||||
}
|
}
|
||||||
|
// Do not auto-fill identity with placeholder if left empty; keep it empty so it's optional
|
||||||
|
|
||||||
// Validate hostname
|
// Validate all fields
|
||||||
if hostname == "" {
|
if err := validation.ValidateHost(name, hostname, port, identity); err != nil {
|
||||||
return editFormSubmitMsg{err: fmt.Errorf("hostname is required")}
|
return editFormSubmitMsg{err: err}
|
||||||
}
|
|
||||||
|
|
||||||
// Validate all host names
|
|
||||||
for _, hostName := range hostNames {
|
|
||||||
if err := validation.ValidateHost(hostName, hostname, port, identity); err != nil {
|
|
||||||
return editFormSubmitMsg{err: err}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse tags
|
// Parse tags
|
||||||
tagsStr := strings.TrimSpace(m.inputs[6].Value()) // tagsInput
|
tagsStr := strings.TrimSpace(m.inputs[tagsInput].Value())
|
||||||
var tags []string
|
var tags []string
|
||||||
if tagsStr != "" {
|
if tagsStr != "" {
|
||||||
for _, tag := range strings.Split(tagsStr, ",") {
|
for _, tag := range strings.Split(tagsStr, ",") {
|
||||||
@ -716,33 +326,25 @@ func (m *editFormModel) submitEditForm() tea.Cmd {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the common host configuration
|
// Create updated host configuration
|
||||||
commonHost := config.SSHHost{
|
host := config.SSHHost{
|
||||||
Hostname: hostname,
|
Name: name,
|
||||||
User: user,
|
Hostname: hostname,
|
||||||
Port: port,
|
User: user,
|
||||||
Identity: identity,
|
Port: port,
|
||||||
ProxyJump: proxyJump,
|
Identity: identity,
|
||||||
Options: options,
|
ProxyJump: proxyJump,
|
||||||
RemoteCommand: remoteCommand,
|
Options: config.ParseSSHOptionsFromCommand(options),
|
||||||
RequestTTY: requestTTY,
|
Tags: tags,
|
||||||
Tags: tags,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Update the configuration
|
||||||
var err error
|
var err error
|
||||||
if len(hostNames) == 1 && len(m.originalHosts) == 1 {
|
if m.configFile != "" {
|
||||||
// Single host editing
|
err = config.UpdateSSHHostInFile(m.originalName, host, m.configFile)
|
||||||
commonHost.Name = hostNames[0]
|
|
||||||
if m.actualConfigFile != "" {
|
|
||||||
err = config.UpdateSSHHostInFile(m.originalName, commonHost, m.actualConfigFile)
|
|
||||||
} else {
|
|
||||||
err = config.UpdateSSHHost(m.originalName, commonHost)
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Multi-host editing or conversion from single to multi
|
err = config.UpdateSSHHost(m.originalName, host)
|
||||||
err = config.UpdateMultiHostBlock(m.originalHosts, hostNames, commonHost, m.actualConfigFile)
|
|
||||||
}
|
}
|
||||||
|
return editFormSubmitMsg{hostname: name, err: err}
|
||||||
return editFormSubmitMsg{hostname: hostNames[0], err: err}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -80,9 +80,6 @@ type Model struct {
|
|||||||
sortMode SortMode
|
sortMode SortMode
|
||||||
configFile string // Path to the SSH config file
|
configFile string // Path to the SSH config file
|
||||||
|
|
||||||
// Application configuration
|
|
||||||
appConfig *config.AppConfig
|
|
||||||
|
|
||||||
// Version update information
|
// Version update information
|
||||||
updateInfo *version.UpdateInfo
|
updateInfo *version.UpdateInfo
|
||||||
currentVersion string
|
currentVersion string
|
||||||
@ -102,10 +99,6 @@ type Model struct {
|
|||||||
height int
|
height int
|
||||||
styles Styles
|
styles Styles
|
||||||
ready bool
|
ready bool
|
||||||
|
|
||||||
// Error handling
|
|
||||||
errorMessage string
|
|
||||||
showingError bool
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// updateTableStyles updates the table header border color based on focus state
|
// updateTableStyles updates the table header border color based on focus state
|
||||||
|
@ -42,7 +42,7 @@ func NewMoveForm(hostName string, styles Styles, width, height int, configFile s
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(files) == 0 {
|
if len(files) == 0 {
|
||||||
return nil, fmt.Errorf("no includes found in SSH config file - move operation requires multiple config files")
|
return nil, fmt.Errorf("no other config files available to move host to")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create a custom file selector for move operation
|
// Create a custom file selector for move operation
|
||||||
|
@ -37,64 +37,35 @@ func sortHostsByName(hosts []config.SSHHost) []config.SSHHost {
|
|||||||
|
|
||||||
// filterHosts filters hosts according to the search query (name or tags)
|
// filterHosts filters hosts according to the search query (name or tags)
|
||||||
func (m Model) filterHosts(query string) []config.SSHHost {
|
func (m Model) filterHosts(query string) []config.SSHHost {
|
||||||
subqueries := strings.Split(query, " ")
|
|
||||||
subqueriesLength := len(subqueries)
|
|
||||||
subfilteredHosts := make([][]config.SSHHost, subqueriesLength)
|
|
||||||
for i, subquery := range subqueries {
|
|
||||||
subfilteredHosts[i] = m.filterHostsByWord(subquery)
|
|
||||||
}
|
|
||||||
|
|
||||||
// return the intersection of search results
|
|
||||||
result := make([]config.SSHHost, 0)
|
|
||||||
tempMap := map[string]int{}
|
|
||||||
for _, hosts := range subfilteredHosts {
|
|
||||||
for _, host := range hosts {
|
|
||||||
if _, ok := tempMap[host.Name]; !ok {
|
|
||||||
tempMap[host.Name] = 1
|
|
||||||
} else {
|
|
||||||
tempMap[host.Name] = tempMap[host.Name] + 1
|
|
||||||
}
|
|
||||||
|
|
||||||
if tempMap[host.Name] == subqueriesLength {
|
|
||||||
result = append(result, host)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
// filterHostsByWord filters hosts according to a single word
|
|
||||||
func (m Model) filterHostsByWord(word string) []config.SSHHost {
|
|
||||||
var filtered []config.SSHHost
|
var filtered []config.SSHHost
|
||||||
|
|
||||||
if word == "" {
|
if query == "" {
|
||||||
filtered = m.hosts
|
filtered = m.hosts
|
||||||
} else {
|
} else {
|
||||||
word = strings.ToLower(word)
|
query = strings.ToLower(query)
|
||||||
|
|
||||||
for _, host := range m.hosts {
|
for _, host := range m.hosts {
|
||||||
// Check the hostname
|
// Check the hostname
|
||||||
if strings.Contains(strings.ToLower(host.Name), word) {
|
if strings.Contains(strings.ToLower(host.Name), query) {
|
||||||
filtered = append(filtered, host)
|
filtered = append(filtered, host)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check the hostname
|
// Check the hostname
|
||||||
if strings.Contains(strings.ToLower(host.Hostname), word) {
|
if strings.Contains(strings.ToLower(host.Hostname), query) {
|
||||||
filtered = append(filtered, host)
|
filtered = append(filtered, host)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check the user
|
// Check the user
|
||||||
if strings.Contains(strings.ToLower(host.User), word) {
|
if strings.Contains(strings.ToLower(host.User), query) {
|
||||||
filtered = append(filtered, host)
|
filtered = append(filtered, host)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check the tags
|
// Check the tags
|
||||||
for _, tag := range host.Tags {
|
for _, tag := range host.Tags {
|
||||||
if strings.Contains(strings.ToLower(tag), word) {
|
if strings.Contains(strings.ToLower(tag), query) {
|
||||||
filtered = append(filtered, host)
|
filtered = append(filtered, host)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
@ -33,8 +33,7 @@ type Styles struct {
|
|||||||
HelpText lipgloss.Style
|
HelpText lipgloss.Style
|
||||||
|
|
||||||
// Error and confirmation styles
|
// Error and confirmation styles
|
||||||
Error lipgloss.Style
|
Error lipgloss.Style
|
||||||
ErrorText lipgloss.Style
|
|
||||||
|
|
||||||
// Form styles (for add/edit forms)
|
// Form styles (for add/edit forms)
|
||||||
FormTitle lipgloss.Style
|
FormTitle lipgloss.Style
|
||||||
@ -98,11 +97,6 @@ func NewStyles(width int) Styles {
|
|||||||
BorderForeground(lipgloss.Color(ErrorColor)).
|
BorderForeground(lipgloss.Color(ErrorColor)).
|
||||||
Padding(1, 2),
|
Padding(1, 2),
|
||||||
|
|
||||||
// Error text style (no border, just red text)
|
|
||||||
ErrorText: lipgloss.NewStyle().
|
|
||||||
Foreground(lipgloss.Color(ErrorColor)).
|
|
||||||
Bold(true),
|
|
||||||
|
|
||||||
// Form styles
|
// Form styles
|
||||||
FormTitle: lipgloss.NewStyle().
|
FormTitle: lipgloss.NewStyle().
|
||||||
Foreground(lipgloss.Color("#FFFDF5")).
|
Foreground(lipgloss.Color("#FFFDF5")).
|
||||||
|
@ -17,15 +17,6 @@ import (
|
|||||||
|
|
||||||
// NewModel creates a new TUI model with the given SSH hosts
|
// NewModel creates a new TUI model with the given SSH hosts
|
||||||
func NewModel(hosts []config.SSHHost, configFile, currentVersion string) Model {
|
func NewModel(hosts []config.SSHHost, configFile, currentVersion string) Model {
|
||||||
// Load application configuration
|
|
||||||
appConfig, err := config.LoadAppConfig()
|
|
||||||
if err != nil {
|
|
||||||
// Log the error but continue with default configuration
|
|
||||||
fmt.Printf("Warning: Could not load application config: %v, using defaults\n", err)
|
|
||||||
defaultConfig := config.GetDefaultAppConfig()
|
|
||||||
appConfig = &defaultConfig
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize the history manager
|
// Initialize the history manager
|
||||||
historyManager, err := history.NewHistoryManager()
|
historyManager, err := history.NewHistoryManager()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -48,7 +39,6 @@ func NewModel(hosts []config.SSHHost, configFile, currentVersion string) Model {
|
|||||||
sortMode: SortByName,
|
sortMode: SortByName,
|
||||||
configFile: configFile,
|
configFile: configFile,
|
||||||
currentVersion: currentVersion,
|
currentVersion: currentVersion,
|
||||||
appConfig: appConfig,
|
|
||||||
styles: styles,
|
styles: styles,
|
||||||
width: 80,
|
width: 80,
|
||||||
height: 24,
|
height: 24,
|
||||||
|
@ -19,7 +19,6 @@ type (
|
|||||||
pingResultMsg *connectivity.HostPingResult
|
pingResultMsg *connectivity.HostPingResult
|
||||||
versionCheckMsg *version.UpdateInfo
|
versionCheckMsg *version.UpdateInfo
|
||||||
versionErrorMsg error
|
versionErrorMsg error
|
||||||
errorMsg string
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// startPingAllCmd creates a command to ping all hosts concurrently
|
// startPingAllCmd creates a command to ping all hosts concurrently
|
||||||
@ -158,14 +157,6 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
// as it might disrupt the user experience
|
// as it might disrupt the user experience
|
||||||
return m, nil
|
return m, nil
|
||||||
|
|
||||||
case errorMsg:
|
|
||||||
// Handle general error messages
|
|
||||||
if string(msg) == "clear" {
|
|
||||||
m.showingError = false
|
|
||||||
m.errorMessage = ""
|
|
||||||
}
|
|
||||||
return m, nil
|
|
||||||
|
|
||||||
case addFormSubmitMsg:
|
case addFormSubmitMsg:
|
||||||
if msg.err != nil {
|
if msg.err != nil {
|
||||||
// Show error in form
|
// Show error in form
|
||||||
@ -394,9 +385,9 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
}
|
}
|
||||||
case ViewEdit:
|
case ViewEdit:
|
||||||
if m.editForm != nil {
|
if m.editForm != nil {
|
||||||
var updatedModel tea.Model
|
var newForm *editFormModel
|
||||||
updatedModel, cmd = m.editForm.Update(msg)
|
newForm, cmd = m.editForm.Update(msg)
|
||||||
m.editForm = updatedModel.(*editFormModel)
|
m.editForm = newForm
|
||||||
return m, cmd
|
return m, cmd
|
||||||
}
|
}
|
||||||
case ViewMove:
|
case ViewMove:
|
||||||
@ -445,9 +436,8 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
|
|
||||||
func (m Model) handleListViewKeys(msg tea.KeyMsg) (tea.Model, tea.Cmd) {
|
func (m Model) handleListViewKeys(msg tea.KeyMsg) (tea.Model, tea.Cmd) {
|
||||||
var cmd tea.Cmd
|
var cmd tea.Cmd
|
||||||
key := msg.String()
|
|
||||||
|
|
||||||
switch key {
|
switch msg.String() {
|
||||||
case "esc", "ctrl+c":
|
case "esc", "ctrl+c":
|
||||||
if m.deleteMode {
|
if m.deleteMode {
|
||||||
// Exit delete mode
|
// Exit delete mode
|
||||||
@ -456,16 +446,10 @@ func (m Model) handleListViewKeys(msg tea.KeyMsg) (tea.Model, tea.Cmd) {
|
|||||||
m.table.Focus()
|
m.table.Focus()
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
// Use configurable key bindings for quit
|
return m, tea.Quit
|
||||||
if m.appConfig != nil && m.appConfig.KeyBindings.ShouldQuitOnKey(key) {
|
|
||||||
return m, tea.Quit
|
|
||||||
}
|
|
||||||
case "q":
|
case "q":
|
||||||
if !m.searchMode && !m.deleteMode {
|
if !m.searchMode && !m.deleteMode {
|
||||||
// Use configurable key bindings for quit
|
return m, tea.Quit
|
||||||
if m.appConfig != nil && m.appConfig.KeyBindings.ShouldQuitOnKey(key) {
|
|
||||||
return m, tea.Quit
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
case "/", "ctrl+f":
|
case "/", "ctrl+f":
|
||||||
if !m.searchMode && !m.deleteMode {
|
if !m.searchMode && !m.deleteMode {
|
||||||
@ -603,13 +587,8 @@ func (m Model) handleListViewKeys(msg tea.KeyMsg) (tea.Model, tea.Cmd) {
|
|||||||
hostName := extractHostNameFromTableRow(selected[0]) // Extract hostname from first column
|
hostName := extractHostNameFromTableRow(selected[0]) // Extract hostname from first column
|
||||||
moveForm, err := NewMoveForm(hostName, m.styles, m.width, m.height, m.configFile)
|
moveForm, err := NewMoveForm(hostName, m.styles, m.width, m.height, m.configFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// Show error message to user
|
// Handle error - could show in UI, e.g., no other config files available
|
||||||
m.errorMessage = err.Error()
|
return m, nil
|
||||||
m.showingError = true
|
|
||||||
return m, func() tea.Msg {
|
|
||||||
time.Sleep(3 * time.Second) // Show error for 3 seconds
|
|
||||||
return errorMsg("clear")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
m.moveForm = moveForm
|
m.moveForm = moveForm
|
||||||
m.viewMode = ViewMove
|
m.viewMode = ViewMove
|
||||||
|
@ -72,20 +72,6 @@ func (m Model) renderListView() string {
|
|||||||
components = append(components, updateStyle.Render(updateText))
|
components = append(components, updateStyle.Render(updateText))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add error message if there's one to show
|
|
||||||
if m.showingError && m.errorMessage != "" {
|
|
||||||
errorStyle := lipgloss.NewStyle().
|
|
||||||
Foreground(lipgloss.Color("9")). // Red color
|
|
||||||
Background(lipgloss.Color("1")). // Dark red background
|
|
||||||
Bold(true).
|
|
||||||
Padding(0, 1).
|
|
||||||
Border(lipgloss.RoundedBorder()).
|
|
||||||
BorderForeground(lipgloss.Color("9")).
|
|
||||||
Align(lipgloss.Center)
|
|
||||||
|
|
||||||
components = append(components, errorStyle.Render("❌ "+m.errorMessage))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the search bar with the appropriate style based on focus
|
// Add the search bar with the appropriate style based on focus
|
||||||
searchPrompt := "Search (/ to focus): "
|
searchPrompt := "Search (/ to focus): "
|
||||||
if m.searchMode {
|
if m.searchMode {
|
||||||
|
@ -128,8 +128,7 @@ func TestValidateIdentityFile(t *testing.T) {
|
|||||||
{"empty path", "", true}, // Optional field
|
{"empty path", "", true}, // Optional field
|
||||||
{"valid file", validFile, true},
|
{"valid file", validFile, true},
|
||||||
{"non-existent file", "/path/to/nonexistent", false},
|
{"non-existent file", "/path/to/nonexistent", false},
|
||||||
// Skip tilde path test in CI environments where ~/.ssh/id_rsa may not exist
|
{"tilde path", "~/.ssh/id_rsa", true}, // Will pass if file exists
|
||||||
// {"tilde path", "~/.ssh/id_rsa", true}, // Will pass if file exists
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tt := range tests {
|
for _, tt := range tests {
|
||||||
@ -139,15 +138,6 @@ func TestValidateIdentityFile(t *testing.T) {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// Test tilde path separately, but only if the file actually exists
|
|
||||||
t.Run("tilde path", func(t *testing.T) {
|
|
||||||
tildeFile := "~/.ssh/id_rsa"
|
|
||||||
// Just test that it doesn't crash, don't assume file exists
|
|
||||||
result := ValidateIdentityFile(tildeFile)
|
|
||||||
// Result can be true or false depending on file existence
|
|
||||||
_ = result // We just care that it doesn't panic
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestValidateHost(t *testing.T) {
|
func TestValidateHost(t *testing.T) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user