/*
 * ========================
 * =====    ===============
 * ======  ================
 * ======  ================
 * ======  ====   ====   ==
 * ======  ===     ==  =  =
 * ======  ===  =  ==     =
 * =  ===  ===  =  ==  ====
 * =  ===  ===  =  ==  =  =
 * ==     =====   ====   ==
 * ========================
 *
 * SPDX-License-Identifier: BSD-3-Clause
 *
 * Copyright (c) 2023-2024, Joe
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice,
 * this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 *    this list of conditions and the following disclaimer in the documentation
 *    and/or other materials provided with the distribution.
 *
 * 3. Neither the name of the organization nor the names of its
 *    contributors may be used to endorse or promote products derived from
 *    this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ''AS IS''
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 *
 * hardflip: src/c_exec.go
 * Fri Feb 02 11:44:44 2024
 * Joe
 *
 * exec the command at some point
 */

package main

import (
	"bytes"
	"os"
	"os/exec"
	"strconv"
	"strings"
)

func c_exec_cmd(cmd_fmt, cmd_env []string, silent bool) (error, string) {
	var errb bytes.Buffer
	cmd := exec.Command(cmd_fmt[0], cmd_fmt[1:]...)

	if cmd_env != nil {
		cmd.Env = os.Environ()
		cmd.Env = append(cmd.Env, cmd_env...)
	}
	if silent == false {
		cmd.Stdin  = os.Stdin
		cmd.Stdout = os.Stdout
		cmd.Stderr = os.Stderr
	} else {
		cmd.Stderr = &errb
	}
	if err := cmd.Run(); err != nil {
		return err, errb.String()
	}
	return nil, ""
}

func c_format_ssh_jump(host *HostNode) string {
		jump_fmt := "-oProxyCommand=ssh"
		if len(host.Jump.Priv) > 0 {
			jump_fmt += " -i " + host.Jump.Priv
		}
		if host.Jump.Port != 0 {
			jump_fmt += " -p " + strconv.Itoa(int(host.Jump.Port))
		}
		if len(host.Jump.User) == 0 {
			jump_fmt += " root"
		} else {
			jump_fmt += " " + host.Jump.User
		}
		jump_fmt += "@" + host.Jump.Host + " -W %h:%p"
		return jump_fmt
}

func c_format_ssh(host *HostNode, pass string) ([]string, []string) {
	cmd_fmt := []string{}
	if len(pass) > 0 {
		cmd_fmt = append(cmd_fmt, "sshpass", "-p", pass)
	}

	cmd_fmt = append(cmd_fmt, "ssh")
	if len(host.Priv) > 0 {
		cmd_fmt = append(cmd_fmt, "-i", host.Priv)
	}
	if len(host.Jump.Host) > 0 {
		cmd_fmt = append(cmd_fmt, c_format_ssh_jump(host))
	}
	if host.Port != 0 {
		cmd_fmt = append(cmd_fmt, "-p", strconv.Itoa(int(host.Port)))
	}
	cmd_fmt = append(cmd_fmt, host.User + "@" + host.Host)
	return cmd_fmt, nil
}

func c_format_rdp(host *HostNode, pass string) ([]string, []string) {
	cmd_fmt := []string{"xfreerdp"}

	cmd_fmt = append(cmd_fmt,
		"/v:" + host.Host,
		"/u:" + host.User)
	if len(host.Domain) > 0 {
		cmd_fmt = append(cmd_fmt, "/d:" + host.Domain)
	}
	if len(pass) > 0 {
		cmd_fmt = append(cmd_fmt, "/p:" + pass)
	}
	if host.Port != 0 {
		cmd_fmt = append(cmd_fmt, "/port:" + strconv.Itoa(int(host.Port)))
	}
	if host.Dynamic == true {
		cmd_fmt = append(cmd_fmt, "/dynamic-resolution")
	}
	if host.Drive != nil {
		for share, path := range host.Drive {
			cmd_fmt = append(cmd_fmt, "/drive:" + share + "," + path)
		}
	}
	if host.Quality == 0 {
		cmd_fmt = append(cmd_fmt,
			"-aero", "-menu-anims", "-window-drag", "-wallpaper",
			"-decorations", "-fonts", "-themes",
			"/bpp:8", "/compression-level:2")
	} else if host.Quality == 1 {
	} else {
		cmd_fmt = append(cmd_fmt,
			"+aero", "+menu-anims", "+window-drag",
			"+decorations", "+fonts", "+themes", "/gfx:RFX", "/rfx", "/gdi:hw",
			"/bpp:32")
	}
	cmd_fmt = append(cmd_fmt,
		"/size:" + strconv.Itoa(int(host.Width)) +
		"x" + strconv.Itoa(int(host.Height)))
	return cmd_fmt, nil
}

