Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in
Toggle navigation
K
koko
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
ops
koko
Commits
63743fc2
Commit
63743fc2
authored
Aug 01, 2019
by
Eric
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
make ssh reuse connection configurable
parent
c28bcc29
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
131 additions
and
139 deletions
+131
-139
config.go
pkg/config/config.go
+2
-0
proxy.go
pkg/proxy/proxy.go
+13
-25
connmanager.go
pkg/srvconn/connmanager.go
+82
-59
sftpconn.go
pkg/srvconn/sftpconn.go
+14
-10
sshconn.go
pkg/srvconn/sshconn.go
+20
-45
No files found.
pkg/config/config.go
View file @
63743fc2
...
@@ -26,6 +26,7 @@ type Config struct {
...
@@ -26,6 +26,7 @@ type Config struct {
MaxIdleTime
time
.
Duration
`json:"SECURITY_MAX_IDLE_TIME"`
MaxIdleTime
time
.
Duration
`json:"SECURITY_MAX_IDLE_TIME"`
SftpRoot
string
`json:"TERMINAL_SFTP_ROOT" yaml:"SFTP_ROOT"`
SftpRoot
string
`json:"TERMINAL_SFTP_ROOT" yaml:"SFTP_ROOT"`
ShowHiddenFile
bool
`yaml:"SFTP_SHOW_HIDDEN_FILE"`
ShowHiddenFile
bool
`yaml:"SFTP_SHOW_HIDDEN_FILE"`
ReuseConnection
bool
`yaml:"REUSE_CONNECTION"`
Name
string
`yaml:"NAME"`
Name
string
`yaml:"NAME"`
SecretKey
string
`yaml:"SECRET_KEY"`
SecretKey
string
`yaml:"SECRET_KEY"`
HostKeyFile
string
`yaml:"HOST_KEY_FILE"`
HostKeyFile
string
`yaml:"HOST_KEY_FILE"`
...
@@ -131,6 +132,7 @@ var Conf = &Config{
...
@@ -131,6 +132,7 @@ var Conf = &Config{
UploadFailedReplay
:
true
,
UploadFailedReplay
:
true
,
SftpRoot
:
"/tmp"
,
SftpRoot
:
"/tmp"
,
ShowHiddenFile
:
false
,
ShowHiddenFile
:
false
,
ReuseConnection
:
true
,
}
}
func
SetConf
(
conf
*
Config
)
{
func
SetConf
(
conf
*
Config
)
{
...
...
pkg/proxy/proxy.go
View file @
63743fc2
...
@@ -4,6 +4,7 @@ import (
...
@@ -4,6 +4,7 @@ import (
"fmt"
"fmt"
"regexp"
"regexp"
"strings"
"strings"
"sync"
"time"
"time"
"github.com/jumpserver/koko/pkg/config"
"github.com/jumpserver/koko/pkg/config"
...
@@ -93,19 +94,18 @@ func (p *ProxyServer) validatePermission() bool {
...
@@ -93,19 +94,18 @@ func (p *ProxyServer) validatePermission() bool {
}
}
// getSSHConn 获取ssh连接
// getSSHConn 获取ssh连接
func
(
p
*
ProxyServer
)
getSSHConn
(
fromCache
...
bool
)
(
srvConn
*
srvconn
.
ServerSSHConnection
,
err
error
)
{
func
(
p
*
ProxyServer
)
getSSHConn
()
(
srvConn
*
srvconn
.
ServerSSHConnection
,
err
error
)
{
pty
:=
p
.
UserConn
.
Pty
()
pty
:=
p
.
UserConn
.
Pty
()
conf
:=
config
.
GetConf
()
srvConn
=
&
srvconn
.
ServerSSHConnection
{
srvConn
=
&
srvconn
.
ServerSSHConnection
{
User
:
p
.
User
,
User
:
p
.
User
,
Asset
:
p
.
Asset
,
Asset
:
p
.
Asset
,
SystemUser
:
p
.
SystemUser
,
SystemUser
:
p
.
SystemUser
,
Overtime
:
time
.
Duration
(
config
.
GetConf
()
.
SSHTimeout
)
*
time
.
Second
,
Overtime
:
conf
.
SSHTimeout
*
time
.
Second
,
}
ReuseConnection
:
conf
.
ReuseConnection
,
if
len
(
fromCache
)
>
0
&&
fromCache
[
0
]
{
CloseOnce
:
new
(
sync
.
Once
),
err
=
srvConn
.
TryConnectFromCache
(
pty
.
Window
.
Height
,
pty
.
Window
.
Width
,
pty
.
Term
)
}
}
else
{
err
=
srvConn
.
Connect
(
pty
.
Window
.
Height
,
pty
.
Window
.
Width
,
pty
.
Term
)
err
=
srvConn
.
Connect
(
pty
.
Window
.
Height
,
pty
.
Window
.
Width
,
pty
.
Term
)
}
return
return
}
}
...
@@ -127,14 +127,6 @@ func (p *ProxyServer) getTelnetConn() (srvConn *srvconn.ServerTelnetConnection,
...
@@ -127,14 +127,6 @@ func (p *ProxyServer) getTelnetConn() (srvConn *srvconn.ServerTelnetConnection,
return
return
}
}
// getServerConnFromCache 从cache中获取ssh server连接
func
(
p
*
ProxyServer
)
getServerConnFromCache
()
(
srvConn
srvconn
.
ServerConnection
,
err
error
)
{
if
p
.
SystemUser
.
Protocol
==
"ssh"
{
srvConn
,
err
=
p
.
getSSHConn
(
true
)
}
return
}
// getServerConn 获取获取server连接
// getServerConn 获取获取server连接
func
(
p
*
ProxyServer
)
getServerConn
()
(
srvConn
srvconn
.
ServerConnection
,
err
error
)
{
func
(
p
*
ProxyServer
)
getServerConn
()
(
srvConn
srvconn
.
ServerConnection
,
err
error
)
{
err
=
p
.
getSystemUserUsernameIfNeed
()
err
=
p
.
getSystemUserUsernameIfNeed
()
...
@@ -154,7 +146,7 @@ func (p *ProxyServer) getServerConn() (srvConn srvconn.ServerConnection, err err
...
@@ -154,7 +146,7 @@ func (p *ProxyServer) getServerConn() (srvConn srvconn.ServerConnection, err err
if
p
.
SystemUser
.
Protocol
==
"telnet"
{
if
p
.
SystemUser
.
Protocol
==
"telnet"
{
return
p
.
getTelnetConn
()
return
p
.
getTelnetConn
()
}
else
{
}
else
{
return
p
.
getSSHConn
(
false
)
return
p
.
getSSHConn
()
}
}
}
}
...
@@ -220,11 +212,7 @@ func (p *ProxyServer) Proxy() {
...
@@ -220,11 +212,7 @@ func (p *ProxyServer) Proxy() {
if
!
p
.
preCheckRequisite
()
{
if
!
p
.
preCheckRequisite
()
{
return
return
}
}
// 先从cache中获取srv连接, 如果没有获得,则连接
srvConn
,
err
:=
p
.
getServerConn
()
srvConn
,
err
:=
p
.
getServerConnFromCache
()
if
err
!=
nil
||
srvConn
==
nil
{
srvConn
,
err
=
p
.
getServerConn
()
}
// 连接后端服务器失败
// 连接后端服务器失败
if
err
!=
nil
{
if
err
!=
nil
{
p
.
sendConnectErrorMsg
(
err
)
p
.
sendConnectErrorMsg
(
err
)
...
...
pkg/srvconn/connmanager.go
View file @
63743fc2
...
@@ -16,9 +16,8 @@ import (
...
@@ -16,9 +16,8 @@ import (
)
)
var
(
var
(
sshClients
=
make
(
map
[
string
]
*
SSHClient
)
sshClients
=
make
(
map
[
string
]
*
SSHClient
)
clientsRefCounter
=
make
(
map
[
*
SSHClient
]
int
)
clientLock
=
new
(
sync
.
RWMutex
)
clientLock
=
new
(
sync
.
RWMutex
)
)
)
var
(
var
(
...
@@ -32,8 +31,43 @@ var (
...
@@ -32,8 +31,43 @@ var (
)
)
type
SSHClient
struct
{
type
SSHClient
struct
{
Client
*
gossh
.
Client
client
*
gossh
.
Client
Username
string
username
string
ref
int
key
string
mu
*
sync
.
RWMutex
}
func
(
s
*
SSHClient
)
refCount
()
int
{
s
.
mu
.
RLock
()
defer
s
.
mu
.
RUnlock
()
return
s
.
ref
}
func
(
s
*
SSHClient
)
increaseRef
()
{
s
.
mu
.
Lock
()
defer
s
.
mu
.
Unlock
()
s
.
ref
++
}
func
(
s
*
SSHClient
)
decreaseRef
()
{
s
.
mu
.
Lock
()
defer
s
.
mu
.
Unlock
()
s
.
ref
--
}
func
(
s
*
SSHClient
)
NewSession
()
(
*
gossh
.
Session
,
error
)
{
return
s
.
client
.
NewSession
()
}
func
(
s
*
SSHClient
)
Close
()
error
{
s
.
mu
.
Lock
()
defer
s
.
mu
.
Unlock
()
if
s
.
ref
>
1
{
return
nil
}
return
s
.
client
.
Close
()
}
}
type
SSHClientConfig
struct
{
type
SSHClientConfig
struct
{
...
@@ -150,7 +184,7 @@ func MakeConfig(asset *model.Asset, systemUser *model.SystemUser, timeout time.D
...
@@ -150,7 +184,7 @@ func MakeConfig(asset *model.Asset, systemUser *model.SystemUser, timeout time.D
}
}
}
}
}
}
if
systemUser
.
Password
==
""
&&
systemUser
.
PrivateKey
==
""
&&
systemUser
.
LoginMode
!=
model
.
LoginModeManual
{
if
systemUser
.
Password
==
""
&&
systemUser
.
PrivateKey
==
""
&&
systemUser
.
LoginMode
!=
model
.
LoginModeManual
{
info
:=
service
.
GetSystemUserAssetAuthInfo
(
systemUser
.
ID
,
asset
.
ID
)
info
:=
service
.
GetSystemUserAssetAuthInfo
(
systemUser
.
ID
,
asset
.
ID
)
systemUser
.
Password
=
info
.
Password
systemUser
.
Password
=
info
.
Password
systemUser
.
PrivateKey
=
info
.
PrivateKey
systemUser
.
PrivateKey
=
info
.
PrivateKey
...
@@ -173,78 +207,67 @@ func newClient(asset *model.Asset, systemUser *model.SystemUser, timeout time.Du
...
@@ -173,78 +207,67 @@ func newClient(asset *model.Asset, systemUser *model.SystemUser, timeout time.Du
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
return
&
SSHClient
{
Client
:
conn
,
Username
:
systemUser
.
Username
},
err
return
&
SSHClient
{
client
:
conn
,
username
:
systemUser
.
Username
,
mu
:
new
(
sync
.
RWMutex
)
},
err
}
}
func
NewClient
(
user
*
model
.
User
,
asset
*
model
.
Asset
,
systemUser
*
model
.
SystemUser
,
timeout
time
.
Duration
)
(
client
*
SSHClient
,
err
error
)
{
func
NewClient
(
user
*
model
.
User
,
asset
*
model
.
Asset
,
systemUser
*
model
.
SystemUser
,
timeout
time
.
Duration
,
client
=
GetClientFromCache
(
user
,
asset
,
systemUser
)
useCache
bool
)
(
client
*
SSHClient
,
err
error
)
{
if
client
!=
nil
{
return
client
,
nil
}
key
:=
fmt
.
Sprintf
(
"%s_%s_%s"
,
user
.
ID
,
asset
.
ID
,
systemUser
.
ID
)
key
:=
fmt
.
Sprintf
(
"%s_%s_%s"
,
user
.
ID
,
asset
.
ID
,
systemUser
.
ID
)
switch
{
case
useCache
:
client
=
getClientFromCache
(
key
)
if
client
!=
nil
{
if
systemUser
.
Username
==
""
{
systemUser
.
Username
=
client
.
username
}
logger
.
Infof
(
"Reuse connection: %s->%s@%s ref: %d"
,
user
.
Username
,
client
.
username
,
asset
.
IP
,
client
.
refCount
())
return
client
,
nil
}
}
client
,
err
=
newClient
(
asset
,
systemUser
,
timeout
)
client
,
err
=
newClient
(
asset
,
systemUser
,
timeout
)
if
err
==
nil
{
if
err
==
nil
&&
useCache
{
clientLock
.
Lock
()
setClientCache
(
key
,
client
)
sshClients
[
key
]
=
client
clientsRefCounter
[
client
]
=
1
clientLock
.
Unlock
()
}
}
return
return
}
}
func
GetClientFromCache
(
user
*
model
.
User
,
asset
*
model
.
Asset
,
systemUser
*
model
.
SystemUser
)
(
client
*
SSHClient
)
{
func
getClientFromCache
(
key
string
)
(
client
*
SSHClient
)
{
key
:=
fmt
.
Sprintf
(
"%s_%s_%s"
,
user
.
ID
,
asset
.
ID
,
systemUser
.
ID
)
clientLock
.
Lock
()
clientLock
.
Lock
()
defer
clientLock
.
Unlock
()
defer
clientLock
.
Unlock
()
client
,
ok
:=
sshClients
[
key
]
client
,
ok
:=
sshClients
[
key
]
if
!
ok
{
if
!
ok
{
return
return
nil
}
if
systemUser
.
Username
==
""
{
systemUser
.
Username
=
client
.
Username
}
}
var
u
=
user
.
Username
client
.
increaseRef
()
var
ip
=
asset
.
IP
clientsRefCounter
[
client
]
++
var
counter
=
clientsRefCounter
[
client
]
logger
.
Infof
(
"Reuse connection: %s->%s@%s ref: %d"
,
u
,
client
.
Username
,
ip
,
counter
)
return
return
}
}
func
RecycleClient
(
client
*
SSHClient
)
{
func
setClientCache
(
key
string
,
client
*
SSHClient
)
{
clientLock
.
RLock
()
counter
,
ok
:=
clientsRefCounter
[
client
]
clientLock
.
RUnlock
()
if
ok
{
if
counter
==
1
{
logger
.
Debug
(
"Recycle client: close it"
)
CloseClient
(
client
)
}
else
{
clientLock
.
Lock
()
clientsRefCounter
[
client
]
--
clientLock
.
Unlock
()
logger
.
Debugf
(
"Recycle client: ref -1: %d"
,
clientsRefCounter
[
client
])
}
}
}
func
CloseClient
(
client
*
SSHClient
)
{
clientLock
.
Lock
()
clientLock
.
Lock
()
defer
clientLock
.
Unlock
()
sshClients
[
key
]
=
client
client
.
increaseRef
()
client
.
key
=
key
clientLock
.
Unlock
()
}
delete
(
clientsRefCounter
,
client
)
func
RecycleClient
(
client
*
SSHClient
)
{
var
key
string
// 0, 1: delete Cache, close client.
for
k
,
v
:=
range
sshClients
{
// default: client ref decrease.
if
v
==
client
{
if
client
==
nil
{
key
=
k
return
break
}
}
}
if
key
!=
""
{
switch
client
.
refCount
()
{
delete
(
sshClients
,
key
)
case
0
,
1
:
clientLock
.
Lock
()
delete
(
sshClients
,
client
.
key
)
clientLock
.
Unlock
()
err
:=
client
.
Close
()
if
err
!=
nil
{
logger
.
Info
(
"Failed to close client err: "
,
err
.
Error
())
}
default
:
client
.
decreaseRef
()
}
}
_
=
client
.
Client
.
Close
()
}
}
pkg/srvconn/sftpconn.go
View file @
63743fc2
...
@@ -27,13 +27,15 @@ func NewUserSFTP(user *model.User, addr string, assets ...model.Asset) *UserSftp
...
@@ -27,13 +27,15 @@ func NewUserSFTP(user *model.User, addr string, assets ...model.Asset) *UserSftp
}
}
type
UserSftp
struct
{
type
UserSftp
struct
{
User
*
model
.
User
User
*
model
.
User
Addr
string
Addr
string
RootPath
string
RootPath
string
ShowHidden
bool
ShowHidden
bool
hosts
map
[
string
]
*
HostnameDir
// key hostname or hostname.orgName
ReuseConnection
bool
sftpClients
map
[
string
]
*
SftpConn
// key %s@%s suName hostName
Overtime
time
.
Duration
hosts
map
[
string
]
*
HostnameDir
// key hostname or hostname.orgName
sftpClients
map
[
string
]
*
SftpConn
// key %s@%s suName hostName
LogChan
chan
*
model
.
FTPLog
LogChan
chan
*
model
.
FTPLog
}
}
...
@@ -42,6 +44,8 @@ func (u *UserSftp) initial(assets []model.Asset) {
...
@@ -42,6 +44,8 @@ func (u *UserSftp) initial(assets []model.Asset) {
conf
:=
config
.
GetConf
()
conf
:=
config
.
GetConf
()
u
.
RootPath
=
conf
.
SftpRoot
u
.
RootPath
=
conf
.
SftpRoot
u
.
ShowHidden
=
conf
.
ShowHiddenFile
u
.
ShowHidden
=
conf
.
ShowHiddenFile
u
.
ReuseConnection
=
conf
.
ReuseConnection
u
.
Overtime
=
conf
.
SSHTimeout
*
time
.
Second
u
.
hosts
=
make
(
map
[
string
]
*
HostnameDir
)
u
.
hosts
=
make
(
map
[
string
]
*
HostnameDir
)
u
.
sftpClients
=
make
(
map
[
string
]
*
SftpConn
)
u
.
sftpClients
=
make
(
map
[
string
]
*
SftpConn
)
u
.
LogChan
=
make
(
chan
*
model
.
FTPLog
,
10
)
u
.
LogChan
=
make
(
chan
*
model
.
FTPLog
,
10
)
...
@@ -92,9 +96,9 @@ func (u *UserSftp) ReadDir(path string) (res []os.FileInfo, err error) {
...
@@ -92,9 +96,9 @@ func (u *UserSftp) ReadDir(path string) (res []os.FileInfo, err error) {
res
,
err
=
conn
.
client
.
ReadDir
(
realPath
)
res
,
err
=
conn
.
client
.
ReadDir
(
realPath
)
if
!
u
.
ShowHidden
{
if
!
u
.
ShowHidden
{
noHiddenFiles
:=
make
([]
os
.
FileInfo
,
0
,
len
(
res
))
noHiddenFiles
:=
make
([]
os
.
FileInfo
,
0
,
len
(
res
))
for
i
:=
0
;
i
<
len
(
res
);
i
++
{
for
i
:=
0
;
i
<
len
(
res
);
i
++
{
if
!
strings
.
HasPrefix
(
res
[
i
]
.
Name
(),
"."
)
{
if
!
strings
.
HasPrefix
(
res
[
i
]
.
Name
(),
"."
)
{
noHiddenFiles
=
append
(
noHiddenFiles
,
res
[
i
])
noHiddenFiles
=
append
(
noHiddenFiles
,
res
[
i
])
}
}
}
}
return
noHiddenFiles
,
err
return
noHiddenFiles
,
err
...
@@ -577,11 +581,11 @@ func (u *UserSftp) SendFTPLog(dataChan <-chan *model.FTPLog) {
...
@@ -577,11 +581,11 @@ func (u *UserSftp) SendFTPLog(dataChan <-chan *model.FTPLog) {
}
}
func
(
u
*
UserSftp
)
GetSftpClient
(
asset
*
model
.
Asset
,
sysUser
*
model
.
SystemUser
)
(
conn
*
SftpConn
,
err
error
)
{
func
(
u
*
UserSftp
)
GetSftpClient
(
asset
*
model
.
Asset
,
sysUser
*
model
.
SystemUser
)
(
conn
*
SftpConn
,
err
error
)
{
sshClient
,
err
:=
NewClient
(
u
.
User
,
asset
,
sysUser
,
config
.
GetConf
()
.
SSHTimeout
*
time
.
Second
)
sshClient
,
err
:=
NewClient
(
u
.
User
,
asset
,
sysUser
,
u
.
Overtime
,
u
.
ReuseConnection
)
if
err
!=
nil
{
if
err
!=
nil
{
return
return
}
}
sftpClient
,
err
:=
sftp
.
NewClient
(
sshClient
.
C
lient
)
sftpClient
,
err
:=
sftp
.
NewClient
(
sshClient
.
c
lient
)
if
err
!=
nil
{
if
err
!=
nil
{
return
return
}
}
...
...
pkg/srvconn/sshconn.go
View file @
63743fc2
package
srvconn
package
srvconn
import
(
import
(
"errors"
"io"
"io"
"sync"
"time"
"time"
gossh
"golang.org/x/crypto/ssh"
gossh
"golang.org/x/crypto/ssh"
...
@@ -11,29 +11,25 @@ import (
...
@@ -11,29 +11,25 @@ import (
)
)
type
ServerSSHConnection
struct
{
type
ServerSSHConnection
struct
{
User
*
model
.
User
User
*
model
.
User
Asset
*
model
.
Asset
Asset
*
model
.
Asset
SystemUser
*
model
.
SystemUser
SystemUser
*
model
.
SystemUser
Overtime
time
.
Duration
Overtime
time
.
Duration
CloseOnce
*
sync
.
Once
client
*
SSHClient
ReuseConnection
bool
session
*
gossh
.
Session
stdin
io
.
WriteCloser
client
*
SSHClient
s
tdout
io
.
Reader
s
ession
*
gossh
.
Session
closed
bool
stdin
io
.
WriteCloser
connected
bool
stdout
io
.
Reader
}
}
func
(
sc
*
ServerSSHConnection
)
Protocol
()
string
{
func
(
sc
*
ServerSSHConnection
)
Protocol
()
string
{
return
"ssh"
return
"ssh"
}
}
func
(
sc
*
ServerSSHConnection
)
Username
()
string
{
return
sc
.
client
.
Username
}
func
(
sc
*
ServerSSHConnection
)
invokeShell
(
h
,
w
int
,
term
string
)
(
err
error
)
{
func
(
sc
*
ServerSSHConnection
)
invokeShell
(
h
,
w
int
,
term
string
)
(
err
error
)
{
sess
,
err
:=
sc
.
client
.
Client
.
NewSession
()
sess
,
err
:=
sc
.
client
.
NewSession
()
if
err
!=
nil
{
if
err
!=
nil
{
return
return
}
}
...
@@ -60,32 +56,15 @@ func (sc *ServerSSHConnection) invokeShell(h, w int, term string) (err error) {
...
@@ -60,32 +56,15 @@ func (sc *ServerSSHConnection) invokeShell(h, w int, term string) (err error) {
}
}
func
(
sc
*
ServerSSHConnection
)
Connect
(
h
,
w
int
,
term
string
)
(
err
error
)
{
func
(
sc
*
ServerSSHConnection
)
Connect
(
h
,
w
int
,
term
string
)
(
err
error
)
{
sc
.
client
,
err
=
NewClient
(
sc
.
User
,
sc
.
Asset
,
sc
.
SystemUser
,
sc
.
Timeout
())
sc
.
client
,
err
=
NewClient
(
sc
.
User
,
sc
.
Asset
,
sc
.
SystemUser
,
sc
.
Timeout
()
,
sc
.
ReuseConnection
)
if
err
!=
nil
{
if
err
!=
nil
{
return
return
}
}
err
=
sc
.
invokeShell
(
h
,
w
,
term
)
if
err
!=
nil
{
return
}
sc
.
connected
=
true
return
nil
}
func
(
sc
*
ServerSSHConnection
)
TryConnectFromCache
(
h
,
w
int
,
term
string
)
(
err
error
)
{
sc
.
client
=
GetClientFromCache
(
sc
.
User
,
sc
.
Asset
,
sc
.
SystemUser
)
if
sc
.
client
==
nil
{
return
errors
.
New
(
"no client in cache"
)
}
err
=
sc
.
invokeShell
(
h
,
w
,
term
)
err
=
sc
.
invokeShell
(
h
,
w
,
term
)
if
err
!=
nil
{
if
err
!=
nil
{
RecycleClient
(
sc
.
client
)
RecycleClient
(
sc
.
client
)
return
}
}
sc
.
connected
=
true
return
return
nil
}
}
func
(
sc
*
ServerSSHConnection
)
SetWinSize
(
h
,
w
int
)
error
{
func
(
sc
*
ServerSSHConnection
)
SetWinSize
(
h
,
w
int
)
error
{
...
@@ -108,13 +87,9 @@ func (sc *ServerSSHConnection) Timeout() time.Duration {
...
@@ -108,13 +87,9 @@ func (sc *ServerSSHConnection) Timeout() time.Duration {
}
}
func
(
sc
*
ServerSSHConnection
)
Close
()
(
err
error
)
{
func
(
sc
*
ServerSSHConnection
)
Close
()
(
err
error
)
{
RecycleClient
(
sc
.
client
)
sc
.
CloseOnce
.
Do
(
func
()
{
if
sc
.
closed
||
!
sc
.
connected
{
RecycleClient
(
sc
.
client
)
return
}
})
err
=
sc
.
session
.
Close
()
return
sc
.
session
.
Close
()
if
err
!=
nil
{
return
}
return
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment