Improve web interface and allow setting boot option

This commit is contained in:
Melora Hugues 2023-08-19 13:49:13 +02:00
parent ca59f1e25f
commit 2ae7327f6b
7 changed files with 176 additions and 111 deletions

View file

@ -1,85 +1,78 @@
package client package client
// import ( import (
// "encoding/json" "encoding/json"
// "fmt" "errors"
// "io" "fmt"
// "net" "io"
// "net/http" "net/http"
// "git.faercol.me/faercol/http-boot-server/bootserver/helpers" "git.faercol.me/faercol/http-boot-server/bootserver/helpers"
// "git.faercol.me/faercol/http-boot-server/bootserver/services" "git.faercol.me/faercol/http-boot-server/bootserver/services"
// "github.com/sirupsen/logrus" "github.com/google/uuid"
// ) "github.com/sirupsen/logrus"
)
// const BootRoute = "/boot" const SetBootRoute = "/config/boot"
// type BootController struct { type setBootOptionPayload struct {
// clientService *services.ClientHandlerService ClientID string `json:"client_id"`
// l *logrus.Logger OptionID string `json:"option_id"`
// } }
// func NewBootController(logger *logrus.Logger, service *services.ClientHandlerService) *BootController { type BootController struct {
// return &BootController{ clientService *services.ClientHandlerService
// clientService: service, l *logrus.Logger
// l: logger, }
// }
// }
// func (bc *BootController) getBootOption(clientIP string, w http.ResponseWriter, r *http.Request) (int, []byte, error) { func NewBootController(logger *logrus.Logger, service *services.ClientHandlerService) *BootController {
// bootOption, err := bc.clientService.GetClientSelectedBootOption(clientIP) return &BootController{
// if err != nil { clientService: service,
// return http.StatusInternalServerError, nil, fmt.Errorf("failed to get boot option: %w", err) l: logger,
// } }
}
// dat, err := json.Marshal(bootOption) func (bc *BootController) setBootOption(w http.ResponseWriter, r *http.Request) (int, []byte, error) {
// if err != nil { dat, err := io.ReadAll(r.Body)
// return http.StatusInternalServerError, nil, fmt.Errorf("failed to serialize body") if err != nil {
// } return http.StatusInternalServerError, nil, fmt.Errorf("failed to read body: %w", err)
}
var payload setBootOptionPayload
if err := json.Unmarshal(dat, &payload); err != nil {
return http.StatusBadRequest, nil, fmt.Errorf("failed to parse body: %w", err)
}
// w.Header().Add("Content-Type", "application/json") clientID, err := uuid.Parse(payload.ClientID)
// return http.StatusOK, dat, nil if err != nil {
// } return http.StatusBadRequest, []byte("bad client ID"), fmt.Errorf("invalid format for client ID: %w", err)
}
optionID, err := uuid.Parse(payload.OptionID)
if err != nil {
return http.StatusBadRequest, []byte("bad option ID"), fmt.Errorf("invalid format for option ID: %w", err)
}
// func (bc *BootController) setBootOption(clientIP string, w http.ResponseWriter, r *http.Request) (int, error) { if err := bc.clientService.SetClientBootOption(clientID, optionID.String()); err != nil {
// dat, err := io.ReadAll(r.Body) if errors.Is(err, services.ErrUnknownClient) {
// if err != nil { return http.StatusNotFound, []byte("unknown client"), err
// return http.StatusInternalServerError, fmt.Errorf("failed to read body: %w", err) }
// } if errors.Is(err, services.ErrUnknownBootOption) {
// var option string return http.StatusNotFound, []byte("unknown boot option"), err
// if err := json.Unmarshal(dat, &option); err != nil { }
// return http.StatusInternalServerError, fmt.Errorf("failed to parse body: %w", err) return http.StatusInternalServerError, nil, fmt.Errorf("failed to set boot option for client: %w", err)
// } }
// if err := bc.clientService.SetClientBootOption(clientIP, option); err != nil { return http.StatusAccepted, nil, nil
// return http.StatusInternalServerError, fmt.Errorf("failed to set boot option for client: %w", err) }
// }
// return http.StatusAccepted, nil func (bc *BootController) ServeHTTP(w http.ResponseWriter, r *http.Request) {
// } if r.Method != http.MethodPut {
helpers.HandleResponse(w, r, http.StatusMethodNotAllowed, nil, bc.l)
return
}
// func (bc *BootController) ServeHTTP(w http.ResponseWriter, r *http.Request) { returncode, content, err := bc.setBootOption(w, r)
// clientIP, _, err := net.SplitHostPort(r.RemoteAddr) if err != nil {
// if err != nil { bc.l.Errorf("Error setting boot option for client: %s", err.Error())
// bc.l.Errorf("Failed to read remote IP: %s", err.Error()) }
// helpers.HandleResponse(w, r, http.StatusInternalServerError, nil, bc.l) helpers.HandleResponse(w, r, returncode, content, bc.l)
// return }
// }
// var returncode int
// var content []byte
// switch r.Method {
// case http.MethodGet:
// returncode, content, err = bc.getBootOption(clientIP, w, r)
// case http.MethodPut:
// returncode, err = bc.setBootOption(clientIP, w, r)
// default:
// returncode = http.StatusMethodNotAllowed
// }
// if err != nil {
// bc.l.Errorf("An error occured while handling boot request: %q", err.Error())
// }
// helpers.HandleResponse(w, r, returncode, content, bc.l)
// }

