forked from phc/cluster-dashboard
Initial commit
commit
64702636e6
@ -0,0 +1,6 @@
|
|||||||
|
MODE=development
|
||||||
|
HOST=:4000
|
||||||
|
BASE_URL=http://localhost:4000
|
||||||
|
|
||||||
|
# For example "node1,node2,node3"
|
||||||
|
NODES=""
|
@ -0,0 +1,8 @@
|
|||||||
|
.env
|
||||||
|
*.local*
|
||||||
|
bin/
|
||||||
|
|
||||||
|
.out/
|
||||||
|
out/
|
||||||
|
dist/
|
||||||
|
node_modules/
|
@ -0,0 +1,69 @@
|
|||||||
|
# Go Vite Kit
|
||||||
|
|
||||||
|
Minimal boilerplate project for a Golang server using [Fiber](https://github.com/gofiber/fiber) and [ViteJS](https://github.com/vitejs/vite) for static pages
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
- ⚡️ [Go Fiber](https://github.com/gofiber/fiber)
|
||||||
|
- 📦 [ViteJS](http://vitejs.dev/)
|
||||||
|
- 🎨 [Sass](https://sass-lang.com/)
|
||||||
|
- 🗄️ [Sqlite3](https://github.com/mattn/go-sqlite3)
|
||||||
|
|
||||||
|
## Architecture
|
||||||
|
|
||||||
|
- `frontend/`
|
||||||
|
|
||||||
|
This is a Vite project for building all the static pages used by this app.
|
||||||
|
|
||||||
|
- `backend/`
|
||||||
|
|
||||||
|
This keeps all server related files
|
||||||
|
|
||||||
|
- `config/`
|
||||||
|
|
||||||
|
Loads env variables and keeps them as globals
|
||||||
|
|
||||||
|
- `database/`
|
||||||
|
|
||||||
|
Module with a `Database` interface and two implementation: `memDB` is an in-memory database for testing purposes. `sqliteDB` is a wrapper for working with an SQLite database.
|
||||||
|
|
||||||
|
- `routes/`
|
||||||
|
|
||||||
|
Various functions for configuring all the server routes.
|
||||||
|
|
||||||
|
A very important file is `backend/routes/router.go` that contains the `HtmlEntrypoints` variable that is used both by the backend and ViteJS to mount HTML entrypoints.
|
||||||
|
|
||||||
|
When building the frontend ViteJS will call `go run ./meta/routes` to read the content of the `HtmlEntrypoints` variable. This is also used while developing to let Vite know add all necessary entrypoints to the dev server.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
To setup the project first install the required npm packages
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Install all JS dependencies
|
||||||
|
$ npm install
|
||||||
|
```
|
||||||
|
|
||||||
|
then you can start versioning the **lock file** of your package manager.
|
||||||
|
|
||||||
|
### Development
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Development
|
||||||
|
$ MODE=dev go run -v ./cmd/server
|
||||||
|
|
||||||
|
# Development with watcher
|
||||||
|
$ fd -e go | MODE=dev entr -r go run -v ./cmd/server
|
||||||
|
```
|
||||||
|
|
||||||
|
### Production
|
||||||
|
|
||||||
|
You can build everything with the following command, it will build first the frontend and then the backend and generate `./out/server`.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Build
|
||||||
|
$ go run -v ./cmd/build
|
||||||
|
|
||||||
|
# Run
|
||||||
|
$ ./out/server
|
||||||
|
```
|
@ -0,0 +1,44 @@
|
|||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/joho/godotenv"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
Mode string
|
||||||
|
Host string
|
||||||
|
BaseURL string
|
||||||
|
|
||||||
|
// Nodes è una lista di nomi dei vari host
|
||||||
|
Nodes []string
|
||||||
|
)
|
||||||
|
|
||||||
|
func loadEnv(key string, defaultValue ...string) string {
|
||||||
|
env := os.Getenv(key)
|
||||||
|
|
||||||
|
if len(defaultValue) > 0 && env == "" {
|
||||||
|
env = defaultValue[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("Environment variable %s = %q", key, env)
|
||||||
|
return env
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
// Setup logger
|
||||||
|
log.SetFlags(log.Lshortfile | log.Ltime | log.Ldate)
|
||||||
|
|
||||||
|
// Load Config
|
||||||
|
godotenv.Load()
|
||||||
|
|
||||||
|
Mode = loadEnv("MODE", "development")
|
||||||
|
Host = loadEnv("HOST", ":4000")
|
||||||
|
BaseURL = loadEnv("BASE_URL", "http://localhost:4000")
|
||||||
|
|
||||||
|
nodes := loadEnv("NODES", "") // for example "node1,node2,node3"
|
||||||
|
Nodes = strings.Split(nodes, ",")
|
||||||
|
}
|
@ -0,0 +1,72 @@
|
|||||||
|
package database
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/executor"
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
// simpleDB è una implementazione di [database.Database] che tiene giusto una cache in memoria e
|
||||||
|
// quando il server viene riavviato perde tutte le statistiche che ha accumulato. Più avanti si
|
||||||
|
// potrebbe pensare di scrivere queste informazioni in un file o usare un vero database.
|
||||||
|
type simpleDB struct {
|
||||||
|
Executor executor.Service
|
||||||
|
|
||||||
|
// lastUpdate tiene traccia di quando abbiamo aggiornato l'ultima volta tutti i dati
|
||||||
|
lastUpdate *time.Time
|
||||||
|
|
||||||
|
// Nodes is a map from hostname to node info
|
||||||
|
nodes map[string]model.Node
|
||||||
|
|
||||||
|
// Jobs is a map from job id to job info
|
||||||
|
jobs map[int]model.Job
|
||||||
|
|
||||||
|
// The following are maps from hostname to a list of sampled temperatures, used memory, used storage space and network upload and download rate.
|
||||||
|
|
||||||
|
temperatureSamples map[string][]model.Sample[float64]
|
||||||
|
memorySamples map[string][]model.Sample[int64]
|
||||||
|
storageSamples map[string][]model.Sample[int64]
|
||||||
|
networkUploadSamples map[string][]model.Sample[int64]
|
||||||
|
networkDownloadSamples map[string][]model.Sample[int64]
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewSimpleDatabase(ex executor.Service) Database {
|
||||||
|
return &simpleDB{Executor: ex}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) GetNodeStatus(hostname string) (*model.Node, error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) GetJobStatus(id int) (*model.Job, error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) AllNodes() ([]*model.Node, error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) AllJobs() ([]*model.Job, error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) QueryTemperatureSamples(from, to time.Time) ([]model.Sample[float64], error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) QueryMemorySamples(from, to time.Time) ([]model.Sample[int64], error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) QueryStorageSamples(from, to time.Time) ([]model.Sample[int64], error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) QueryNetworkUploadSamples(from, to time.Time) ([]model.Sample[int64], error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *simpleDB) QueryNetworkDownloadSamples(from, to time.Time) ([]model.Sample[int64], error) {
|
||||||
|
panic("todo")
|
||||||
|
}
|
@ -0,0 +1,21 @@
|
|||||||
|
package database
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Database interface {
|
||||||
|
GetNodeStatus(hostname string) (*model.Node, error)
|
||||||
|
GetJobStatus(id int) (*model.Job, error)
|
||||||
|
|
||||||
|
AllNodes() ([]*model.Node, error)
|
||||||
|
AllJobs() ([]*model.Job, error)
|
||||||
|
|
||||||
|
QueryTemperatureSamples(from, to time.Time) ([]model.Sample[float64], error)
|
||||||
|
QueryMemorySamples(from, to time.Time) ([]model.Sample[int64], error)
|
||||||
|
QueryStorageSamples(from, to time.Time) ([]model.Sample[int64], error)
|
||||||
|
QueryNetworkUploadSamples(from, to time.Time) ([]model.Sample[int64], error)
|
||||||
|
QueryNetworkDownloadSamples(from, to time.Time) ([]model.Sample[int64], error)
|
||||||
|
}
|
@ -0,0 +1,16 @@
|
|||||||
|
package executor
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
// Service is a service that handles executing commands on the main host and does a first processing of the raw data it gets from the system
|
||||||
|
type Service interface {
|
||||||
|
SlurmQueue() []string
|
||||||
|
SlurmJobs() []string
|
||||||
|
|
||||||
|
NodeUptime(hostname string) time.Time
|
||||||
|
|
||||||
|
Temperature(hostname string) float64
|
||||||
|
MemoryUsage(hostname string) int64
|
||||||
|
StorageUsage(hostname string) int64
|
||||||
|
NetworkUploadDownload(hostname string) (int64, int64)
|
||||||
|
}
|
@ -0,0 +1,43 @@
|
|||||||
|
package executor
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
var _ Service = &Mock{}
|
||||||
|
|
||||||
|
type Mock struct {
|
||||||
|
SlurmQueueFunc func() []string
|
||||||
|
SlurmJobsFunc func() []string
|
||||||
|
NodeUptimeFunc func(hostname string) time.Time
|
||||||
|
TemperatureFunc func(hostname string) float64
|
||||||
|
MemoryUsageFunc func(hostname string) int64
|
||||||
|
StorageUsageFunc func(hostname string) int64
|
||||||
|
NetworkUploadDownloadFunc func(hostname string) (int64, int64)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) SlurmQueue() []string {
|
||||||
|
return ex.SlurmQueueFunc()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) SlurmJobs() []string {
|
||||||
|
return ex.SlurmJobsFunc()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) NodeUptime(hostname string) time.Time {
|
||||||
|
return ex.NodeUptimeFunc(hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) Temperature(hostname string) float64 {
|
||||||
|
return ex.TemperatureFunc(hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) MemoryUsage(hostname string) int64 {
|
||||||
|
return ex.MemoryUsageFunc(hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) StorageUsage(hostname string) int64 {
|
||||||
|
return ex.StorageUsageFunc(hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ex *Mock) NetworkUploadDownload(hostname string) (int64, int64) {
|
||||||
|
return ex.NetworkUploadDownloadFunc(hostname)
|
||||||
|
}
|
@ -0,0 +1,31 @@
|
|||||||
|
package model
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/exp/constraints"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Numeric = constraints.Ordered
|
||||||
|
|
||||||
|
// Job contiene le informazioni su un lavoro di slurm (il nostro gestore di code di lavori)
|
||||||
|
type Job struct {
|
||||||
|
Id int `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
|
||||||
|
Nodes []string `json:"nodes"`
|
||||||
|
Resources []string `json:"resources"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sample di un valore di tipo N (preferibilmente numerico) in un certo momento temporale.
|
||||||
|
type Sample[N Numeric] struct {
|
||||||
|
Timestamp time.Time `json:"timestamp"`
|
||||||
|
Value N `json:"value"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Node contiene le informazioni relative ad un nodo nel nostro sistema
|
||||||
|
type Node struct {
|
||||||
|
Hostname string `json:"hostname"`
|
||||||
|
StartTime time.Time `json:"startTime"`
|
||||||
|
}
|
@ -0,0 +1,9 @@
|
|||||||
|
package routes
|
||||||
|
|
||||||
|
import "github.com/gofiber/fiber/v2"
|
||||||
|
|
||||||
|
func (r *Router) Api(api fiber.Router) {
|
||||||
|
api.Get("/status", func(c *fiber.Ctx) error {
|
||||||
|
return c.JSON("ok")
|
||||||
|
})
|
||||||
|
}
|
@ -0,0 +1,19 @@
|
|||||||
|
package routes
|
||||||
|
|
||||||
|
import "git.phc.dm.unipi.it/phc/cluster-dashboard/backend/database"
|
||||||
|
|
||||||
|
// Router is the main service that defines all routes, this only depends on the Database service
|
||||||
|
type Router struct {
|
||||||
|
Database database.Database
|
||||||
|
}
|
||||||
|
|
||||||
|
type htmlEntrypoint struct {
|
||||||
|
Route string `json:"route"`
|
||||||
|
Filename string `json:"filename"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var HtmlEntrypoints = []htmlEntrypoint{
|
||||||
|
{"/", "./index.html"},
|
||||||
|
{"/dashboard", "./index.html"},
|
||||||
|
{"/jobs", "./index.html"},
|
||||||
|
}
|
@ -0,0 +1,77 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/fatih/color"
|
||||||
|
|
||||||
|
"github.com/gofiber/fiber/v2"
|
||||||
|
"github.com/gofiber/fiber/v2/middleware/logger"
|
||||||
|
"github.com/gofiber/fiber/v2/middleware/recover"
|
||||||
|
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/config"
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/database"
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/executor"
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/routes"
|
||||||
|
)
|
||||||
|
|
||||||
|
const FrontendOutDir = "./out/frontend"
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
ex := &executor.Mock{} // to try out locally this project in a reasonable way
|
||||||
|
db := database.NewSimpleDatabase(ex)
|
||||||
|
|
||||||
|
router := &routes.Router{
|
||||||
|
Database: db,
|
||||||
|
}
|
||||||
|
|
||||||
|
app := fiber.New()
|
||||||
|
|
||||||
|
app.Use(logger.New())
|
||||||
|
app.Use(recover.New())
|
||||||
|
|
||||||
|
app.Static("/", FrontendOutDir)
|
||||||
|
for _, entrypoint := range routes.HtmlEntrypoints {
|
||||||
|
app.Get(entrypoint.Route, func(c *fiber.Ctx) error {
|
||||||
|
return c.SendFile(path.Join(FrontendOutDir, entrypoint.Filename))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
app.Route("/api", router.Api)
|
||||||
|
|
||||||
|
if strings.HasPrefix(config.Mode, "dev") {
|
||||||
|
setupDevServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Fatal(app.Listen(config.Host))
|
||||||
|
}
|
||||||
|
|
||||||
|
func setupDevServer() {
|
||||||
|
log.Printf(`Running dev server for frontend: "npm run dev"`)
|
||||||
|
cmd := exec.Command("sh", "-c", "npm run dev")
|
||||||
|
cmdStdout, _ := cmd.StdoutPipe()
|
||||||
|
cmdStderr, _ := cmd.StderrPipe()
|
||||||
|
|
||||||
|
viteLogger := log.New(os.Stderr, color.HiGreenString("[ViteJS]")+" ", log.Ltime|log.Lmsgprefix)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
s := bufio.NewScanner(io.MultiReader(cmdStdout, cmdStderr))
|
||||||
|
for s.Scan() {
|
||||||
|
viteLogger.Print(s.Text())
|
||||||
|
}
|
||||||
|
if err := s.Err(); err != nil {
|
||||||
|
viteLogger.Fatal(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
err := cmd.Start()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,14 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>Homepage</title>
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="/styles/main.scss">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<script type="module" src="/src/main.jsx"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -0,0 +1,29 @@
|
|||||||
|
import Router from 'preact-router'
|
||||||
|
import { route } from 'preact-router'
|
||||||
|
|
||||||
|
import { render } from 'preact'
|
||||||
|
import { useEffect } from 'preact/hooks'
|
||||||
|
|
||||||
|
const Redirect = ({ to }) => {
|
||||||
|
useEffect(() => {
|
||||||
|
route(to, true)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
Redirecting to <pre>{to}</pre>...
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const App = () => {
|
||||||
|
return (
|
||||||
|
<Router>
|
||||||
|
<DashboardPage path="/dashboard" />
|
||||||
|
<JobsPage path="/jobs" />
|
||||||
|
<Redirect default to="/dashboard" />
|
||||||
|
</Router>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
render(<App />, document.body)
|
@ -0,0 +1,43 @@
|
|||||||
|
*,
|
||||||
|
*::before,
|
||||||
|
*::after {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
|
||||||
|
width: 100%;
|
||||||
|
min-height: 100vh;
|
||||||
|
|
||||||
|
font-family: 'Inter', 'Segoe UI', 'Helvetica', 'Arial', sans-serif;
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Headings
|
||||||
|
|
||||||
|
$base-font-size: 18px;
|
||||||
|
$heading-scale: 1.33;
|
||||||
|
|
||||||
|
@function pow($number, $exponent) {
|
||||||
|
$value: 1;
|
||||||
|
|
||||||
|
@if $exponent > 0 {
|
||||||
|
@for $i from 1 through $exponent {
|
||||||
|
$value: $value * $number;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@return $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@for $i from 1 through 5 {
|
||||||
|
h#{$i} {
|
||||||
|
margin: 0;
|
||||||
|
|
||||||
|
$factor: pow($heading-scale, 5 - $i);
|
||||||
|
font-size: $base-font-size * $factor;
|
||||||
|
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,21 @@
|
|||||||
|
module git.phc.dm.unipi.it/phc/cluster-dashboard
|
||||||
|
|
||||||
|
go 1.18
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/gofiber/fiber/v2 v2.34.1
|
||||||
|
github.com/joho/godotenv v1.4.0
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/andybalholm/brotli v1.0.4 // indirect
|
||||||
|
github.com/fatih/color v1.15.0 // indirect
|
||||||
|
github.com/klauspost/compress v1.15.0 // indirect
|
||||||
|
github.com/mattn/go-colorable v0.1.13 // indirect
|
||||||
|
github.com/mattn/go-isatty v0.0.17 // indirect
|
||||||
|
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
||||||
|
github.com/valyala/fasthttp v1.37.0 // indirect
|
||||||
|
github.com/valyala/tcplisten v1.0.0 // indirect
|
||||||
|
golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53 // indirect
|
||||||
|
golang.org/x/sys v0.6.0 // indirect
|
||||||
|
)
|
@ -0,0 +1,40 @@
|
|||||||
|
github.com/andybalholm/brotli v1.0.4 h1:V7DdXeJtZscaqfNuAdSRuRFzuiKlHSC/Zh3zl9qY3JY=
|
||||||
|
github.com/andybalholm/brotli v1.0.4/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
|
||||||
|
github.com/fatih/color v1.15.0 h1:kOqh6YHBtK8aywxGerMG2Eq3H6Qgoqeo13Bk2Mv/nBs=
|
||||||
|
github.com/fatih/color v1.15.0/go.mod h1:0h5ZqXfHYED7Bhv2ZJamyIOUej9KtShiJESRwBDUSsw=
|
||||||
|
github.com/gofiber/fiber/v2 v2.34.1 h1:C6saXB7385HvtXX+XMzc5Dqj5S/aEXOfKCW7JNep4rA=
|
||||||
|
github.com/gofiber/fiber/v2 v2.34.1/go.mod h1:ozRQfS+D7EL1+hMH+gutku0kfx1wLX4hAxDCtDzpj4U=
|
||||||
|
github.com/joho/godotenv v1.4.0 h1:3l4+N6zfMWnkbPEXKng2o2/MR5mSwTrBih4ZEkkz1lg=
|
||||||
|
github.com/joho/godotenv v1.4.0/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
|
||||||
|
github.com/klauspost/compress v1.15.0 h1:xqfchp4whNFxn5A4XFyyYtitiWI8Hy5EW59jEwcyL6U=
|
||||||
|
github.com/klauspost/compress v1.15.0/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
|
||||||
|
github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA=
|
||||||
|
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
|
||||||
|
github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
|
||||||
|
github.com/mattn/go-isatty v0.0.17 h1:BTarxUcIeDqL27Mc+vyvdWYSL28zpIhv3RoTdsLMPng=
|
||||||
|
github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
|
||||||
|
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
|
||||||
|
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
|
||||||
|
github.com/valyala/fasthttp v1.37.0 h1:7WHCyI7EAkQMVmrfBhWTCOaeROb1aCBiTopx63LkMbE=
|
||||||
|
github.com/valyala/fasthttp v1.37.0/go.mod h1:t/G+3rLek+CyY9bnIE+YlMRddxVAAGjhxndDB4i4C0I=
|
||||||
|
github.com/valyala/tcplisten v1.0.0 h1:rBHj/Xf+E1tRGZyWIWwJDiRY0zc1Js+CV5DqwacVSA8=
|
||||||
|
github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc=
|
||||||
|
golang.org/x/crypto v0.0.0-20220214200702-86341886e292/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||||
|
golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53 h1:5llv2sWeaMSnA3w2kS57ouQQ4pudlXrR0dCgw51QK9o=
|
||||||
|
golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w=
|
||||||
|
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
|
||||||
|
golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
|
||||||
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9 h1:nhht2DYV/Sn3qOayu8lM+cU1ii9sTLUeBQwQQfUHtrs=
|
||||||
|
golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.6.0 h1:MVltZSvRTcU2ljQOhs94SXPftV6DCNnZViHeQps87pQ=
|
||||||
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
|
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
||||||
|
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
|
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
|
||||||
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
@ -0,0 +1,39 @@
|
|||||||
|
import { join } from 'path'
|
||||||
|
import express from 'express'
|
||||||
|
import { createServer as createViteServer } from 'vite'
|
||||||
|
import { readFile } from 'fs/promises'
|
||||||
|
|
||||||
|
import { getBuildRoutes } from './routes.js'
|
||||||
|
|
||||||
|
async function createServer() {
|
||||||
|
const app = express()
|
||||||
|
|
||||||
|
// In middleware mode, if you want to use Vite's own HTML serving logic
|
||||||
|
// use `'html'` as the `middlewareMode` (ref https://vitejs.dev/config/#server-middlewaremode)
|
||||||
|
const vite = await createViteServer({
|
||||||
|
server: { middlewareMode: 'html' },
|
||||||
|
})
|
||||||
|
|
||||||
|
const routes = await getBuildRoutes()
|
||||||
|
|
||||||
|
console.log(`mounting static routes...`)
|
||||||
|
for (const [route, file] of Object.entries(routes)) {
|
||||||
|
const filePath = join('./frontend', file)
|
||||||
|
console.log(`- "%s" => %s`, route, filePath)
|
||||||
|
|
||||||
|
app.get(route, async (req, res) => {
|
||||||
|
const htmlFile = await readFile(filePath, 'utf8')
|
||||||
|
const htmlViteHooksFile = await vite.transformIndexHtml(req.originalUrl, htmlFile)
|
||||||
|
|
||||||
|
res.setHeader('Content-Type', 'text/html')
|
||||||
|
return res.send(htmlViteHooksFile)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
app.use(vite.middlewares)
|
||||||
|
|
||||||
|
console.log('Started dev server on port :3000')
|
||||||
|
app.listen(3000)
|
||||||
|
}
|
||||||
|
|
||||||
|
createServer()
|
@ -0,0 +1,39 @@
|
|||||||
|
import { spawn } from 'child_process'
|
||||||
|
|
||||||
|
function transformRoutes(entrypoints) {
|
||||||
|
return Object.fromEntries(entrypoints.map(({ route, filename }) => [route, filename]))
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getBuildRoutes() {
|
||||||
|
// Thanks to ChatGPT
|
||||||
|
function readCommandOutputAsJSON(command) {
|
||||||
|
const [cmd, ...args] = command.split(' ')
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const child = spawn(cmd, args)
|
||||||
|
|
||||||
|
let stdout = ''
|
||||||
|
|
||||||
|
child.stdout.on('data', data => {
|
||||||
|
stdout += data.toString()
|
||||||
|
})
|
||||||
|
|
||||||
|
child.on('close', code => {
|
||||||
|
if (code !== 0) {
|
||||||
|
reject(`Command ${cmd} ${args.join(' ')} failed with code ${code}`)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const output = JSON.parse(stdout)
|
||||||
|
resolve(output)
|
||||||
|
} catch (e) {
|
||||||
|
reject(`Error parsing JSON output: ${e.message}`)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('loading build entrypoints...')
|
||||||
|
return transformRoutes(await readCommandOutputAsJSON('go run ./meta/routes'))
|
||||||
|
}
|
@ -0,0 +1,17 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"git.phc.dm.unipi.it/phc/cluster-dashboard/backend/routes"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
enc := json.NewEncoder(os.Stdout)
|
||||||
|
enc.SetIndent("", " ")
|
||||||
|
if err := enc.Encode(routes.HtmlEntrypoints); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,23 @@
|
|||||||
|
{
|
||||||
|
"name": "frontend",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Javascript frontend for this Golang server built using ViteJS",
|
||||||
|
"main": "index.js",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "node meta/dev-server.js",
|
||||||
|
"build": "vite build"
|
||||||
|
},
|
||||||
|
"license": "MIT",
|
||||||
|
"devDependencies": {
|
||||||
|
"@preact/preset-vite": "^2.5.0",
|
||||||
|
"express": "^4.18.1",
|
||||||
|
"sass": "^1.53.0",
|
||||||
|
"vite": "^2.9.13"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"axios": "^1.4.0",
|
||||||
|
"preact": "^10.13.2",
|
||||||
|
"preact-router": "^4.1.0"
|
||||||
|
}
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,34 @@
|
|||||||
|
import { defineConfig } from 'vite'
|
||||||
|
import { getBuildRoutes } from './meta/routes.js'
|
||||||
|
|
||||||
|
import preactPlugin from '@preact/preset-vite'
|
||||||
|
|
||||||
|
import { join } from 'path'
|
||||||
|
|
||||||
|
export default defineConfig(async () => {
|
||||||
|
const routes = await getBuildRoutes()
|
||||||
|
console.log('html entrypoints:')
|
||||||
|
for (const [route, filename] of Object.entries(routes)) {
|
||||||
|
console.log(`- "${route}" => ${filename}`)
|
||||||
|
}
|
||||||
|
console.log()
|
||||||
|
|
||||||
|
const entryPoints = Object.values(routes)
|
||||||
|
|
||||||
|
return {
|
||||||
|
root: './frontend',
|
||||||
|
build: {
|
||||||
|
outDir: '../out/frontend',
|
||||||
|
emptyOutDir: true,
|
||||||
|
rollupOptions: {
|
||||||
|
input: entryPoints.map(e => join('./frontend', e)),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
server: {
|
||||||
|
proxy: {
|
||||||
|
'/api': 'http://127.0.0.1:4000/',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
plugins: [preactPlugin()],
|
||||||
|
}
|
||||||
|
})
|
Loading…
Reference in New Issue