2023-10-17 06:31:13 +02:00
|
|
|
// Copyright 2023, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2022-07-07 02:16:45 +02:00
|
|
|
package shexec
|
|
|
|
|
|
|
|
import (
|
2022-09-16 21:27:14 +02:00
|
|
|
"context"
|
2022-07-07 02:16:45 +02:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os/exec"
|
|
|
|
"time"
|
|
|
|
|
2023-10-16 22:25:53 +02:00
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/base"
|
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/packet"
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
"golang.org/x/crypto/ssh"
|
2022-09-26 22:02:34 +02:00
|
|
|
"golang.org/x/mod/semver"
|
2022-07-07 02:16:45 +02:00
|
|
|
)
|
|
|
|
|
2022-07-08 07:45:14 +02:00
|
|
|
// TODO - track buffer sizes for sending input
|
|
|
|
|
2022-10-22 23:45:31 +02:00
|
|
|
const NotFoundVersion = "v0.0"
|
|
|
|
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
type CmdWrap struct {
|
|
|
|
Cmd *exec.Cmd
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) Kill() {
|
|
|
|
cw.Cmd.Process.Kill()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) Wait() error {
|
|
|
|
return cw.Cmd.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) Sender() (*packet.PacketSender, io.WriteCloser, error) {
|
|
|
|
inputWriter, err := cw.Cmd.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("creating stdin pipe: %v", err)
|
|
|
|
}
|
|
|
|
sender := packet.MakePacketSender(inputWriter, nil)
|
|
|
|
return sender, inputWriter, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) Parser() (*packet.PacketParser, io.ReadCloser, io.ReadCloser, error) {
|
|
|
|
stdoutReader, err := cw.Cmd.StdoutPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, fmt.Errorf("creating stdout pipe: %v", err)
|
|
|
|
}
|
|
|
|
stderrReader, err := cw.Cmd.StderrPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, fmt.Errorf("creating stderr pipe: %v", err)
|
|
|
|
}
|
|
|
|
stdoutPacketParser := packet.MakePacketParser(stdoutReader, &packet.PacketParserOpts{IgnoreUntilValid: true})
|
|
|
|
stderrPacketParser := packet.MakePacketParser(stderrReader, nil)
|
|
|
|
packetParser := packet.CombinePacketParsers(stdoutPacketParser, stderrPacketParser, true)
|
|
|
|
return packetParser, stdoutReader, stderrReader, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) Start() error {
|
2024-02-29 20:37:03 +01:00
|
|
|
defer func() {
|
|
|
|
for _, extraFile := range cw.Cmd.ExtraFiles {
|
|
|
|
if extraFile != nil {
|
|
|
|
extraFile.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
return cw.Cmd.Start()
|
|
|
|
}
|
|
|
|
|
2024-02-29 20:37:03 +01:00
|
|
|
func (cw CmdWrap) StdinPipe() (io.WriteCloser, error) {
|
|
|
|
return cw.Cmd.StdinPipe()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) StdoutPipe() (io.ReadCloser, error) {
|
|
|
|
return cw.Cmd.StdoutPipe()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cw CmdWrap) StderrPipe() (io.ReadCloser, error) {
|
|
|
|
return cw.Cmd.StderrPipe()
|
|
|
|
}
|
|
|
|
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
type SessionWrap struct {
|
|
|
|
Session *ssh.Session
|
|
|
|
StartCmd string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) Kill() {
|
|
|
|
sw.Session.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) Wait() error {
|
|
|
|
return sw.Session.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) Start() error {
|
|
|
|
return sw.Session.Start(sw.StartCmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) Sender() (*packet.PacketSender, io.WriteCloser, error) {
|
|
|
|
inputWriter, err := sw.Session.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("creating stdin pipe: %v", err)
|
|
|
|
}
|
|
|
|
sender := packet.MakePacketSender(inputWriter, nil)
|
|
|
|
return sender, inputWriter, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) Parser() (*packet.PacketParser, io.ReadCloser, io.ReadCloser, error) {
|
|
|
|
stdoutReader, err := sw.Session.StdoutPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, fmt.Errorf("creating stdout pipe: %v", err)
|
|
|
|
}
|
|
|
|
stderrReader, err := sw.Session.StderrPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, fmt.Errorf("creating stderr pipe: %v", err)
|
|
|
|
}
|
|
|
|
stdoutPacketParser := packet.MakePacketParser(stdoutReader, &packet.PacketParserOpts{IgnoreUntilValid: true})
|
|
|
|
stderrPacketParser := packet.MakePacketParser(stderrReader, nil)
|
|
|
|
packetParser := packet.CombinePacketParsers(stdoutPacketParser, stderrPacketParser, true)
|
|
|
|
return packetParser, io.NopCloser(stdoutReader), io.NopCloser(stderrReader), nil
|
|
|
|
}
|
|
|
|
|
2024-02-29 20:37:03 +01:00
|
|
|
func (sw SessionWrap) StdinPipe() (io.WriteCloser, error) {
|
|
|
|
return sw.Session.StdinPipe()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) StdoutPipe() (io.ReadCloser, error) {
|
|
|
|
stdoutReader, err := sw.Session.StdoutPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return io.NopCloser(stdoutReader), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sw SessionWrap) StderrPipe() (io.ReadCloser, error) {
|
|
|
|
stderrReader, err := sw.Session.StderrPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return io.NopCloser(stderrReader), nil
|
|
|
|
}
|
|
|
|
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
type ConnInterface interface {
|
|
|
|
Kill()
|
|
|
|
Wait() error
|
|
|
|
Sender() (*packet.PacketSender, io.WriteCloser, error)
|
|
|
|
Parser() (*packet.PacketParser, io.ReadCloser, io.ReadCloser, error)
|
|
|
|
Start() error
|
2024-02-29 20:37:03 +01:00
|
|
|
StdinPipe() (io.WriteCloser, error)
|
|
|
|
StdoutPipe() (io.ReadCloser, error)
|
|
|
|
StderrPipe() (io.ReadCloser, error)
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
}
|
|
|
|
|
2022-07-07 02:16:45 +02:00
|
|
|
type ClientProc struct {
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
Cmd ConnInterface
|
2022-07-07 03:59:46 +02:00
|
|
|
InitPk *packet.InitPacketType
|
2022-07-07 02:16:45 +02:00
|
|
|
StartTs time.Time
|
|
|
|
StdinWriter io.WriteCloser
|
|
|
|
StdoutReader io.ReadCloser
|
|
|
|
StderrReader io.ReadCloser
|
|
|
|
Input *packet.PacketSender
|
|
|
|
Output *packet.PacketParser
|
|
|
|
}
|
|
|
|
|
2024-02-29 20:37:03 +01:00
|
|
|
type WaveshellLaunchError struct {
|
|
|
|
InitPk *packet.InitPacketType
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wle WaveshellLaunchError) Error() string {
|
|
|
|
if wle.InitPk.NotFound {
|
|
|
|
return "waveshell client not found"
|
|
|
|
} else if semver.MajorMinor(wle.InitPk.Version) != semver.MajorMinor(base.MShellVersion) {
|
|
|
|
return fmt.Sprintf("invalid remote waveshell version '%s', must be '=%s'", wle.InitPk.Version, semver.MajorMinor(base.MShellVersion))
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("invalid waveshell: init packet=%v", *wle.InitPk)
|
|
|
|
}
|
|
|
|
|
|
|
|
type InvalidPacketError struct {
|
|
|
|
InvalidPk *packet.PacketType
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ipe InvalidPacketError) Error() string {
|
|
|
|
if ipe.InvalidPk == nil {
|
|
|
|
return "no init packet received from waveshell client"
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("invalid packet received from waveshell client: %s", packet.AsString(*ipe.InvalidPk))
|
|
|
|
}
|
|
|
|
|
2022-10-22 23:45:31 +02:00
|
|
|
// returns (clientproc, initpk, error)
|
2024-02-29 20:37:03 +01:00
|
|
|
func MakeClientProc(ctx context.Context, ecmd ConnInterface) (*ClientProc, error) {
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
startTs := time.Now()
|
|
|
|
sender, inputWriter, err := ecmd.Sender()
|
2022-07-07 02:16:45 +02:00
|
|
|
if err != nil {
|
2024-02-29 20:37:03 +01:00
|
|
|
return nil, err
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
packetParser, stdoutReader, stderrReader, err := ecmd.Parser()
|
2022-07-07 02:16:45 +02:00
|
|
|
if err != nil {
|
2024-02-29 20:37:03 +01:00
|
|
|
return nil, err
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
|
|
|
err = ecmd.Start()
|
|
|
|
if err != nil {
|
2024-02-29 20:37:03 +01:00
|
|
|
return nil, fmt.Errorf("running local client: %w", err)
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
|
|
|
cproc := &ClientProc{
|
|
|
|
Cmd: ecmd,
|
|
|
|
StartTs: startTs,
|
|
|
|
StdinWriter: inputWriter,
|
|
|
|
StdoutReader: stdoutReader,
|
|
|
|
StderrReader: stderrReader,
|
|
|
|
Input: sender,
|
|
|
|
Output: packetParser,
|
|
|
|
}
|
2022-09-16 21:27:14 +02:00
|
|
|
|
|
|
|
var pk packet.PacketType
|
|
|
|
select {
|
|
|
|
case pk = <-packetParser.MainCh:
|
|
|
|
case <-ctx.Done():
|
|
|
|
cproc.Close()
|
2024-02-29 20:37:03 +01:00
|
|
|
return nil, ctx.Err()
|
2022-09-16 21:27:14 +02:00
|
|
|
}
|
2024-02-29 20:37:03 +01:00
|
|
|
if pk == nil {
|
|
|
|
cproc.Close()
|
|
|
|
return nil, InvalidPacketError{}
|
|
|
|
}
|
|
|
|
if pk.GetType() != packet.InitPacketStr {
|
|
|
|
cproc.Close()
|
|
|
|
return nil, InvalidPacketError{InvalidPk: &pk}
|
|
|
|
}
|
|
|
|
initPk := pk.(*packet.InitPacketType)
|
|
|
|
if initPk.NotFound {
|
|
|
|
cproc.Close()
|
|
|
|
return nil, WaveshellLaunchError{InitPk: initPk}
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
2024-02-29 20:37:03 +01:00
|
|
|
if semver.MajorMinor(initPk.Version) != semver.MajorMinor(base.MShellVersion) {
|
2022-07-07 02:16:45 +02:00
|
|
|
cproc.Close()
|
2024-02-29 20:37:03 +01:00
|
|
|
return nil, WaveshellLaunchError{InitPk: initPk}
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
2024-02-29 20:37:03 +01:00
|
|
|
cproc.InitPk = initPk
|
|
|
|
return cproc, nil
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (cproc *ClientProc) Close() {
|
|
|
|
if cproc.Input != nil {
|
|
|
|
cproc.Input.Close()
|
|
|
|
}
|
|
|
|
if cproc.StdinWriter != nil {
|
|
|
|
cproc.StdinWriter.Close()
|
|
|
|
}
|
|
|
|
if cproc.StdoutReader != nil {
|
|
|
|
cproc.StdoutReader.Close()
|
|
|
|
}
|
|
|
|
if cproc.StderrReader != nil {
|
|
|
|
cproc.StderrReader.Close()
|
|
|
|
}
|
|
|
|
if cproc.Cmd != nil {
|
Use ssh library for remote connections (#250)
* create proof of concept ssh library integration
This is a first attempt to integrate the golang crypto/ssh library for
handling remote connections. As it stands, this features is limited to
identity files without passphrases. It needs to be expanded to include
key+passphrase and password verifications as well.
* add password and keyboard-interactive ssh auth
This adds several new ssh auth methods. In addition to the PublicKey
method used previously, this adds password authentication,
keyboard-interactive authentication, and PublicKey+Passphrase
authentication.
Furthermore, it refactores the ssh connection code into its own wavesrv
file rather than storing int in waveshell's shexec file.
* clean up old mshell launch methods
In the debugging the addition of the ssh library, i had several versions
of the MShellProc Launch function. Since this seems mostly stable, I
have removed the old version and the experimental version in favor of
the combined version.
* allow switching between new and old ssh for dev
It is inconvenient to create milestones without being able to merge into
the main branch. But due to the experimental nature of the ssh changes,
it is not desired to use these changes in the main branch yet. This
change disables the new ssh launcher by default. It can be used by
changing the UseSshLibrary constant to true in remote.go. With this, it
becomes possible to merge these changes into the main branch without
them being used in production.
* fix: allow retry after ssh auth failure
Previously, the error status was not set when an ssh connection failed.
Because of this, an ssh connection failure would lock the failed remote
until waveterm was rebooted. This fix properly sets the error status so
this cannot happen.
2024-01-25 19:18:11 +01:00
|
|
|
cproc.Cmd.Kill()
|
2022-07-07 02:16:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-27 22:47:18 +01:00
|
|
|
func (cproc *ClientProc) ProxySingleOutput(ck base.CommandKey, sender *packet.PacketSender, packetCallback func(packet.PacketType)) {
|
2022-07-07 02:16:45 +02:00
|
|
|
sentDonePk := false
|
|
|
|
for pk := range cproc.Output.MainCh {
|
2022-11-27 22:47:18 +01:00
|
|
|
if packetCallback != nil {
|
|
|
|
packetCallback(pk)
|
|
|
|
}
|
2022-07-07 02:16:45 +02:00
|
|
|
if pk.GetType() == packet.CmdDonePacketStr {
|
|
|
|
sentDonePk = true
|
|
|
|
}
|
|
|
|
sender.SendPacket(pk)
|
|
|
|
}
|
|
|
|
exitErr := cproc.Cmd.Wait()
|
|
|
|
if !sentDonePk {
|
|
|
|
endTs := time.Now()
|
|
|
|
cmdDuration := endTs.Sub(cproc.StartTs)
|
2022-07-07 03:59:46 +02:00
|
|
|
donePacket := packet.MakeCmdDonePacket(ck)
|
2022-07-07 02:16:45 +02:00
|
|
|
donePacket.Ts = endTs.UnixMilli()
|
|
|
|
donePacket.ExitCode = GetExitCode(exitErr)
|
|
|
|
donePacket.DurationMs = int64(cmdDuration / time.Millisecond)
|
|
|
|
sender.SendPacket(donePacket)
|
|
|
|
}
|
|
|
|
}
|