Skip to content

Commit c6867b5

Browse files
committedApr 1, 2024
Initial release;
1 parent 4e992c1 commit c6867b5

12 files changed

+495
-0
lines changed
 

‎.github/workflows/make-release.yml

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
name: Make release
2+
3+
on:
4+
workflow_dispatch:
5+
6+
permissions:
7+
contents: write
8+
9+
jobs:
10+
goreleaser:
11+
runs-on: ubuntu-latest
12+
steps:
13+
14+
- name: Checkout
15+
uses: actions/checkout@v4
16+
with:
17+
fetch-depth: 0
18+
19+
- name: Set up Go
20+
uses: actions/setup-go@v5
21+
with:
22+
go-version: "1.22"
23+
24+
- name: Run GoReleaser
25+
uses: goreleaser/goreleaser-action@v5
26+
with:
27+
distribution: goreleaser
28+
version: latest
29+
args: release --clean
30+
env:
31+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

‎.gitignore

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
# If you prefer the allow list template instead of the deny list, see community template:
2+
# https://github.com/github/gitignore/blob/main/community/Golang/Go.AllowList.gitignore
3+
#
4+
# Binaries for programs and plugins
5+
*.exe
6+
*.exe~
7+
*.dll
8+
*.so
9+
*.dylib
10+
11+
# Test binary, built with `go test -c`
12+
*.test
13+
14+
# Output of the go coverage tool, specifically when used with LiteIDE
15+
*.out
16+
17+
# Dependency directories (remove the comment below to include it)
18+
# vendor/
19+
20+
# Go workspace file
21+
go.work
22+
23+
# Project changes
24+
bin/
25+
dist/
26+
sshgut
27+
*.zip
28+
config.yaml

‎.goreleaser.yaml

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
---
2+
version: 1
3+
4+
before:
5+
hooks:
6+
- go mod tidy
7+
- go test .
8+
builds:
9+
- env:
10+
- CGO_ENABLED=0
11+
goos:
12+
- linux
13+
- windows
14+
- darwin
15+
goarch:
16+
- "386"
17+
- amd64
18+
- arm
19+
- arm64
20+
- ppc64
21+
goarm:
22+
- "7"
23+
mod_timestamp: "{{ .CommitTimestamp }}"
24+
ldflags:
25+
- -X main.Version={{ .Version }}
26+
flags:
27+
- -trimpath
28+
checksum:
29+
name_template: "checksums.txt"
30+
31+
32+
archives:
33+
- format: tar.gz
34+
name_template: >-
35+
{{ .ProjectName }}_
36+
{{- title .Os }}_
37+
{{- if eq .Arch "amd64" }}x86_64
38+
{{- else if eq .Arch "386" }}i386
39+
{{- else }}{{ .Arch }}{{ end }}
40+
{{- if .Arm }}v{{ .Arm }}{{ end }}
41+
format_overrides:
42+
- goos: windows
43+
format: zip
44+
files:
45+
- LICENSE
46+
- config.yaml.example
47+
48+
changelog:
49+
sort: asc
50+
filters:
51+
exclude:
52+
- "^docs:"
53+
- "^test:"
54+
- "merge conflict"
55+
- Merge pull request
56+
- Merge branch
57+
58+
release:
59+
draft: true
60+
replace_existing_draft: true

‎Changelog.md

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# 0.1.0 - 1 April 2024
2+
Initial release

‎README.md

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
# SSHGUT
2+
3+
The project is used to set up ssh tunnels and is essentially a wrapper around https://github.com/rgzr/sshtun.
4+
5+
6+
## Configuration
7+
Example of the configuration file `config.yaml.example`
8+
9+
```
10+
---
11+
remotes:
12+
- server: server1 # remote server that should be available on the localhost via `sshServer.host`
13+
remotePort: 80 # server's remote port that should be available on `localPort`
14+
localPort: 38081 # port on the localhost which
15+
localHost: 1.1.1.1 # bind tunnel to ip (default: 127.0.0.1)
16+
sshServer:
17+
host: server # SSH server is used to establish connection
18+
port: 22 # SSH port of the `host`
19+
user: user # SSH user
20+
keyPath: id_rsa # path to the ssh key file; allowed values [path to key] or `embedKey` in the second case, the key from the embedKey.ssh file will be added at build time.
21+
useKeyPass: false # Passphrase for decryption ssh key, passphrase is prompted at startup
22+
23+
- server: server2
24+
remotePort: 80
25+
localPort: 38082
26+
sshServer:
27+
host: server
28+
port: 22
29+
user: user
30+
keyPath: id_rsa
31+
useKeyPass: false
32+
33+
- server: server3
34+
remotePort: 80
35+
localPort: 38083
36+
sshServer:
37+
host: server
38+
port: 22
39+
user: user
40+
keyPath: id_rsa
41+
useKeyPass: false
42+
```
43+
44+
## How to run
45+
46+
Put config.yaml with sshgut and run it or set the path to the configuration file as below:
47+
```
48+
$ ./sshgut
49+
or
50+
$ ./sshgut --config your_config.yaml
51+
```

