mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-24 22:01:33 +01:00
207 lines
4.7 KiB
Go
207 lines
4.7 KiB
Go
// Copyright 2022 Dashborg Inc
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
// file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
package mpio
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"sync"
|
|
|
|
"github.com/scripthaus-dev/mshell/pkg/packet"
|
|
)
|
|
|
|
const ReadBufSize = 128 * 1024
|
|
const WriteBufSize = 128 * 1024
|
|
const MaxSingleWriteSize = 4 * 1024
|
|
|
|
type Multiplexer struct {
|
|
Lock *sync.Mutex
|
|
SessionId string
|
|
CmdId string
|
|
FdReaders map[int]*FdReader // synchronized
|
|
FdWriters map[int]*FdWriter // synchronized
|
|
CloseAfterStart []*os.File // synchronized
|
|
|
|
Sender *packet.PacketSender
|
|
Input chan packet.PacketType
|
|
Started bool
|
|
}
|
|
|
|
func MakeMultiplexer(sessionId string, cmdId string) *Multiplexer {
|
|
return &Multiplexer{
|
|
Lock: &sync.Mutex{},
|
|
SessionId: sessionId,
|
|
CmdId: cmdId,
|
|
FdReaders: make(map[int]*FdReader),
|
|
FdWriters: make(map[int]*FdWriter),
|
|
}
|
|
}
|
|
|
|
func (m *Multiplexer) Close() {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
|
|
for _, fd := range m.FdReaders {
|
|
fd.Close()
|
|
}
|
|
for _, fd := range m.FdWriters {
|
|
fd.Close()
|
|
}
|
|
for _, fd := range m.CloseAfterStart {
|
|
fd.Close()
|
|
}
|
|
}
|
|
|
|
// returns the *writer* to connect to process, reader is put in FdReaders
|
|
func (m *Multiplexer) MakeReaderPipe(fdNum int) (*os.File, error) {
|
|
pr, pw, err := os.Pipe()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
m.FdReaders[fdNum] = MakeFdReader(m, pr, fdNum)
|
|
m.CloseAfterStart = append(m.CloseAfterStart, pw)
|
|
return pw, nil
|
|
}
|
|
|
|
// returns the *reader* to connect to process, writer is put in FdWriters
|
|
func (m *Multiplexer) MakeWriterPipe(fdNum int) (*os.File, error) {
|
|
pr, pw, err := os.Pipe()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
m.FdWriters[fdNum] = MakeFdWriter(m, pw, fdNum)
|
|
m.CloseAfterStart = append(m.CloseAfterStart, pr)
|
|
return pr, nil
|
|
}
|
|
|
|
func (m *Multiplexer) makeDataAckPacket(fdNum int, ackLen int, err error) *packet.DataAckPacketType {
|
|
ack := packet.MakeDataAckPacket()
|
|
ack.SessionId = m.SessionId
|
|
ack.CmdId = m.CmdId
|
|
ack.FdNum = fdNum
|
|
ack.AckLen = ackLen
|
|
if err != nil {
|
|
ack.Error = err.Error()
|
|
}
|
|
return ack
|
|
}
|
|
|
|
func (m *Multiplexer) makeDataPacket(fdNum int, data []byte, err error) *packet.DataPacketType {
|
|
pk := packet.MakeDataPacket()
|
|
pk.SessionId = m.SessionId
|
|
pk.CmdId = m.CmdId
|
|
pk.FdNum = fdNum
|
|
pk.Data = string(data)
|
|
if err != nil {
|
|
pk.Error = err.Error()
|
|
}
|
|
return pk
|
|
}
|
|
|
|
func (m *Multiplexer) sendPacket(p packet.PacketType) {
|
|
m.Sender.SendPacket(p)
|
|
}
|
|
|
|
func (m *Multiplexer) launchWriters() {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
for _, fw := range m.FdWriters {
|
|
go fw.WriteLoop()
|
|
}
|
|
}
|
|
|
|
func (m *Multiplexer) launchReaders(wg *sync.WaitGroup) {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
wg.Add(len(m.FdReaders))
|
|
for _, fr := range m.FdReaders {
|
|
go fr.ReadLoop(wg)
|
|
}
|
|
}
|
|
|
|
func (m *Multiplexer) startIO(packetCh chan packet.PacketType, sender *packet.PacketSender) {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
if m.Started {
|
|
panic("Multiplexer is already running, cannot start again")
|
|
}
|
|
m.Input = packetCh
|
|
m.Sender = sender
|
|
m.Started = true
|
|
}
|
|
|
|
func (m *Multiplexer) runPacketInputLoop() {
|
|
for pk := range m.Input {
|
|
if pk.GetType() == packet.DataPacketStr {
|
|
dataPacket := pk.(*packet.DataPacketType)
|
|
err := m.processDataPacket(dataPacket)
|
|
if err != nil {
|
|
errPacket := m.makeDataAckPacket(dataPacket.FdNum, 0, err)
|
|
m.sendPacket(errPacket)
|
|
}
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.DataAckPacketStr {
|
|
ackPacket := pk.(*packet.DataAckPacketType)
|
|
m.processAckPacket(ackPacket)
|
|
}
|
|
// other packet types are ignored
|
|
}
|
|
}
|
|
|
|
func (m *Multiplexer) processDataPacket(dataPacket *packet.DataPacketType) error {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
fw := m.FdWriters[dataPacket.FdNum]
|
|
if fw == nil {
|
|
// add a closed FdWriter as a placeholder so we only send one error
|
|
fw := MakeFdWriter(m, nil, dataPacket.FdNum)
|
|
fw.Close()
|
|
m.FdWriters[dataPacket.FdNum] = fw
|
|
return fmt.Errorf("write to closed file")
|
|
}
|
|
err := fw.AddData([]byte(dataPacket.Data), dataPacket.Eof)
|
|
if err != nil {
|
|
fw.Close()
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *Multiplexer) processAckPacket(ackPacket *packet.DataAckPacketType) {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
fr := m.FdReaders[ackPacket.FdNum]
|
|
if fr == nil {
|
|
return
|
|
}
|
|
fr.NotifyAck(ackPacket.AckLen)
|
|
}
|
|
|
|
func (m *Multiplexer) closeTempStartFds() {
|
|
m.Lock.Lock()
|
|
defer m.Lock.Unlock()
|
|
for _, fd := range m.CloseAfterStart {
|
|
fd.Close()
|
|
}
|
|
m.CloseAfterStart = nil
|
|
}
|
|
|
|
func (m *Multiplexer) RunIOAndWait(packetCh chan packet.PacketType, sender *packet.PacketSender) {
|
|
m.startIO(packetCh, sender)
|
|
m.closeTempStartFds()
|
|
var wg sync.WaitGroup
|
|
m.launchReaders(&wg)
|
|
m.launchWriters()
|
|
go m.runPacketInputLoop()
|
|
wg.Wait()
|
|
}
|