View file

@ -7,6 +7,7 @@ import (
"io" "io"
"net/http" "net/http"
"path/filepath" "path/filepath"
"sort"
"git.faercol.me/faercol/http-boot-server/bootserver/helpers" "git.faercol.me/faercol/http-boot-server/bootserver/helpers"
"git.faercol.me/faercol/http-boot-server/bootserver/services" "git.faercol.me/faercol/http-boot-server/bootserver/services"
@ -19,6 +20,7 @@ type templateBootOption struct {
ID string ID string
Name string Name string
Path string Path string
DiskID string
Selected bool Selected bool
} }
@ -65,8 +67,12 @@ func (uc *UIController) serveUI(w http.ResponseWriter, r *http.Request) (int, in
Name: bo.Name, Name: bo.Name,
Path: bo.Path, Path: bo.Path,
ID: id, ID: id,
DiskID: bo.DiskID,
Selected: id == clt.SelectedOption, Selected: id == clt.SelectedOption,
}) })
sort.Slice(tplBO, func(i, j int) bool {
return tplBO[i].ID < tplBO[j].ID
})
} }
dat.Clients = append(dat.Clients, templateClient{ dat.Clients = append(dat.Clients, templateClient{
ID: clt.ID.String(), ID: clt.ID.String(),
@ -87,10 +93,16 @@ func (uc *UIController) serveUI(w http.ResponseWriter, r *http.Request) (int, in
} }
func (uc *UIController) ServeHTTP(w http.ResponseWriter, r *http.Request) { func (uc *UIController) ServeHTTP(w http.ResponseWriter, r *http.Request) {
if r.RequestURI != "/" {
uc.l.Errorf("Unhandled route %q", r.RequestURI)
helpers.HandleResponse(w, r, http.StatusNotFound, nil, uc.l)
return
}
returncode, contentLength, err := uc.serveUI(w, r) returncode, contentLength, err := uc.serveUI(w, r)
if err != nil { if err != nil {
uc.l.Errorf("Error serving UI: %s", err.Error()) uc.l.Errorf("Error serving UI: %s", err.Error())
helpers.HandleResponse(w, r, returncode, nil, uc.l) helpers.HandleResponse(w, r, returncode, nil, uc.l)
} else {
helpers.AddToContext(r, returncode, contentLength)
} }
helpers.AddToContext(r, returncode, contentLength)
} }

View file

@ -67,10 +67,11 @@ func New(appConf *config.AppConfig, logger *logrus.Logger) (*Server, error) {
} }
service := services.NewClientHandlerService(appConf.DataFilepath, logger) service := services.NewClientHandlerService(appConf.DataFilepath, logger)
controllers := map[string]http.Handler{ controllers := map[string]http.Handler{
client.EnrollRoute: middlewares.WithLogger(client.NewEnrollController(logger, service), logger), client.EnrollRoute: middlewares.WithLogger(client.NewEnrollController(logger, service), logger),
client.ConfigRoute: middlewares.WithLogger(client.NewGetConfigController(logger, service, appConf), logger), client.ConfigRoute: middlewares.WithLogger(client.NewGetConfigController(logger, service, appConf), logger),
ui.StaticRoute: &ui.StaticController{}, client.SetBootRoute: middlewares.WithLogger(client.NewBootController(logger, service), logger),
ui.UIRoute: middlewares.WithLogger(ui.NewUIController(logger, service), logger), ui.StaticRoute: &ui.StaticController{},
ui.UIRoute: middlewares.WithLogger(ui.NewUIController(logger, service), logger),
} }
m := http.NewServeMux() m := http.NewServeMux()

View file

@ -5,6 +5,7 @@ import (
"errors" "errors"
"fmt" "fmt"
"os" "os"
"sort"
"time" "time"
"git.faercol.me/faercol/http-boot-server/bootserver/bootoption" "git.faercol.me/faercol/http-boot-server/bootserver/bootoption"
@ -111,6 +112,9 @@ func (chs *ClientHandlerService) GetAllClientConfig() ([]*bootoption.Client, err
clt.ID = id clt.ID = id
clientList = append(clientList, clt) clientList = append(clientList, clt)
} }
sort.Slice(clientList, func(i, j int) bool {
return clientList[i].ID.String() < clientList[j].ID.String()
})
return clientList, nil return clientList, nil
} }

