feat: add home page
This commit is contained in:
parent
536483b73f
commit
e0b5fe07f0
|
@ -18,7 +18,7 @@ For best results place this behind a TLS termination that has a wildcard certifi
|
||||||
on your local machine have a ssh private and public key available:
|
on your local machine have a ssh private and public key available:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
$ export LOCAL_PORT=3000; export PRIV_KEY=~/.ssh/id_ed25519; sh -c "$(shell http --form POST example.com:2222 pub=@$(PRIV_KEY).pub)"
|
$ export LOCAL_PORT=3000; export PRIV_KEY=~/.ssh/id_ed25519; sh -c "$(shell http --form POST :2222 pub=@$(PRIV_KEY).pub)"
|
||||||
```
|
```
|
||||||
|
|
||||||
This will setup a reverse proxy on the example host that you can then use to access the local port. It will print a name unique to your ssh key.
|
This will setup a reverse proxy on the example host that you can then use to access the local port. It will print a name unique to your ssh key.
|
||||||
|
|
6
assets/bootstrap.min.css
vendored
Normal file
6
assets/bootstrap.min.css
vendored
Normal file
File diff suppressed because one or more lines are too long
1
assets/bootstrap.min.css.map
Normal file
1
assets/bootstrap.min.css.map
Normal file
File diff suppressed because one or more lines are too long
58
assets/sshfwd.css
Normal file
58
assets/sshfwd.css
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
/* Space out content a bit */
|
||||||
|
body {
|
||||||
|
padding-top: 20px;
|
||||||
|
padding-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Everything but the jumbotron gets side spacing for mobile first views */
|
||||||
|
.header,
|
||||||
|
.footer {
|
||||||
|
padding-right: 15px;
|
||||||
|
padding-left: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Custom page header */
|
||||||
|
.header {
|
||||||
|
padding-bottom: 20px;
|
||||||
|
border-bottom: 1px solid #e5e5e5;
|
||||||
|
}
|
||||||
|
/* Make the masthead heading the same height as the navigation */
|
||||||
|
.header h3 {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
line-height: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Custom page footer */
|
||||||
|
.footer {
|
||||||
|
padding-top: 19px;
|
||||||
|
color: #777;
|
||||||
|
border-top: 1px solid #e5e5e5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel-heading a {
|
||||||
|
color: white;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container-narrow > hr {
|
||||||
|
margin: 30px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.table tbody tr th {
|
||||||
|
width: 70%
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
body, .panel-body {
|
||||||
|
color: white;
|
||||||
|
background-color: #121212;
|
||||||
|
}
|
||||||
|
.table-striped > tbody > tr:nth-of-type(2n+1) {
|
||||||
|
background-color: darkslategray;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-color-scheme: light) {
|
||||||
|
|
||||||
|
}
|
19
go.mod
19
go.mod
|
@ -1,14 +1,19 @@
|
||||||
module github.com/jonlundy/sshfwd
|
module github.com/jonlundy/sshfwd
|
||||||
|
|
||||||
go 1.15
|
go 1.18
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/gliderlabs/ssh v0.3.2
|
||||||
|
github.com/soheilhy/cmux v0.1.5
|
||||||
|
github.com/wolfeidau/humanhash v1.1.0
|
||||||
|
go.uber.org/multierr v1.7.0
|
||||||
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be // indirect
|
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be // indirect
|
||||||
github.com/gliderlabs/ssh v0.3.2
|
go.uber.org/atomic v1.7.0 // indirect
|
||||||
github.com/soheilhy/cmux v0.1.5
|
|
||||||
github.com/tjarratt/babble v0.0.0-20210505082055-cbca2a4833c1
|
|
||||||
github.com/wolfeidau/humanhash v1.1.0 // indirect
|
|
||||||
go.uber.org/multierr v1.7.0
|
|
||||||
golang.org/dl v0.0.0-20210816190658-eea66df5a73d // indirect
|
|
||||||
golang.org/x/crypto v0.0.0-20210506145944-38f3c27a63bf // indirect
|
golang.org/x/crypto v0.0.0-20210506145944-38f3c27a63bf // indirect
|
||||||
|
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110 // indirect
|
||||||
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68 // indirect
|
||||||
|
golang.org/x/text v0.3.3 // indirect
|
||||||
)
|
)
|
||||||
|
|
8
go.sum
8
go.sum
|
@ -1,25 +1,24 @@
|
||||||
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be h1:9AeTilPcZAjCFIImctFaOjnTIavg87rW78vTPkQqLI8=
|
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be h1:9AeTilPcZAjCFIImctFaOjnTIavg87rW78vTPkQqLI8=
|
||||||
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be/go.mod h1:ySMOLuWl6zY27l47sB3qLNK6tF2fkHG55UZxx8oIVo4=
|
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be/go.mod h1:ySMOLuWl6zY27l47sB3qLNK6tF2fkHG55UZxx8oIVo4=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/gliderlabs/ssh v0.3.2 h1:gcfd1Aj/9RQxvygu4l3sak711f/5+VOwBw9C/7+N4EI=
|
github.com/gliderlabs/ssh v0.3.2 h1:gcfd1Aj/9RQxvygu4l3sak711f/5+VOwBw9C/7+N4EI=
|
||||||
github.com/gliderlabs/ssh v0.3.2/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0=
|
github.com/gliderlabs/ssh v0.3.2/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/soheilhy/cmux v0.1.5 h1:jjzc5WVemNEDTLwv9tlmemhC73tI08BNOIGwBOo10Js=
|
github.com/soheilhy/cmux v0.1.5 h1:jjzc5WVemNEDTLwv9tlmemhC73tI08BNOIGwBOo10Js=
|
||||||
github.com/soheilhy/cmux v0.1.5/go.mod h1:T7TcVDs9LWfQgPlPsdngu6I6QIoyIFZDDC6sNE1GqG0=
|
github.com/soheilhy/cmux v0.1.5/go.mod h1:T7TcVDs9LWfQgPlPsdngu6I6QIoyIFZDDC6sNE1GqG0=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
|
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
|
||||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/tjarratt/babble v0.0.0-20210505082055-cbca2a4833c1 h1:j8whCiEmvLCXI3scVn+YnklCU8mwJ9ZJ4/DGAKqQbRE=
|
|
||||||
github.com/tjarratt/babble v0.0.0-20210505082055-cbca2a4833c1/go.mod h1:O5hBrCGqzfb+8WyY8ico2AyQau7XQwAfEQeEQ5/5V9E=
|
|
||||||
github.com/wolfeidau/humanhash v1.1.0 h1:06KgtyyABJGBbrfMONrW7S+b5TTYVyrNB/jss5n7F3E=
|
github.com/wolfeidau/humanhash v1.1.0 h1:06KgtyyABJGBbrfMONrW7S+b5TTYVyrNB/jss5n7F3E=
|
||||||
github.com/wolfeidau/humanhash v1.1.0/go.mod h1:jkpynR1bfyfkmKEQudIC0osWKynFAoayRjzH9OJdVIg=
|
github.com/wolfeidau/humanhash v1.1.0/go.mod h1:jkpynR1bfyfkmKEQudIC0osWKynFAoayRjzH9OJdVIg=
|
||||||
go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
|
go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
|
||||||
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||||
go.uber.org/multierr v1.7.0 h1:zaiO/rmgFjbmCXdSYJWQcdvOCsthmdaHfr3Gm2Kx4Ec=
|
go.uber.org/multierr v1.7.0 h1:zaiO/rmgFjbmCXdSYJWQcdvOCsthmdaHfr3Gm2Kx4Ec=
|
||||||
go.uber.org/multierr v1.7.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak=
|
go.uber.org/multierr v1.7.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak=
|
||||||
golang.org/dl v0.0.0-20210816190658-eea66df5a73d h1:fY+sw1TVAhVSrszhxX7Ew04Y6V9Znfa8s5O1HTzTsOQ=
|
|
||||||
golang.org/dl v0.0.0-20210816190658-eea66df5a73d/go.mod h1:IUMfjQLJQd4UTqG1Z90tenwKoCX93Gn3MAQJMOSBsDQ=
|
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
||||||
golang.org/x/crypto v0.0.0-20210506145944-38f3c27a63bf h1:B2n+Zi5QeYRDAEodEu72OS36gmTWjgpXr2+cWcBW90o=
|
golang.org/x/crypto v0.0.0-20210506145944-38f3c27a63bf h1:B2n+Zi5QeYRDAEodEu72OS36gmTWjgpXr2+cWcBW90o=
|
||||||
|
@ -41,4 +40,5 @@ golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo=
|
||||||
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
|
33
layouts/main.go.tpl
Normal file
33
layouts/main.go.tpl
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
{{define "main"}}
|
||||||
|
<!DOCTYPE html>
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
{{template "meta" .}}
|
||||||
|
<title>SSH Fwd</title>
|
||||||
|
|
||||||
|
<link href="/assets/bootstrap.min.css" rel="stylesheet" integrity="sha384-1q8mTJOASx8j1Au+a5WDVnPi2lkFfwwEAa8hDDdjZlpLegxhjVME1fgjWPGmkzs7" crossorigin="anonymous">
|
||||||
|
<link href="/assets/sshfwd.css" rel="stylesheet" crossorigin="anonymous">
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<div class="container-fluid">
|
||||||
|
<div class="header clearfix">
|
||||||
|
<nav>
|
||||||
|
<ul class="nav nav-pills pull-right">
|
||||||
|
<li role="presentation"><a href="/">Home</a></li>
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
<h3 class="text-muted">SSH Fwd</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=container>
|
||||||
|
{{template "content" .}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
{{end}}
|
5
main.go
5
main.go
|
@ -2,7 +2,6 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
@ -44,7 +43,7 @@ func run(ctx context.Context) {
|
||||||
)
|
)
|
||||||
|
|
||||||
hostKeys := envMust("SSH_HOSTKEYS")
|
hostKeys := envMust("SSH_HOSTKEYS")
|
||||||
files, err := ioutil.ReadDir(hostKeys)
|
files, err := os.ReadDir(hostKeys)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -52,6 +51,8 @@ func run(ctx context.Context) {
|
||||||
opts = append(opts, ssh.HostKeyFile(filepath.Join(hostKeys, f.Name())))
|
opts = append(opts, ssh.HostKeyFile(filepath.Join(hostKeys, f.Name())))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
loadTemplates()
|
||||||
|
|
||||||
srv := &server{
|
srv := &server{
|
||||||
bindHost: envDefault("SSH_HOST", bindHost),
|
bindHost: envDefault("SSH_HOST", bindHost),
|
||||||
domainName: envDefault("SSH_DOMAIN", domainName),
|
domainName: envDefault("SSH_DOMAIN", domainName),
|
||||||
|
|
61
pages/home.go.tpl
Normal file
61
pages/home.go.tpl
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
{{template "main" .}}
|
||||||
|
|
||||||
|
{{define "meta"}}
|
||||||
|
<meta http-equiv="refresh" content="30">
|
||||||
|
{{end}}
|
||||||
|
|
||||||
|
{{define "content"}}
|
||||||
|
<h2>What is this?</h2>
|
||||||
|
|
||||||
|
<p>This is a reverse proxy service that uses SSH as the transport. It works similar to ngrok or localtunnel.me.</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
You run the service on a internet addressible host and ssh to it. Using ssh remote forwards (ie. ssh -R) the port
|
||||||
|
on the remote host will be forwared to the configured port on your local machine.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<h2>How does it work?</h2>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<ol>
|
||||||
|
<li>You add your SSH public key</li>
|
||||||
|
<li>Connect to SSH</li>
|
||||||
|
<li>???</li>
|
||||||
|
<li>Profit!</li>
|
||||||
|
</ol>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<form class="form-inline" method="POST" action="/peers/req">
|
||||||
|
<label>SSH Public Key:</label>
|
||||||
|
<div class="input-group input-group-sm">
|
||||||
|
<input class="form-control" type="text" name="pub" placeholder="ssh-key ...">
|
||||||
|
</div>
|
||||||
|
<button class="btn btn-default" type="submit">Submit</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<div class=row>
|
||||||
|
<h2>Connections</h2>
|
||||||
|
{{ with $args := . }}
|
||||||
|
{{ range $user := .Users }}
|
||||||
|
<div class="panel panel-primary">
|
||||||
|
<div class="panel-heading">
|
||||||
|
<a href="/user/{{ $user.Name }}">
|
||||||
|
{{ $user.Name }}
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<div style='float:right'>
|
||||||
|
{{ if $user.Active }}
|
||||||
|
<a href="/user/{{ $user.Name }}" class='btn btn-success'>Active</a>
|
||||||
|
{{ else }}
|
||||||
|
<a href="/user/{{ $user.Name }}" class='btn btn-danger'>Disconnected</a>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="panel-body">
|
||||||
|
<pre>ssh -T -p {{ $args.ListenPort }} {{ $user.Name }}@{{ $args.DomainName }} -R "{{ $user.BindPort }}:localhost:$LOCAL_PORT" -i $PRIV_KEY</pre>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
{{ end }}
|
146
server.go
146
server.go
|
@ -3,38 +3,49 @@ package main
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"crypto/sha256"
|
"crypto/sha256"
|
||||||
|
"embed"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io/fs"
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/http/httputil"
|
"net/http/httputil"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
"text/template"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gliderlabs/ssh"
|
"github.com/gliderlabs/ssh"
|
||||||
"github.com/wolfeidau/humanhash"
|
"github.com/wolfeidau/humanhash"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
//go:embed pages/* layouts/* assets/*
|
||||||
|
files embed.FS
|
||||||
|
templates map[string]*template.Template
|
||||||
|
)
|
||||||
|
|
||||||
type user struct {
|
type user struct {
|
||||||
name string
|
Name string
|
||||||
pubkey ssh.PublicKey
|
Pubkey ssh.PublicKey
|
||||||
bindHost string
|
BindHost string
|
||||||
bindPort uint32
|
BindPort uint32
|
||||||
ctx ssh.Context
|
ctx ssh.Context
|
||||||
proxy http.Handler
|
proxy http.Handler
|
||||||
lastLogin time.Time
|
LastLogin time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *user) Active() bool { return u.ctx != nil }
|
||||||
|
|
||||||
func (u *user) String() string {
|
func (u *user) String() string {
|
||||||
var b strings.Builder
|
var b strings.Builder
|
||||||
fmt.Fprintln(&b, "User: ", u.name)
|
fmt.Fprintln(&b, "User: ", u.Name)
|
||||||
fmt.Fprintf(&b, " Ptr: %p\n", u)
|
fmt.Fprintf(&b, " Ptr: %p\n", u)
|
||||||
fmt.Fprintf(&b, " Pubkey: %x\n", u.pubkey)
|
fmt.Fprintf(&b, " Pubkey: %x\n", u.Pubkey)
|
||||||
fmt.Fprintln(&b, " Host: ", u.bindHost)
|
fmt.Fprintln(&b, " Host: ", u.BindHost)
|
||||||
fmt.Fprintln(&b, " Port: ", u.bindPort)
|
fmt.Fprintln(&b, " Port: ", u.BindPort)
|
||||||
fmt.Fprintf(&b, " Active: %t\n", u.ctx != nil)
|
fmt.Fprintf(&b, " Active: %t\n", u.ctx != nil)
|
||||||
fmt.Fprintln(&b, " LastLog:", u.lastLogin)
|
fmt.Fprintln(&b, " LastLog:", u.LastLogin)
|
||||||
return b.String()
|
return b.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -67,19 +78,19 @@ func (s *server) String() string {
|
||||||
func (srv *server) addUser(pubkey ssh.PublicKey) *user {
|
func (srv *server) addUser(pubkey ssh.PublicKey) *user {
|
||||||
u := &user{}
|
u := &user{}
|
||||||
|
|
||||||
u.lastLogin = time.Now()
|
u.LastLogin = time.Now()
|
||||||
u.name = fingerprintHuman(pubkey)
|
u.Name = fingerprintHuman(pubkey)
|
||||||
u.name = strings.ToLower(u.name)
|
u.Name = strings.ToLower(u.Name)
|
||||||
u.name = filterName.ReplaceAllString(u.name, "")
|
u.Name = filterName.ReplaceAllString(u.Name, "")
|
||||||
|
|
||||||
if g, ok := srv.users.LoadOrStore(u.name, u); ok {
|
if g, ok := srv.users.LoadOrStore(u.Name, u); ok {
|
||||||
u = g.(*user)
|
u = g.(*user)
|
||||||
return u
|
return u
|
||||||
}
|
}
|
||||||
|
|
||||||
u.pubkey = pubkey
|
u.Pubkey = pubkey
|
||||||
u.bindPort = srv.nextPort()
|
u.BindPort = srv.nextPort()
|
||||||
u.bindHost = srv.bindHost
|
u.BindHost = srv.bindHost
|
||||||
|
|
||||||
return u
|
return u
|
||||||
}
|
}
|
||||||
|
@ -87,7 +98,7 @@ func (srv *server) disconnectUser(name string) {
|
||||||
if u, ok := srv.getUserByName(name); ok {
|
if u, ok := srv.getUserByName(name); ok {
|
||||||
u.ctx = nil
|
u.ctx = nil
|
||||||
u.proxy = nil
|
u.proxy = nil
|
||||||
srv.ports.Delete(u.bindPort)
|
srv.ports.Delete(u.BindPort)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
func (srv *server) getUserByPort(port uint32) (*user, bool) {
|
func (srv *server) getUserByPort(port uint32) (*user, bool) {
|
||||||
|
@ -121,18 +132,6 @@ func (srv *server) listUsers() []*user {
|
||||||
|
|
||||||
return lis
|
return lis
|
||||||
}
|
}
|
||||||
func (srv *server) listConnectedUsers() []*user {
|
|
||||||
var lis []*user
|
|
||||||
srv.ports.Range(func(key, value interface{}) bool {
|
|
||||||
if u, ok := value.(*user); ok {
|
|
||||||
lis = append(lis, u)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
})
|
|
||||||
|
|
||||||
return lis
|
|
||||||
}
|
|
||||||
func (srv *server) nextPort() uint32 {
|
func (srv *server) nextPort() uint32 {
|
||||||
if srv.portNext < srv.portStart || srv.portNext > srv.portEnd {
|
if srv.portNext < srv.portStart || srv.portNext > srv.portEnd {
|
||||||
srv.portNext = srv.portStart
|
srv.portNext = srv.portStart
|
||||||
|
@ -160,7 +159,7 @@ func (srv *server) newSession(ctx context.Context) func(ssh.Session) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if u, ok := srv.getUserByName(s.User()); ok {
|
if u, ok := srv.getUserByName(s.User()); ok {
|
||||||
host := fmt.Sprintf("%v:%v", u.bindHost, u.bindPort)
|
host := fmt.Sprintf("%v:%v", u.BindHost, u.BindPort)
|
||||||
director := func(req *http.Request) {
|
director := func(req *http.Request) {
|
||||||
if h := req.Header.Get("X-Forwarded-Host"); h == "" {
|
if h := req.Header.Get("X-Forwarded-Host"); h == "" {
|
||||||
req.Header.Set("X-Forwarded-Host", req.Host)
|
req.Header.Set("X-Forwarded-Host", req.Host)
|
||||||
|
@ -176,7 +175,7 @@ func (srv *server) newSession(ctx context.Context) func(ssh.Session) {
|
||||||
fmt.Fprintln(s, string(requestDump))
|
fmt.Fprintln(s, string(requestDump))
|
||||||
}
|
}
|
||||||
u.proxy = &httputil.ReverseProxy{Director: director}
|
u.proxy = &httputil.ReverseProxy{Director: director}
|
||||||
fmt.Fprintf(s, "Created HTTP listener at: %v%v\n\n", u.name, srv.domainSuffix)
|
fmt.Fprintf(s, "Created HTTP listener at: %v%v\n\n", u.Name, srv.domainSuffix)
|
||||||
}
|
}
|
||||||
|
|
||||||
select {
|
select {
|
||||||
|
@ -201,11 +200,11 @@ func (srv *server) optAuthUser() []ssh.Option {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
if ssh.KeysEqual(key, u.pubkey) {
|
if ssh.KeysEqual(key, u.Pubkey) {
|
||||||
log.Println("User:", ctx.User(), "Authorized:", u.bindHost, u.bindPort, ctx.ClientVersion(), ctx.SessionID(), ctx.LocalAddr(), ctx.RemoteAddr())
|
log.Println("User:", ctx.User(), "Authorized:", u.BindHost, u.BindPort, ctx.ClientVersion(), ctx.SessionID(), ctx.LocalAddr(), ctx.RemoteAddr())
|
||||||
u.ctx = ctx
|
u.ctx = ctx
|
||||||
u.lastLogin = time.Now()
|
u.LastLogin = time.Now()
|
||||||
if _, loaded := srv.ports.LoadOrStore(u.bindPort, u); loaded {
|
if _, loaded := srv.ports.LoadOrStore(u.BindPort, u); loaded {
|
||||||
log.Println("User:", ctx.User(), "already connected!")
|
log.Println("User:", ctx.User(), "already connected!")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@ -232,11 +231,11 @@ func (srv *server) optAuthUser() []ssh.Option {
|
||||||
}
|
}
|
||||||
|
|
||||||
if u.ctx.SessionID() != ctx.SessionID() {
|
if u.ctx.SessionID() != ctx.SessionID() {
|
||||||
log.Println("Port", bindPort, "in use by", u.name, u.ctx.SessionID())
|
log.Println("Port", bindPort, "in use by", u.Name, u.ctx.SessionID())
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
if bindHost != strings.Trim(u.bindHost, "[]") || bindPort != u.bindPort {
|
if bindHost != strings.Trim(u.BindHost, "[]") || bindPort != u.BindPort {
|
||||||
log.Println("User", ctx.User(), "Not Allowed: ", bindHost, bindPort, ctx.ClientVersion(), ctx.SessionID(), ctx.LocalAddr(), ctx.RemoteAddr())
|
log.Println("User", ctx.User(), "Not Allowed: ", bindHost, bindPort, ctx.ClientVersion(), ctx.SessionID(), ctx.LocalAddr(), ctx.RemoteAddr())
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@ -282,25 +281,43 @@ func (srv *server) handleHTTP(rw http.ResponseWriter, r *http.Request) {
|
||||||
pubkey, _, _, _, err := ssh.ParseAuthorizedKey([]byte(r.FormValue("pub")))
|
pubkey, _, _, _, err := ssh.ParseAuthorizedKey([]byte(r.FormValue("pub")))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rw.WriteHeader(400)
|
rw.WriteHeader(400)
|
||||||
fmt.Fprintln(rw, "ERR READING KEY")
|
fmt.Fprintln(rw, "ERR READING KEY", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
u := srv.addUser(pubkey)
|
u := srv.addUser(pubkey)
|
||||||
rw.WriteHeader(201)
|
rw.Header().Set("Location", "/")
|
||||||
fmt.Fprintf(rw, `ssh -T -p %v %v@%v -R "%v:%v:localhost:$LOCAL_PORT" -i $PRIV_KEY`+"\n", srv.listenPort, u.name, srv.domainName, u.bindHost, u.bindPort)
|
rw.WriteHeader(http.StatusFound)
|
||||||
|
fmt.Fprintf(rw, `ssh -T -p %v %v@%v -R "%v:%v:localhost:$LOCAL_PORT" -i $PRIV_KEY`+"\n", srv.listenPort, u.Name, srv.domainName, u.BindHost, u.BindPort)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintln(rw, "Hello!")
|
// fmt.Fprintln(rw, "Hello!")
|
||||||
fmt.Fprintln(rw, srv)
|
// fmt.Fprintln(rw, srv)
|
||||||
fmt.Fprintln(rw, "Registered Users")
|
// fmt.Fprintln(rw, "Registered Users")
|
||||||
for _, u := range srv.listUsers() {
|
// for _, u := range srv.listUsers() {
|
||||||
fmt.Fprintln(rw, u)
|
// fmt.Fprintln(rw, u)
|
||||||
|
// }
|
||||||
|
|
||||||
|
// fmt.Fprintln(rw, "Connected Users")
|
||||||
|
// for _, u := range srv.listConnectedUsers() {
|
||||||
|
// fmt.Fprintln(rw, u)
|
||||||
|
// }
|
||||||
|
|
||||||
|
a, _ := fs.Sub(files, "assets")
|
||||||
|
assets := http.StripPrefix("/assets/", http.FileServer(http.FS(a)))
|
||||||
|
if strings.HasPrefix(r.URL.Path, "/assets/") {
|
||||||
|
assets.ServeHTTP(rw, r)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintln(rw, "Connected Users")
|
t := templates["home.go.tpl"]
|
||||||
for _, u := range srv.listConnectedUsers() {
|
err := t.Execute(rw, map[string]any{
|
||||||
fmt.Fprintln(rw, u)
|
"Users": srv.listUsers(),
|
||||||
|
"ListenPort": srv.listenPort,
|
||||||
|
"DomainName": srv.domainName,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -309,3 +326,32 @@ func fingerprintHuman(pubKey ssh.PublicKey) string {
|
||||||
h, _ := humanhash.Humanize(sha256sum[:], 3)
|
h, _ := humanhash.Humanize(sha256sum[:], 3)
|
||||||
return h
|
return h
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var funcMap = map[string]any{}
|
||||||
|
|
||||||
|
func loadTemplates() error {
|
||||||
|
if templates != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
templates = make(map[string]*template.Template)
|
||||||
|
tmplFiles, err := fs.ReadDir(files, "pages")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tmpl := range tmplFiles {
|
||||||
|
if tmpl.IsDir() {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
pt := template.New(tmpl.Name())
|
||||||
|
pt.Funcs(funcMap)
|
||||||
|
pt, err = pt.ParseFS(files, "pages/"+tmpl.Name(), "layouts/*.go.tpl")
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
templates[tmpl.Name()] = pt
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user