func c_format_openstack(host *HostNode, pass string) ([]string, []string) {
	cmd_fmt := []string{"openstack"}
	cmd_env := []string{
		"OS_USERNAME="             + host.User,
		"OS_PASSWORD="             + pass,
		"OS_AUTH_URL="             + host.Host,
		"OS_USER_DOMAIN_ID="       + host.Stack.UserDomainID, 
		"OS_PROJECT_ID="           + host.Stack.ProjectID, 
		"OS_IDENTITY_API_VERSION=" + host.Stack.IdentityAPI, 
		"OS_IMAGE_API_VERSION="    + host.Stack.ImageAPI, 
		"OS_NETWORK_API_VERSION="  + host.Stack.NetworkAPI, 
		"OS_VOLUME_API_VERSION="   + host.Stack.VolumeAPI, 
		"OS_REGION_NAME="          + host.Stack.RegionName, 
		"OS_ENDPOINT_TYPE="        + host.Stack.EndpointType, 
		"OS_INTERFACE="            + host.Stack.Interface,
	}
	return cmd_fmt, cmd_env
}

func c_format_command(host *HostNode, pass string) ([]string, []string){
	return append(host.Shell, host.Host), nil
}

func c_format_cmd(host *HostNode, opts HardOpts,
				  ui *HardUI) ([]string, []string) {
	type format_func func(*HostNode, string) ([]string, []string)
	var pass string
	gpg, term := opts.GPG, opts.Term

	if host.Protocol > PROTOCOL_MAX {
		return nil, nil
	}
	if len(gpg) > 0 && gpg != "plain" && len(host.Pass) > 0 {
		i_draw_msg(ui.s, 1, ui.style[STYLE_BOX], ui.dim, " GnuPG ")
		text := "decryption using gpg..."
		left, right := i_left_right(len(text), ui)
		i_draw_text(ui.s, left, ui.dim[H] - 3, right, ui.dim[H] - 3,
			ui.style[STYLE_DEF], text)
		ui.s.Show()
		var err error
		pass, err = c_decrypt_str(host.Pass)
		if err != nil {
			c_error_mode(host.Parent.path() + host.Filename +
				": password decryption failed", err, ui)
			return nil, nil
		}
		pass = strings.TrimSuffix(pass, "\n")
	}
	fp := [PROTOCOL_MAX + 1]format_func{
		c_format_ssh,
		c_format_rdp,
		c_format_command,
		c_format_openstack,
	}
	cmd_fmt, cmd_env := fp[host.Protocol](host, pass)
	if len(term) > 0 {
		// TODO: setsid
		if term == "$TERMINAL" {
			term = os.Getenv("TERMINAL")
		}
		cmd_fmt = append([]string{"setsid", term, "-e"}, cmd_fmt...)
	}
	return cmd_fmt, cmd_env
}

func c_exec(host *HostNode, opts HardOpts, ui *HardUI) {
	if host == nil {
		return
	}
	cmd_fmt, cmd_env := c_format_cmd(host, opts, ui)
	if cmd_fmt == nil {
		return
	}
	silent := false
	if host.Protocol == 2 {
		silent = host.Silent
	}
	if silent == false {
		if err := ui.s.Suspend(); err != nil {
			c_error_mode("screen", err, ui)
			return
		}
	} else {
		i_draw_msg(ui.s, 1, ui.style[STYLE_BOX], ui.dim, " Exec ")
		text := "running command..."
		left, right := i_left_right(len(text), ui)
		i_draw_text(ui.s, left, ui.dim[H] - 3, right, ui.dim[H] - 3,
			ui.style[STYLE_DEF], text)
		ui.s.Show()
	}
	if err, err_str := c_exec_cmd(cmd_fmt, cmd_env, silent);
	   err != nil && host.Protocol == 2 {
		c_error_mode(err_str, err, ui)
	}
	if opts.Loop == false {
		ui.s.Fini()
		os.Exit(0)
	} else if silent == false {
		if err := ui.s.Resume(); err != nil {
			ui.s.Fini()
			c_die("view", err)
		}
	}
}