‎config.yaml.example

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
---
2+
remotes:
3+
- server: server1 # remote server that should be available on the localhost via `sshServer.host`
4+
remotePort: 80 # server's remote port that should be available on `localPort`
5+
localPort: 38081 # port on the localhost which
6+
localHost: 1.1.1.1 # bind tunnel to ip (default: 127.0.0.1)
7+
sshServer:
8+
host: server # SSH server is used to establish connection
9+
port: 22 # SSH port of the `host`
10+
user: user # SSH user
11+
keyPath: id_rsa # path to the ssh key file; allowed values [path to key] or `embedKey` in the second case, the key from the embedKey.ssh file will be added at build time.
12+
useKeyPass: false # Passphrase for decryption ssh key, passphrase is prompted at startup
13+
14+
- server: server2
15+
remotePort: 80
16+
localPort: 38082
17+
sshServer:
18+
host: server
19+
port: 22
20+
user: user
21+
keyPath: id_rsa
22+
useKeyPass: false
23+
24+
- server: server3
25+
remotePort: 80
26+
localPort: 38083
27+
sshServer:
28+
host: server
29+
port: 22
30+
user: user
31+
keyPath: id_rsa
32+
useKeyPass: false

‎embedKey.ssh

Whitespace-only changes.

‎go.mod

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
module sshgut
2+
3+
go 1.22
4+
5+
require (
6+
github.com/alecthomas/kingpin/v2 v2.4.0
7+
github.com/rgzr/sshtun v1.0.0
8+
github.com/rs/zerolog v1.32.0
9+
golang.org/x/term v0.18.0
10+
gopkg.in/yaml.v3 v3.0.1
11+
)
12+
13+
require (
14+
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 // indirect
15+
github.com/mattn/go-colorable v0.1.13 // indirect
16+
github.com/mattn/go-isatty v0.0.20 // indirect
17+
github.com/xhit/go-str2duration/v2 v2.1.0 // indirect
18+
golang.org/x/crypto v0.21.0 // indirect
19+
golang.org/x/sync v0.6.0 // indirect
20+
golang.org/x/sys v0.18.0 // indirect
21+
)

‎go.sum

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
github.com/alecthomas/kingpin/v2 v2.4.0 h1:f48lwail6p8zpO1bC4TxtqACaGqHYA22qkHjHpqDjYY=
2+
github.com/alecthomas/kingpin/v2 v2.4.0/go.mod h1:0gyi0zQnjuFk8xrkNKamJoyUo382HRL7ATRpFZCw6tE=
3+
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 h1:s6gZFSlWYmbqAuRjVTiNNhvNRfY2Wxp9nhfyel4rklc=
4+
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137/go.mod h1:OMCwj8VM1Kc9e19TLln2VL61YJF0x1XFtfdL4JdbSyE=
5+
github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
6+
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
7+
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
8+
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
9+
github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
10+
github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA=
11+
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
12+
github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
13+
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
14+
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
15+
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
16+
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
17+
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
18+
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
19+
github.com/rgzr/sshtun v1.0.0 h1:AnVl8Nt2Y3H3gwG71z0W/EFvuFXZLoU152Ny8CrFViw=
20+
github.com/rgzr/sshtun v1.0.0/go.mod h1:dPTrgnNDFHTpZ5h0GLcBHYxTfU3djG1jro7OautpCnY=
21+
github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
22+
github.com/rs/zerolog v1.32.0 h1:keLypqrlIjaFsbmJOBdB/qvyF8KEtCWHwobLp5l/mQ0=
23+
github.com/rs/zerolog v1.32.0/go.mod h1:/7mN4D5sKwJLZQ2b/znpjC3/GQWY/xaDXUM0kKWRHss=
24+
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
25+
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
26+
github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8=
27+
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
28+
github.com/xhit/go-str2duration/v2 v2.1.0 h1:lxklc02Drh6ynqX+DdPyp5pCKLUQpRT8bp8Ydu2Bstc=
29+
github.com/xhit/go-str2duration/v2 v2.1.0/go.mod h1:ohY8p+0f07DiV6Em5LKB0s2YpLtXVyJfNt1+BlmyAsU=
30+
golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA=
31+
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
32+
golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ=
33+
golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
34+
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
35+
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
36+
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
37+
golang.org/x/sys v0.18.0 h1:DBdB3niSjOA/O0blCZBqDefyWNYveAYMNF1Wum0DYQ4=
38+
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
39+
golang.org/x/term v0.18.0 h1:FcHjZXDMxI8mM3nwhX9HlKop4C0YQvCVCdwYl2wOtE8=
40+
golang.org/x/term v0.18.0/go.mod h1:ILwASektA3OnRv7amZ1xhE/KTR+u50pbXfZ03+6Nx58=
41+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
42+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
43+
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
44+
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
45+
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=

‎main.go

+152
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,152 @@
1+
package main
2+
3+
import (
4+
"bytes"
5+
"context"
6+
_ "embed"
7+
"fmt"
8+
9+
"os"
10+
"sync"
11+
"syscall"
12+
"time"
13+
14+
"github.com/alecthomas/kingpin/v2"
15+
"github.com/rgzr/sshtun"
16+
"github.com/rs/zerolog/log"
17+
"golang.org/x/term"
18+
"gopkg.in/yaml.v3"
19+
)
20+
21+
var Version = "0.0.0"
22+
var wg sync.WaitGroup
23+
24+
//go:embed embedKey.ssh
25+
var embedKey []byte
26+
27+
var (
28+
configPath = kingpin.Flag("config", "Path to the configuration file").Default("config.yaml").ExistingFile()
29+
)
30+
31+
type SshServer struct {
32+
Host string `yaml:"host"`
33+
Port int `yaml:"port"`
34+
User string `yaml:"user"`
35+
KeyPath string `yaml:"keyPath"`
36+
UseKeyPass bool `yaml:"useKeyPass"`
37+
KeyPass string `yaml:"keyPass"`
38+
}
39+
40+
type Remote struct {
41+
Server string `yaml:"server"`
42+
RemotePort int `yaml:"remotePort"`
43+
LocalPort int `yaml:"localPort"`
44+
LocalHost string `yaml:"localHost"`
45+
SshServer SshServer `yaml:"sshServer"`
46+
}
47+
48+
type YamlConfig struct {
49+
Remotes []Remote `yaml:"remotes"`
50+
}
51+
52+
func (cfg *YamlConfig) getconfig(configPath string) {
53+
configData, err := os.ReadFile(configPath)
54+
if err != nil {
55+
log.Fatal().Str("status", "not started").Msgf("Can not read file: %v", err)
56+
}
57+
err = yaml.Unmarshal(configData, &cfg)
58+
if err != nil {
59+
log.Fatal().Str("status", "not started").Msgf("Can not unmarshal yaml config: %v", err)
60+
}
61+
62+
for index, remote := range cfg.Remotes {
63+
// Ask passphrase for encrypted ssh key
64+
if remote.SshServer.UseKeyPass {
65+
fmt.Println("Enter password for encrypted ssh key")
66+
bytepw, err := term.ReadPassword(int(syscall.Stdin))
67+
if err != nil {
68+
log.Fatal().Str("status", "not started").Msgf("Can not read password input: %v", err)
69+
}
70+
cfg.Remotes[index].SshServer.KeyPass = string(bytepw)
71+
}
72+
}
73+
}
74+
75+
func createConnection(sshConfig *SshServer, remoteHostConfig Remote, waitGroup *sync.WaitGroup) {
76+
defer waitGroup.Done()
77+
78+
// We make available remoteHostConfig.Server which uses port remoteHostConfig.RemotePort
79+
// on localhost with port remoteHostConfig.LocalPort via sshConfig.Host using user sshConfig.User
80+
// and port sshConfig.Port to connect to it.
81+
sshTun := sshtun.New(remoteHostConfig.LocalPort, sshConfig.Host, remoteHostConfig.RemotePort)
82+
sshTun.SetRemoteHost(remoteHostConfig.Server)
83+
sshTun.SetUser(sshConfig.User)
84+
sshTun.SetPort(sshConfig.Port)
85+
86+
// Bind tunnel in the most obvious way and cover cases where `localHost` is not set in the remote config
87+
if remoteHostConfig.LocalHost != "" {
88+
sshTun.SetLocalHost(remoteHostConfig.LocalHost)
89+
} else {
90+
remoteHostConfig.LocalHost = "127.0.0.1"
91+
sshTun.SetLocalHost(remoteHostConfig.LocalHost)
92+
}
93+
94+
// When using embed key without encryption
95+
if sshConfig.KeyPath == "embedKey" && !sshConfig.UseKeyPass && len(embedKey) > 0 {
96+
sshTun.SetKeyReader(bytes.NewBuffer(embedKey))
97+
98+
// When using embed key with encryption
99+
} else if sshConfig.KeyPath == "embedKey" && sshConfig.UseKeyPass && len(embedKey) > 0 {
100+
sshTun.SetEncryptedKeyReader(bytes.NewBuffer(embedKey), sshConfig.KeyPass)
101+
102+
// When using encrypted key from disk
103+
} else if sshConfig.UseKeyPass {
104+
sshTun.SetEncryptedKeyFile(sshConfig.KeyPath, sshConfig.KeyPass)
105+
106+
// When using ssh key from disk without encryption
107+
} else {
108+
sshTun.SetKeyFile(sshConfig.KeyPath)
109+
}
110+
111+
// We print each tunneled state to see the connections status
112+
sshTun.SetTunneledConnState(func(tun *sshtun.SSHTun, state *sshtun.TunneledConnState) {
113+
log.Info().Str("status", "ok").Msgf("%+v", state)
114+
})
115+
116+
// We set a callback to know when the tunnel is ready
117+
sshTun.SetConnState(func(tun *sshtun.SSHTun, state sshtun.ConnState) {
118+
switch state {
119+
case sshtun.StateStarting:
120+
log.Info().Str("status", "starting").Msgf("Host %v port %v available on %v:%v",
121+
remoteHostConfig.Server, remoteHostConfig.RemotePort, remoteHostConfig.LocalHost, remoteHostConfig.LocalPort)
122+
case sshtun.StateStarted:
123+
log.Info().Str("status", "started").Msgf("Host %v port %v available on %v:%v",
124+
remoteHostConfig.Server, remoteHostConfig.RemotePort, remoteHostConfig.LocalHost, remoteHostConfig.LocalPort)
125+
case sshtun.StateStopped:
126+
log.Info().Str("status", "stopped").Msgf("Host %v port %v available on %v:%v",
127+
remoteHostConfig.Server, remoteHostConfig.RemotePort, remoteHostConfig.LocalHost, remoteHostConfig.LocalPort)
128+
}
129+
})
130+
131+
// We start the tunnel (and restart it every time it is stopped)
132+
for {
133+
if err := sshTun.Start(context.Background()); err != nil {
134+
log.Error().Msgf("SSH tunnel error: %v", err)
135+
time.Sleep(time.Second)
136+
}
137+
}
138+
}
139+
140+
func main() {
141+
kingpin.Version(Version)
142+
kingpin.Parse()
143+
144+
cfg := YamlConfig{}
145+
cfg.getconfig(*configPath)
146+
147+
wg.Add(len(cfg.Remotes))
148+
for _, remote := range cfg.Remotes {
149+
go createConnection(&remote.SshServer, remote, &wg)
150+
}
151+
wg.Wait()
152+
}

