package terminal

import (
	"context"
	"encoding/base64"
	"encoding/json"
	"fmt"
	"net"
	"sync"

	"github.com/1Panel-dev/1Panel/backend/global"
	"github.com/gorilla/websocket"
	"github.com/pkg/errors"
)

type ExecWsSession struct {
	conn   net.Conn
	wsConn *websocket.Conn

	writeMutex sync.Mutex
}

func NewExecConn(cols, rows int, wsConn *websocket.Conn, hijacked net.Conn, commands ...string) (*ExecWsSession, error) {
	_, _ = hijacked.Write([]byte(fmt.Sprintf("stty cols %d rows %d && clear \r", cols, rows)))
	for _, command := range commands {
		_, _ = hijacked.Write([]byte(fmt.Sprintf("%s \r", command)))
	}

	return &ExecWsSession{
		conn:   hijacked,
		wsConn: wsConn,
	}, nil
}

func (sws *ExecWsSession) Start(ctx context.Context, quitChan chan bool) {
	go sws.handleSlaveEvent(ctx, quitChan)
	go sws.receiveWsMsg(ctx, quitChan)
}

func (sws *ExecWsSession) handleSlaveEvent(ctx context.Context, exitCh chan bool) {
	defer setQuit(exitCh)

	buffer := make([]byte, 1024)
	for {
		n, err := sws.conn.Read(buffer)
		if err != nil && errors.Is(err, net.ErrClosed) {
			return
		}

		if err := sws.masterWrite(buffer[:n]); err != nil {
			if errors.Is(err, websocket.ErrCloseSent) {
				return
			}
		}
	}
}

func (sws *ExecWsSession) masterWrite(data []byte) error {
	sws.writeMutex.Lock()
	defer sws.writeMutex.Unlock()
	err := sws.wsConn.WriteMessage(websocket.TextMessage, data)
	if err != nil {
		return errors.Wrapf(err, "failed to write to master")
	}

	return nil
}

func (sws *ExecWsSession) receiveWsMsg(ctx context.Context, exitCh chan bool) {
	wsConn := sws.wsConn
	defer setQuit(exitCh)
	for {
		_, wsData, err := wsConn.ReadMessage()
		if err != nil {
			global.LOG.Errorf("reading webSocket message failed, err: %v", err)
			return
		}
		msgObj := wsMsg{}
		_ = json.Unmarshal(wsData, &msgObj)
		switch msgObj.Type {
		case wsMsgResize:
			if msgObj.Cols > 0 && msgObj.Rows > 0 {
				sws.ResizeTerminal(msgObj.Rows, msgObj.Cols)
			}
		case wsMsgCmd:
			decodeBytes, err := base64.StdEncoding.DecodeString(msgObj.Cmd)
			if err != nil {
				global.LOG.Errorf("websock cmd string base64 decoding failed, err: %v", err)
				return
			}
			sws.sendWebsocketInputCommandToSshSessionStdinPipe(decodeBytes)
		case wsMsgClose:
			_, _ = sws.conn.Write([]byte("exit\r"))
			return
		}
	}
}

func (sws *ExecWsSession) sendWebsocketInputCommandToSshSessionStdinPipe(cmdBytes []byte) {
	_, _ = sws.conn.Write(cmdBytes)
}

func (sws *ExecWsSession) ResizeTerminal(width int, height int) {
	_, _ = sws.conn.Write([]byte(fmt.Sprintf("stty cols %d rows %d && clear \r", width, height)))
}