View file

@ -0,0 +1,20 @@
function selectBootOption(clientID, bootID) {
const Http = new XMLHttpRequest;
var host = window.location.protocol + "//" + window.location.host;
const url = host + "/config/boot"
console.debug(`Sending request to ${url}`);
Http.open("PUT", url);
Http.setRequestHeader("Content-Type", "application/json");
const body = JSON.stringify({
client_id: clientID,
option_id: bootID,
});
Http.onload = () => {
if (Http.readyState === 4 && Http.status === 202) {
location.reload();
} else {
console.error(`Unexpected returncode ${Http.status}`)
}
};
Http.send(body);
}

View file

@ -1,24 +1,48 @@
:root {
--page-background: #000000;
--background: #121212;
--surface: #181818;
--surface-active: #303030;
--surface-hover: #aaaaaa;
--primary: #3b80c9;
--primary-variant: #c9953b;
--secondary: #c94e3b;
--text-main: #ffffff;
--text-variant: #9B9B9B;
}
body { body {
background-color: black; background-color: var(--page-background);
color: white; color: var(--text-main);
}
.page-content {
width: 90%;
max-width: 1200px;
margin: auto;
}
.container {
border-radius: 5px;
background-color: var(--surface);
padding: 5px;
border-radius: 5px 5px;
margin: 5px auto;
} }
.main-container { .main-container {
border-radius: 5px;
background-color: #121212;
padding: 20px; padding: 20px;
} }
.title-container { .title-container {
border-radius: 5px 5px 0px 0px; background-color: var(--primary);
padding: 5px;
background-color: #1A3A5D;
} }
.client-list-container { .client-list-container {
border-radius: 0px 0px 5px 5px; border-radius: 0px 0px 5px 5px;
padding: 10px; padding: 10px;
background-color: #232323; background-color: var(--surface);
} }
.client-container { .client-container {
@ -36,7 +60,7 @@ body {
.client-uuid { .client-uuid {
font-size: smaller; font-size: smaller;
color: #9B9B9B; color: var(--text-variant);
} }
} }
@ -48,9 +72,16 @@ body {
.client-boot-option { .client-boot-option {
margin: 3px auto; margin: 3px auto;
padding: 8px 8px; padding: 8px 8px;
cursor: pointer;
&.selected { &.selected {
background-color: #3E4042; background-color: var(--surface-active);
cursor: default;
}
&:hover {
background-color: var(--surface-hover);
--text-variant: #000000;
} }
} }
@ -61,7 +92,7 @@ body {
.boot-option-uuid { .boot-option-uuid {
font-size: smaller; font-size: smaller;
color: #9B9B9B; color: var(--text-variant);
} }
.boot-option-path { .boot-option-path {

View file

@ -5,43 +5,47 @@
<meta charset="utf-8"> <meta charset="utf-8">
<title>HTTP boot server</title> <title>HTTP boot server</title>
<link rel="stylesheet" href="/static/stylesheets/main.css"> <link rel="stylesheet" href="/static/stylesheets/main.css">
<script src="/static/scripts/index.js"></script>
</head> </head>
<body> <body>
<div class="main-container"> <div class="page-content">
<div class="title-container"> <div class="title-container container">
<h1>HTTP boot server admin page</h1> <h1>HTTP boot server admin page</h1>
</div> </div>
<div class="client-list-container"> <div class="main-container container">
<h2>Currently enrolled clients</h2> <div class="client-list">
<h2>{{len .Clients}} enrolled clients</h2>
{{range .Clients}} {{range .Clients}}
<div class="client-container"> <div class="client-container">
<div class="client-header"> <div class="client-header">
<span class="client-name">{{.Name}}</span> <span class="client-name">{{.Name}}</span>
<span class="client-uuid">{{.ID}}</span> <span class="client-uuid">{{.ID}}</span>
</div> </div>
<div class="client-content"> <div class="client-content">
<div class="client-boot-options"> <div class="client-boot-options">
{{range .BootOptions}} {{$cid := .ID}}{{range .BootOptions}}
<div class="client-boot-option{{if .Selected}} selected{{end}}"> <div class="client-boot-option{{if .Selected}} selected{{end}}"
<div> onclick="selectBootOption('{{$cid}}', '{{.ID}}')">
<span class="boot-option-name">{{.Name}}</span> <div>
<span class="boot-option-uuid">{{.ID}}</span> <span class="boot-option-name">{{.Name}}</span>
</div> <span class="boot-option-uuid">{{.ID}}</span>
<div> </div>
<span class="boot-option-path">{{.Path}}</span> <div>
<span class="boot-option-path">{{.Path}}</span>
</div>
</div> </div>
{{end}}
</div> </div>
{{end}}
</div> </div>
</div> </div>
{{end}}
</div> </div>
{{end}}
</div>
</div>
</div> </div>
</body> </body>