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
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
104 additions
and
112 deletions
+104
-112
config.go
pkg/config/config.go
+2
-0
proxy.go
pkg/proxy/proxy.go
+8
-20
connmanager.go
pkg/srvconn/connmanager.go
+76
-53
sftpconn.go
pkg/srvconn/sftpconn.go
+8
-4
sshconn.go
pkg/srvconn/sshconn.go
+10
-35
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
,
CloseOnce
:
new
(
sync
.
Once
),
}
}
if
len
(
fromCache
)
>
0
&&
fromCache
[
0
]
{
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
...
@@ -17,7 +17,6 @@ import (
...
@@ -17,7 +17,6 @@ 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
)
)
)
...
@@ -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
)
{
key
:=
fmt
.
Sprintf
(
"%s_%s_%s"
,
user
.
ID
,
asset
.
ID
,
systemUser
.
ID
)
switch
{
case
useCache
:
client
=
getClientFromCache
(
key
)
if
client
!=
nil
{
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
return
client
,
nil
}
}
}
key
:=
fmt
.
Sprintf
(
"%s_%s_%s"
,
user
.
ID
,
asset
.
ID
,
systemUser
.
ID
)
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
()
clientLock
.
Lock
()
clientsRefCounter
[
client
]
--
sshClients
[
key
]
=
client
client
.
increaseRef
()
client
.
key
=
key
clientLock
.
Unlock
()
clientLock
.
Unlock
()
logger
.
Debugf
(
"Recycle client: ref -1: %d"
,
clientsRefCounter
[
client
])
}
}
}
}
func
CloseClient
(
client
*
SSHClient
)
{
func
RecycleClient
(
client
*
SSHClient
)
{
clientLock
.
Lock
()
// 0, 1: delete Cache, close client.
defer
clientLock
.
Unlock
()
// default: client ref decrease.
if
client
==
nil
{
delete
(
clientsRefCounter
,
client
)
return
var
key
string
for
k
,
v
:=
range
sshClients
{
if
v
==
client
{
key
=
k
break
}
}
switch
client
.
refCount
()
{
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
())
}
}
if
key
!=
""
{
default
:
delete
(
sshClients
,
key
)
client
.
decreaseRef
(
)
}
}
_
=
client
.
Client
.
Close
()
}
}
pkg/srvconn/sftpconn.go
View file @
63743fc2
...
@@ -32,6 +32,8 @@ type UserSftp struct {
...
@@ -32,6 +32,8 @@ type UserSftp struct {
RootPath
string
RootPath
string
ShowHidden
bool
ShowHidden
bool
ReuseConnection
bool
Overtime
time
.
Duration
hosts
map
[
string
]
*
HostnameDir
// key hostname or hostname.orgName
hosts
map
[
string
]
*
HostnameDir
// key hostname or hostname.orgName
sftpClients
map
[
string
]
*
SftpConn
// key %s@%s suName hostName
sftpClients
map
[
string
]
*
SftpConn
// key %s@%s suName hostName
...
@@ -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"
...
@@ -15,25 +15,21 @@ type ServerSSHConnection struct {
...
@@ -15,25 +15,21 @@ type ServerSSHConnection struct {
Asset
*
model
.
Asset
Asset
*
model
.
Asset
SystemUser
*
model
.
SystemUser
SystemUser
*
model
.
SystemUser
Overtime
time
.
Duration
Overtime
time
.
Duration
CloseOnce
*
sync
.
Once
ReuseConnection
bool
client
*
SSHClient
client
*
SSHClient
session
*
gossh
.
Session
session
*
gossh
.
Session
stdin
io
.
WriteCloser
stdin
io
.
WriteCloser
stdout
io
.
Reader
stdout
io
.
Reader
closed
bool
connected
bool
}
}
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
{
return
}
err
=
sc
.
invokeShell
(
h
,
w
,
term
)
if
err
!=
nil
{
if
err
!=
nil
{
return
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
)
{
sc
.
CloseOnce
.
Do
(
func
()
{
RecycleClient
(
sc
.
client
)
RecycleClient
(
sc
.
client
)
if
sc
.
closed
||
!
sc
.
connected
{
return
})
}
return
sc
.
session
.
Close
()
err
=
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