179 lines
3.5 KiB
Go
179 lines
3.5 KiB
Go
package refresher
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"gitea.ceperka.net/rosti/lobby2/nodes"
|
|
)
|
|
|
|
const refreshIntervalSeconds = 15
|
|
|
|
// Refresher loads local node info and sends them to the master node.
|
|
type Refresher struct {
|
|
configPath string
|
|
nodePath string
|
|
}
|
|
|
|
func NewRefresher(nodePath string, configPath string) *Refresher {
|
|
return &Refresher{
|
|
nodePath: nodePath,
|
|
configPath: configPath,
|
|
}
|
|
}
|
|
|
|
// Run loop the process that updates node in the master node.
|
|
func (r *Refresher) Loop() {
|
|
log.Println("Start refresher loop")
|
|
|
|
for {
|
|
log.Println("Refreshing node")
|
|
err := r.Refresh()
|
|
if err != nil {
|
|
log.Printf("failed to refresh node: %v\n", err)
|
|
}
|
|
time.Sleep(time.Second * refreshIntervalSeconds)
|
|
}
|
|
}
|
|
|
|
// Refresh loads labels from the local filesystem and sends them to the master node.
|
|
func (r *Refresher) Refresh() error {
|
|
// Load config
|
|
cfg, err := r.getConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Load labels
|
|
node, err := r.loadNode()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
nodeBytes, err := json.Marshal(node)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal labels: %w", err)
|
|
}
|
|
|
|
// Send labels to master
|
|
req, err := http.NewRequest("POST", fmt.Sprintf("%s://%s:%d/nodes/%s", cfg.MasterProto, cfg.MasterHost, cfg.MasterPort, node.HostName), bytes.NewBuffer(nodeBytes))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
req.Header.Set("Authorization", "Bearer "+cfg.MasterToken)
|
|
|
|
client := &http.Client{}
|
|
resp, err := client.Do(req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != http.StatusNoContent {
|
|
return fmt.Errorf("unexpected status: %s", resp.Status)
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// Returns the node config.
|
|
func (r *Refresher) getConfig() (NodeConfig, error) {
|
|
cfg := NodeConfig{}
|
|
|
|
content, err := os.ReadFile(r.configPath)
|
|
if err != nil {
|
|
return cfg, fmt.Errorf("failed to read config file: %w", err)
|
|
}
|
|
|
|
err = json.Unmarshal(content, &cfg)
|
|
if err != nil {
|
|
return cfg, fmt.Errorf("failed to unmarshal config: %w", err)
|
|
}
|
|
|
|
if cfg.MasterProto == "" {
|
|
cfg.MasterProto = "http"
|
|
}
|
|
|
|
if cfg.MasterPort == 0 {
|
|
cfg.MasterPort = 1352
|
|
}
|
|
|
|
return cfg, nil
|
|
}
|
|
|
|
// TODO: rewrite this to load Node structure
|
|
func (r *Refresher) loadNode() (*nodes.Node, error) {
|
|
filePath := r.nodePath
|
|
_, err := os.Stat(filePath)
|
|
if os.IsNotExist(err) {
|
|
err = r.initNodeFile()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
content, err := os.ReadFile(r.nodePath)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read file: %w", err)
|
|
}
|
|
|
|
node := &nodes.Node{}
|
|
err = json.Unmarshal(content, &node)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to unmarshal node: %w", err)
|
|
}
|
|
|
|
return node, err
|
|
}
|
|
|
|
func (r *Refresher) initNodeFile() error {
|
|
err := r.createNodePath()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create node path: %w", err)
|
|
}
|
|
|
|
hostname, err := os.Hostname()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get hostname: %w", err)
|
|
}
|
|
|
|
node := &nodes.Node{
|
|
HostName: hostname,
|
|
}
|
|
|
|
nodeBytes, err := json.MarshalIndent(node, "", " ")
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal node: %w", err)
|
|
}
|
|
|
|
err = os.WriteFile(r.nodePath, nodeBytes, 0640)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to write node file: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (r *Refresher) createNodePath() error {
|
|
d := filepath.Dir(r.nodePath)
|
|
|
|
_, err := os.Stat(d)
|
|
if os.IsNotExist(err) {
|
|
err = os.MkdirAll(d, 0755)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create node dir: %w", err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|