2023-05-22 00:54:49 +09:00
|
|
|
package server
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/json"
|
2023-06-11 16:04:04 +09:00
|
|
|
"errors"
|
2023-05-22 00:54:49 +09:00
|
|
|
"io"
|
2023-06-11 16:04:04 +09:00
|
|
|
"io/fs"
|
2023-05-22 00:54:49 +09:00
|
|
|
"net/http"
|
|
|
|
|
"os"
|
|
|
|
|
"path"
|
2023-05-22 15:40:01 +09:00
|
|
|
"strconv"
|
2023-05-22 00:54:49 +09:00
|
|
|
"time"
|
2023-05-22 02:13:03 +09:00
|
|
|
|
2023-05-24 15:14:04 +09:00
|
|
|
"repositories.action2quare.com/ayo/gocommon/logger"
|
2023-05-22 02:13:03 +09:00
|
|
|
"repositories.action2quare.com/ayo/houston/shared"
|
2023-05-22 00:54:49 +09:00
|
|
|
)
|
|
|
|
|
|
2023-05-24 15:14:04 +09:00
|
|
|
/*
|
|
|
|
|
*
|
|
|
|
|
현재 접속 중인 Agent 목록을 보여줍니다.
|
|
|
|
|
- http method : GET
|
|
|
|
|
*/
|
2023-05-22 00:54:49 +09:00
|
|
|
func (h *houstonHandler) GetAgents(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
enc := json.NewEncoder(w)
|
|
|
|
|
enc.Encode(h.Operation().Hosts())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *houstonHandler) GetDeploySources(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
files, err := os.ReadDir("deploys")
|
2023-06-11 16:04:04 +09:00
|
|
|
if err != nil {
|
|
|
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
|
|
|
err = os.MkdirAll("deploys", os.ModePerm)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-22 00:54:49 +09:00
|
|
|
if err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
getVersions := func(name string) []string {
|
|
|
|
|
var out []string
|
|
|
|
|
files, _ := os.ReadDir(path.Join("deploys", name))
|
|
|
|
|
for _, fd := range files {
|
|
|
|
|
if fd.IsDir() {
|
|
|
|
|
out = append(out, fd.Name())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return out
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
out := make(map[string][]string)
|
|
|
|
|
for _, fd := range files {
|
|
|
|
|
if fd.IsDir() {
|
|
|
|
|
out[fd.Name()] = getVersions(fd.Name())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
enc := json.NewEncoder(w)
|
|
|
|
|
enc.Encode(out)
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-23 13:54:25 +09:00
|
|
|
func (h *houstonHandler) UploadDeploySource(w http.ResponseWriter, r *http.Request) {
|
2023-05-22 00:54:49 +09:00
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="file" name="file">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// <input type="submit" value="업로드">
|
|
|
|
|
// </form>
|
|
|
|
|
file, header, err := r.FormFile("file")
|
|
|
|
|
if err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
|
|
contents, err := io.ReadAll(file)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
version := r.FormValue("version")
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
ext := path.Ext(header.Filename)
|
|
|
|
|
|
2023-06-12 11:43:29 +09:00
|
|
|
var filename string
|
|
|
|
|
|
|
|
|
|
if version == "config" {
|
|
|
|
|
filename = path.Join("deploys", name, header.Filename)
|
|
|
|
|
} else {
|
|
|
|
|
// deploys 폴더는 파일시스템 서비스이므로 다운로드 가능
|
|
|
|
|
filename = path.Join("deploys", name, version, name+ext)
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-22 00:54:49 +09:00
|
|
|
if err = os.MkdirAll(path.Dir(filename), os.ModePerm); err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 파일 저장
|
|
|
|
|
err = os.WriteFile(filename, contents, 0644)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-30 12:12:23 +09:00
|
|
|
func (h *houstonHandler) DeleteDeploySource(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// </form>
|
|
|
|
|
version := r.FormValue("version")
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
|
|
|
|
|
if len(version) == 0 || len(name) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// deploys 폴더는 파일시스템 서비스이므로 다운로드 가능
|
|
|
|
|
targetpath := path.Join("deploys", name, version)
|
|
|
|
|
if err := os.RemoveAll(targetpath); err != nil {
|
|
|
|
|
logger.Println("deleteDeploySource failed :", err)
|
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-22 00:54:49 +09:00
|
|
|
func (h *houstonHandler) Deploy(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// <input type="text" name="targets">
|
|
|
|
|
// <input type="submit" value="업로드">
|
|
|
|
|
// </form>
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
version := r.FormValue("version")
|
|
|
|
|
traws := r.FormValue("targets")
|
|
|
|
|
|
|
|
|
|
var targets []string
|
2023-05-23 13:54:25 +09:00
|
|
|
if len(traws) > 0 {
|
|
|
|
|
if err := json.Unmarshal([]byte(traws), &targets); err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-05-22 00:54:49 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(name) == 0 || len(version) == 0 || len(targets) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
relPath := path.Join("deploys", name, version)
|
|
|
|
|
files, err := os.ReadDir(relPath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logger.Error(err)
|
2023-06-11 16:04:04 +09:00
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
2023-05-22 00:54:49 +09:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
latestTime := time.Time{}
|
|
|
|
|
var latestFilename string
|
|
|
|
|
for _, fd := range files {
|
|
|
|
|
if fd.IsDir() {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fi, _ := fd.Info()
|
|
|
|
|
if fi.ModTime().After(latestTime) {
|
|
|
|
|
latestFilename = fi.Name()
|
|
|
|
|
latestTime = fi.ModTime()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if len(latestFilename) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
h.Operation().Deploy(MakeDeployRequest(
|
|
|
|
|
shared.DeployRequest{
|
|
|
|
|
Name: name,
|
|
|
|
|
Version: version,
|
|
|
|
|
Url: path.Join(relPath, latestFilename),
|
|
|
|
|
},
|
|
|
|
|
targets,
|
|
|
|
|
))
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-30 12:12:23 +09:00
|
|
|
func (h *houstonHandler) Undeploy(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// <input type="text" name="targets">
|
|
|
|
|
// </form>
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
version := r.FormValue("version")
|
|
|
|
|
traws := r.FormValue("targets")
|
|
|
|
|
|
|
|
|
|
var targets []string
|
|
|
|
|
if len(traws) > 0 {
|
|
|
|
|
if err := json.Unmarshal([]byte(traws), &targets); err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(name) == 0 || len(version) == 0 || len(targets) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
h.Operation().Withdraw(MakeWithdrawRequest(
|
|
|
|
|
shared.WithdrawRequest{
|
|
|
|
|
Name: name,
|
|
|
|
|
Version: version,
|
|
|
|
|
},
|
|
|
|
|
targets,
|
|
|
|
|
))
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-22 00:54:49 +09:00
|
|
|
func (h *houstonHandler) StartProcess(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// <input type="text" name="args">
|
|
|
|
|
// <input type="text" name="targets">
|
|
|
|
|
// <input type="submit" value="업로드">
|
|
|
|
|
// </form>
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
version := r.FormValue("version")
|
|
|
|
|
args := r.FormValue("args")
|
|
|
|
|
traws := r.FormValue("targets")
|
|
|
|
|
|
|
|
|
|
var targets []string
|
2023-05-23 13:54:25 +09:00
|
|
|
if len(traws) > 0 {
|
|
|
|
|
if err := json.Unmarshal([]byte(traws), &targets); err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-05-22 00:54:49 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(name) == 0 || len(version) == 0 || len(targets) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
h.Operation().StartProcess(MakeStartProcessRequest(shared.StartProcessRequest{
|
|
|
|
|
Name: name,
|
|
|
|
|
Version: version,
|
|
|
|
|
Args: args, // 실행 파일 포함 e.g. "biglocal.exe -port=8090 -dev",
|
|
|
|
|
}, targets))
|
2023-05-22 15:40:01 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *houstonHandler) StopProcess(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// <input type="text" name="args">
|
|
|
|
|
// <input type="text" name="targets">
|
|
|
|
|
// <input type="submit" value="업로드">
|
|
|
|
|
// </form>
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
if len(name) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-05-22 00:54:49 +09:00
|
|
|
|
2023-05-22 15:40:01 +09:00
|
|
|
version := r.FormValue("version") // option
|
|
|
|
|
pidstr := r.FormValue("pid") // option
|
|
|
|
|
traws := r.FormValue("targets")
|
|
|
|
|
|
|
|
|
|
var targets []string
|
2023-05-23 13:54:25 +09:00
|
|
|
if len(traws) > 0 {
|
|
|
|
|
if err := json.Unmarshal([]byte(traws), &targets); err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-05-22 15:40:01 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pid, _ := strconv.Atoi(pidstr)
|
|
|
|
|
|
|
|
|
|
h.Operation().StopProcess(MakeStopProcessRequest(shared.StopProcessRequest{
|
|
|
|
|
Name: name,
|
|
|
|
|
Version: version,
|
|
|
|
|
Pid: int32(pid),
|
|
|
|
|
}, targets))
|
2023-05-22 00:54:49 +09:00
|
|
|
}
|
2023-05-23 13:54:25 +09:00
|
|
|
|
|
|
|
|
func (h *houstonHandler) UploadLogs(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// <input type="text" name="filter">
|
|
|
|
|
// <input type="text" name="targets">
|
|
|
|
|
// <input type="submit" value="업로드">
|
|
|
|
|
// </form>
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
if len(name) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
version := r.FormValue("version") // option
|
|
|
|
|
traws := r.FormValue("targets")
|
|
|
|
|
filter := r.FormValue("filter")
|
|
|
|
|
|
|
|
|
|
if len(filter) == 0 {
|
|
|
|
|
filter = "logs/*"
|
|
|
|
|
}
|
|
|
|
|
var targets []string
|
|
|
|
|
if err := json.Unmarshal([]byte(traws), &targets); err != nil {
|
|
|
|
|
logger.Error(err)
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
h.Operation().Upload(MakeUploadRequest(shared.UploadRequest{
|
|
|
|
|
Name: name,
|
|
|
|
|
Version: version,
|
|
|
|
|
Url: "upload",
|
|
|
|
|
Filter: filter,
|
|
|
|
|
}, targets))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *houstonHandler) GetLogFileLinks(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// <form action="/houston" method="post" enctype="multipart/form-data">
|
|
|
|
|
// <input type="text" name="name">
|
|
|
|
|
// <input type="text" name="version">
|
|
|
|
|
// </form>
|
|
|
|
|
name := r.FormValue("name")
|
|
|
|
|
version := r.FormValue("version")
|
|
|
|
|
if len(name) == 0 || len(version) == 0 {
|
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if version == "latest" {
|
|
|
|
|
version, _ = shared.FindLastestVersion(path.Join("downloads", name))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
root := path.Join("downloads", name, version)
|
|
|
|
|
logfiles, err := os.ReadDir(root)
|
|
|
|
|
if err != nil {
|
2023-06-11 16:04:04 +09:00
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
2023-05-23 13:54:25 +09:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var out []string
|
|
|
|
|
for _, lf := range logfiles {
|
|
|
|
|
out = append(out, path.Join(root, lf.Name()))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
enc := json.NewEncoder(w)
|
|
|
|
|
enc.Encode(out)
|
|
|
|
|
}
|