/*
 * Copyright (C) 2018 "IoT.bzh"
 * Author Sebastien Douheret <sebastien@iot.bzh>
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package xdsserver

import (
	"fmt"

	"gerrit.automotivelinux.org/gerrit/src/xds/xds-server/lib/xsapiv1"
	"github.com/syncthing/syncthing/lib/sync"
)

// Terminals Represent a XDS terminals
type Terminals struct {
	*Context
	terms map[string]*ITERMINAL
}

// Mutex to make add/delete atomic
var tmMutex = sync.NewMutex()

// TerminalsConstructor Create a new instance of Model Terminal
func TerminalsConstructor(ctx *Context) *Terminals {
	return &Terminals{
		Context: ctx,
		terms:   make(map[string]*ITERMINAL),
	}
}

// New Create a new terminal
func (t *Terminals) New(cfg xsapiv1.TerminalConfig, targetID string) (*xsapiv1.TerminalConfig, error) {

	tmMutex.Lock()
	defer tmMutex.Unlock()

	var newT ITERMINAL

	// For now, only SSH term is supported
	switch cfg.Type {
	case xsapiv1.TypeTermSSH:
		newT = NewTermSSH(t.Context, cfg, targetID)
	default:
		return nil, fmt.Errorf("terminal type not set")
	}

	termCfg := newT.GetConfig()

	t.terms[termCfg.ID] = &newT

	termCfg.Status = xsapiv1.StatusTermEnable

	// Notify terminal add
	if err := t.events.Emit(xsapiv1.EVTTargetTerminalAdd, &termCfg, ""); err != nil {
		t.Log.Errorf("WS Emit EVTTargetTerminalAdd : %v", err)
	}

	return &termCfg, nil
}

// Free a specific terminal
func (t *Terminals) Free(id string) (*xsapiv1.TerminalConfig, error) {

	tmMutex.Lock()
	defer tmMutex.Unlock()

	tc := t.Get(id)
	if tc == nil {
		return nil, fmt.Errorf("Unknown id")
	}

	if _, err := (*tc).Close(); err != nil {
		return nil, err
	}

	resTerm := (*tc).GetConfig()

	delete(t.terms, id)

	// Notify terminal state change or add
	if err := t.events.Emit(xsapiv1.EVTTargetTerminalRemove, &resTerm, ""); err != nil {
		t.Log.Errorf("WS Emit EVTTargetTerminalRemove : %v", err)
	}

	return &resTerm, nil
}

// Get returns the terminal config or nil if not existing
func (t *Terminals) Get(id string) *ITERMINAL {
	if id == "" {
		return nil
	}
	tc, exist := t.terms[id]
	if !exist {
		return nil
	}
	return tc
}

// GetConfigArr returns the config of all terminals as an array
func (t *Terminals) GetConfigArr() []xsapiv1.TerminalConfig {
	tmMutex.Lock()
	defer tmMutex.Unlock()

	return t.getConfigArrUnsafe()
}

// getConfigArrUnsafe Same as GetConfigArr without mutex protection
func (t *Terminals) getConfigArrUnsafe() []xsapiv1.TerminalConfig {
	conf := []xsapiv1.TerminalConfig{}
	for _, v := range t.terms {
		conf = append(conf, (*v).GetConfig())
	}
	return conf
}

// Open adds a new terminal
func (t *Terminals) Open(id string, sess *ClientSession) (*xsapiv1.TerminalConfig, error) {
	tc := t.Get(id)
	if tc == nil {
		return nil, fmt.Errorf("Unknown id")
	}

	if sess.IOSocket == nil {
		return nil, fmt.Errorf("Websocket not established")
	}

	term, err := (*tc).Open(sess.IOSocket, sess.ID)

	// Notify term state change
	if errEmit := t.events.Emit(xsapiv1.EVTTargetTerminalStateChange, &term, sess.ID); errEmit != nil {
		t.Log.Errorf("WS Emit EVTTargetTerminalStateChange : %v", errEmit)
	}

	return term, err
}

// Close a specific terminal
func (t *Terminals) Close(id string, sess *ClientSession) (*xsapiv1.TerminalConfig, error) {
	tc := t.Get(id)
	if tc == nil {
		return nil, fmt.Errorf("Unknown id")
	}

	term, err := (*tc).Close()

	// Notify term state change
	if errEmit := t.events.Emit(xsapiv1.EVTTargetTerminalStateChange, &term, sess.ID); errEmit != nil {
		t.Log.Errorf("WS Emit EVTTargetTerminalStateChange : %v", errEmit)
	}

	return term, err
}

// Resize a specific terminal
func (t *Terminals) Resize(id string, cols, rows uint16, sess *ClientSession) (*xsapiv1.TerminalConfig, error) {
	tmMutex.Lock()
	defer tmMutex.Unlock()

	tc := t.Get(id)
	if tc == nil {
		return nil, fmt.Errorf("Unknown id")
	}

	term, err := (*tc).Resize(cols, rows)

	// Notify term state change
	if errEmit := t.events.Emit(xsapiv1.EVTTargetTerminalStateChange, &term, sess.ID); errEmit != nil {
		t.Log.Errorf("WS Emit EVTTargetTerminalStateChange : %v", errEmit)
	}

	return term, err
}

// Signal Send a Signal a specific terminal
func (t *Terminals) Signal(id, sigName string) error {
	tmMutex.Lock()
	defer tmMutex.Unlock()

	tc := t.Get(id)
	if tc == nil {
		return fmt.Errorf("Unknown id")
	}
	return (*tc).Signal(sigName)
}