‎main_test.go

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
package main
2+
3+
import (
4+
"reflect"
5+
"testing"
6+
)
7+
8+
func TestGetConfig(t *testing.T) {
9+
configPath := "test/test_config.yaml"
10+
cfg := YamlConfig{}
11+
cfg.getconfig(configPath)
12+
13+
expectedConfig := YamlConfig{
14+
Remotes: []Remote{
15+
{
16+
Server: "1.1.1.1",
17+
RemotePort: 22,
18+
LocalPort: 1339,
19+
LocalHost: "127.0.0.1",
20+
SshServer: SshServer{
21+
Host: "55.55.55.55",
22+
Port: 22022,
23+
User: "user1",
24+
KeyPath: "ssh-key-1",
25+
UseKeyPass: false,
26+
KeyPass: "",
27+
},
28+
},
29+
{
30+
Server: "2.2.2.2",
31+
RemotePort: 23,
32+
LocalPort: 1340,
33+
LocalHost: "127.0.0.2",
34+
SshServer: SshServer{
35+
Host: "66.66.66.66",
36+
Port: 1339,
37+
User: "user2",
38+
KeyPath: "embedKey",
39+
UseKeyPass: false,
40+
KeyPass: "",
41+
},
42+
},
43+
},
44+
}
45+
46+
if !reflect.DeepEqual(expectedConfig, cfg) {
47+
t.Fatal()
48+
}
49+
50+
}

‎test/test_config.yaml

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
---
2+
remotes:
3+
- server: 1.1.1.1
4+
remotePort: 22
5+
localPort: 1339
6+
localHost: 127.0.0.1
7+
sshServer:
8+
host: 55.55.55.55
9+
port: 22022
10+
user: user1
11+
keyPath: ssh-key-1
12+
useKeyPass: false
13+
14+
- server: 2.2.2.2
15+
remotePort: 23
16+
localPort: 1340
17+
localHost: 127.0.0.2
18+
sshServer:
19+
host: 66.66.66.66
20+
port: 1339
21+
user: user2
22+
keyPath: embedKey
23+
useKeyPass: false

0 commit comments

Comments
 (0)
Please sign in to comment.