diff --git a/app/app.go b/app/app.go index 85bb623..0da2b82 100644 --- a/app/app.go +++ b/app/app.go @@ -67,50 +67,50 @@ func Run() { }) router.GET("/nodes/:node", func(c *gin.Context) { - node := c.Param("node") + nodeid := c.Param("node") - host, err := cluster.GetHost(node) + node, err := cluster.GetNode(nodeid) if err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"error": err}) + c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()}) return } else { - c.JSON(http.StatusOK, gin.H{"node": host}) + c.JSON(http.StatusOK, gin.H{"node": node}) return } }) router.GET("/nodes/:node/devices", func(c *gin.Context) { - node := c.Param("node") + nodeid := c.Param("node") - host, err := cluster.GetHost(node) + node, err := cluster.GetNode(nodeid) if err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"error": err}) + c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()}) return } else { - c.JSON(http.StatusOK, gin.H{"devices": host.Devices}) + c.JSON(http.StatusOK, gin.H{"devices": node.Devices}) return } }) - router.GET("/nodes/:node/instances/:instance", func(c *gin.Context) { - node := c.Param("node") - vmid, err := strconv.ParseUint(c.Param("instance"), 10, 64) + router.GET("/nodes/:node/instances/:vmid", func(c *gin.Context) { + nodeid := c.Param("node") + vmid, err := strconv.ParseUint(c.Param("vmid"), 10, 64) if err != nil { c.JSON(http.StatusInternalServerError, gin.H{"error": fmt.Sprintf("%s could not be converted to vmid (uint)", c.Param("instance"))}) return } - host, err := cluster.GetHost(node) + node, err := cluster.GetNode(nodeid) if err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"error": fmt.Sprintf("vmid %s not found in cluster", node)}) + c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()}) return } else { - instance, err := host.GetInstance(uint(vmid)) + instance, err := node.GetInstance(uint(vmid)) if err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"error": fmt.Sprintf("%d not found in %s", vmid, node)}) + c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()}) return } else { c.JSON(http.StatusOK, gin.H{"instance": instance}) @@ -119,5 +119,65 @@ func Run() { } }) + router.POST("/sync", func(c *gin.Context) { + go func() { + start := time.Now() + log.Printf("Starting cluster sync\n") + cluster.Sync() + log.Printf("Synced cluster in %fs\n", time.Since(start).Seconds()) + }() + }) + + router.POST("/nodes/:node/sync", func(c *gin.Context) { + nodeid := c.Param("node") + go func() { + start := time.Now() + log.Printf("Starting %s sync\n", nodeid) + err := cluster.RebuildHost(nodeid) + if err != nil { + log.Printf("Failed to sync %s: %s", nodeid, err.Error()) + return + } else { + log.Printf("Synced %s in %fs\n", nodeid, time.Since(start).Seconds()) + return + } + }() + }) + + router.POST("/nodes/:node/instances/:vmid/sync", func(c *gin.Context) { + nodeid := c.Param("node") + vmid, err := strconv.ParseUint(c.Param("vmid"), 10, 64) + if err != nil { + c.JSON(http.StatusInternalServerError, gin.H{"error": fmt.Sprintf("%s could not be converted to vmid (uint)", c.Param("instance"))}) + return + } + + go func() { + start := time.Now() + log.Printf("Starting %s.%d sync\n", nodeid, vmid) + + node, err := cluster.GetNode(nodeid) + if err != nil { + log.Printf("Failed to sync %s.%d: %s", nodeid, vmid, err.Error()) + return + } + + instance, err := node.GetInstance(uint(vmid)) + if err != nil { + log.Printf("Failed to sync %s.%d: %s", nodeid, vmid, err.Error()) + return + } + + err = node.RebuildInstance(instance.Type, uint(vmid)) + if err != nil { + log.Printf("Failed to sync %s.%d: %s", nodeid, vmid, err.Error()) + return + } else { + log.Printf("Synced %s.%d in %fs\n", nodeid, vmid, time.Since(start).Seconds()) + return + } + }() + }) + router.Run("0.0.0.0:" + strconv.Itoa(config.ListenPort)) } diff --git a/app/model.go b/app/model.go index 2435a65..8c4e06b 100644 --- a/app/model.go +++ b/app/model.go @@ -15,7 +15,7 @@ func (cluster *Cluster) Sync() error { cluster.lock.Lock() defer cluster.lock.Unlock() - cluster.Hosts = make(map[string]*Host) + cluster.Nodes = make(map[string]*Node) // get all nodes nodes, err := cluster.pve.Nodes() @@ -35,8 +35,8 @@ func (cluster *Cluster) Sync() error { } // get a node in the cluster -func (cluster *Cluster) GetHost(hostName string) (*Host, error) { - host_ch := make(chan *Host) +func (cluster *Cluster) GetNode(hostName string) (*Node, error) { + host_ch := make(chan *Node) err_ch := make(chan error) go func() { @@ -44,21 +44,23 @@ func (cluster *Cluster) GetHost(hostName string) (*Host, error) { cluster.lock.Lock() defer cluster.lock.Unlock() // get host - host, ok := cluster.Hosts[hostName] + host, ok := cluster.Nodes[hostName] if !ok { host_ch <- nil err_ch <- fmt.Errorf("%s not in cluster", hostName) - } - // aquire host lock to wait in case of a concurrent write - host.lock.Lock() - defer host.lock.Unlock() + } else { + // aquire host lock to wait in case of a concurrent write + host.lock.Lock() + defer host.lock.Unlock() - host_ch <- host - err_ch <- nil + host_ch <- host + err_ch <- nil + } }() host := <-host_ch err := <-err_ch + return host, err } @@ -72,7 +74,7 @@ func (cluster *Cluster) RebuildHost(hostName string) error { host.lock.Lock() defer host.lock.Unlock() - cluster.Hosts[hostName] = host + cluster.Nodes[hostName] = host // get node's VMs vms, err := host.VirtualMachines() @@ -81,7 +83,7 @@ func (cluster *Cluster) RebuildHost(hostName string) error { } for _, vmid := range vms { - err := host.RebuildVM(vmid) + err := host.RebuildInstance(VM, vmid) if err != nil { return err } @@ -93,7 +95,7 @@ func (cluster *Cluster) RebuildHost(hostName string) error { return err } for _, vmid := range cts { - err := host.RebuildCT(vmid) + err := host.RebuildInstance(CT, vmid) if err != nil { return err } @@ -102,7 +104,7 @@ func (cluster *Cluster) RebuildHost(hostName string) error { return nil } -func (host *Host) GetInstance(vmid uint) (*Instance, error) { +func (host *Node) GetInstance(vmid uint) (*Instance, error) { instance_ch := make(chan *Instance) err_ch := make(chan error) @@ -115,13 +117,14 @@ func (host *Host) GetInstance(vmid uint) (*Instance, error) { if !ok { instance_ch <- nil err_ch <- fmt.Errorf("vmid %d not in host %s", vmid, host.Name) - } - // aquire instance lock to wait in case of a concurrent write - instance.lock.Lock() - defer instance.lock.Unlock() + } else { + // aquire instance lock to wait in case of a concurrent write + instance.lock.Lock() + defer instance.lock.Unlock() - instance_ch <- instance - err_ch <- nil + instance_ch <- instance + err_ch <- nil + } }() instance := <-instance_ch @@ -129,10 +132,21 @@ func (host *Host) GetInstance(vmid uint) (*Instance, error) { return instance, err } -func (host *Host) RebuildVM(vmid uint) error { - instance, err := host.VirtualMachine(vmid) - if err != nil { - return err +func (host *Node) RebuildInstance(instancetype InstanceType, vmid uint) error { + var instance *Instance + if instancetype == VM { + var err error + instance, err = host.VirtualMachine(vmid) + if err != nil { + return err + } + } else if instancetype == CT { + var err error + instance, err = host.Container(vmid) + if err != nil { + return err + } + } // aquire lock on instance, release on return @@ -156,33 +170,10 @@ func (host *Host) RebuildVM(vmid uint) error { return nil } -func (host *Host) RebuildCT(vmid uint) error { - instance, err := host.Container(vmid) - if err != nil { - return err - } - - // aquire lock on instance, release on return - instance.lock.Lock() - defer instance.lock.Unlock() - - host.Instances[vmid] = instance - - for volid := range instance.configDisks { - instance.RebuildVolume(host, volid) - } - - for netid := range instance.configNets { - instance.RebuildNet(netid) - } - - return nil -} - -func (instance *Instance) RebuildVolume(host *Host, volid string) error { +func (instance *Instance) RebuildVolume(host *Node, volid string) error { volumeDataString := instance.configDisks[volid] - volume, _, _, err := GetVolumeInfo(host, volumeDataString) + volume, err := GetVolumeInfo(host, volumeDataString) if err != nil { return err } @@ -209,7 +200,7 @@ func (instance *Instance) RebuildNet(netid string) error { return nil } -func (instance *Instance) RebuildDevice(host *Host, deviceid string) error { +func (instance *Instance) RebuildDevice(host *Node, deviceid string) error { instanceDevice, ok := instance.configHostPCIs[deviceid] if !ok { // if device does not exist return fmt.Errorf("%s not found in devices", deviceid) diff --git a/app/proxmox.go b/app/proxmox.go index 67b05f4..3793d7c 100644 --- a/app/proxmox.go +++ b/app/proxmox.go @@ -57,8 +57,8 @@ func (pve ProxmoxClient) Nodes() ([]string, error) { } // Gets a Node's resources but does not recursively expand instances -func (pve ProxmoxClient) Node(nodeName string) (*Host, error) { - host := Host{} +func (pve ProxmoxClient) Node(nodeName string) (*Node, error) { + host := Node{} host.Devices = make(map[string]*Device) host.Instances = make(map[uint]*Instance) @@ -87,7 +87,7 @@ func (pve ProxmoxClient) Node(nodeName string) (*Host, error) { } // Get all VM IDs on specified host -func (host *Host) VirtualMachines() ([]uint, error) { +func (host *Node) VirtualMachines() ([]uint, error) { vms, err := host.pvenode.VirtualMachines(context.Background()) if err != nil { return nil, err @@ -100,7 +100,7 @@ func (host *Host) VirtualMachines() ([]uint, error) { } // Get a VM's CPU, Memory but does not recursively link Devices, Disks, Drives, Nets -func (host *Host) VirtualMachine(VMID uint) (*Instance, error) { +func (host *Node) VirtualMachine(VMID uint) (*Instance, error) { instance := Instance{} vm, err := host.pvenode.VirtualMachine(context.Background(), int(VMID)) if err != nil { @@ -135,7 +135,7 @@ func MergeVMDisksAndUnused(vmc *proxmox.VirtualMachineConfig) map[string]string } // Get all CT IDs on specified host -func (host *Host) Containers() ([]uint, error) { +func (host *Node) Containers() ([]uint, error) { cts, err := host.pvenode.Containers(context.Background()) if err != nil { return nil, err @@ -148,7 +148,7 @@ func (host *Host) Containers() ([]uint, error) { } // Get a CT's CPU, Memory, Swap but does not recursively link Devices, Disks, Drives, Nets -func (host *Host) Container(VMID uint) (*Instance, error) { +func (host *Node) Container(VMID uint) (*Instance, error) { instance := Instance{} ct, err := host.pvenode.Container(context.Background(), int(VMID)) if err != nil { @@ -185,31 +185,32 @@ func MergeCTDisksAndUnused(cc *proxmox.ContainerConfig) map[string]string { return mergedDisks } -// get volume fornmat, size, volumeid, and storageid from instance volume data string (eg: local:100/vm-100-disk-0.raw ... ) -func GetVolumeInfo(host *Host, volume string) (*Volume, string, string, error) { +// get volume format, size, volumeid, and storageid from instance volume data string (eg: local:100/vm-100-disk-0.raw ... ) +func GetVolumeInfo(host *Node, volume string) (*Volume, error) { volumeData := Volume{} storageID := strings.Split(volume, ":")[0] volumeID := strings.Split(volume, ",")[0] storage, err := host.pvenode.Storage(context.Background(), storageID) if err != nil { - return &volumeData, volumeID, storageID, nil + return &volumeData, nil } content, err := storage.GetContent(context.Background()) if err != nil { - return &volumeData, volumeID, storageID, nil + return &volumeData, nil } for _, c := range content { if c.Volid == volumeID { + volumeData.Storage = storageID volumeData.Format = c.Format volumeData.Size = uint64(c.Size) volumeData.Volid = volumeID } } - return &volumeData, volumeID, storageID, nil + return &volumeData, nil } func GetNetInfo(net string) (*Net, error) { diff --git a/app/types.go b/app/types.go index ce3aa3a..1ced31f 100644 --- a/app/types.go +++ b/app/types.go @@ -9,17 +9,17 @@ import ( type Cluster struct { lock sync.Mutex pve ProxmoxClient - Hosts map[string]*Host + Nodes map[string]*Node } -type Host struct { +type Node struct { lock sync.Mutex - Name string - Cores uint64 - Memory uint64 - Swap uint64 - Devices map[string]*Device - Instances map[uint]*Instance + Name string `json:"name"` + Cores uint64 `json:"cores"` + Memory uint64 `json:"memory"` + Swap uint64 `json:"swap"` + Devices map[string]*Device `json:"devices"` + Instances map[uint]*Instance `json:"instances"` pvenode *proxmox.Node } @@ -32,15 +32,15 @@ const ( type Instance struct { lock sync.Mutex - Type InstanceType - Name string - Proctype string - Cores uint64 - Memory uint64 - Swap uint64 - Volumes map[string]*Volume - Nets map[uint]*Net - Devices map[uint][]*Device + Type InstanceType `json:"type"` + Name string `json:"name"` + Proctype string `json:"cpu"` + Cores uint64 `json:"cores"` + Memory uint64 `json:"memory"` + Swap uint64 `json:"swap"` + Volumes map[string]*Volume `json:"volumes"` + Nets map[uint]*Net `json:"nets"` + Devices map[uint][]*Device `json:"devices"` pveconfig interface{} configDisks map[string]string configNets map[string]string @@ -48,15 +48,15 @@ type Instance struct { } type Volume struct { - Path string - Format string - Size uint64 - Volid string + Storage string `json:"storage"` + Format string `json:"format"` + Size uint64 `json:"size"` + Volid string `json:"volid"` } type Net struct { - Rate uint64 - VLAN uint64 + Rate uint64 `json:"rate"` + VLAN uint64 `json:"vlan"` } type Device struct { @@ -65,5 +65,5 @@ type Device struct { VendorName string `json:"vendor_name"` SubsystemDeviceName string `json:"subsystem_device_name"` SubsystemVendorName string `json:"subsystem_vendor_name"` - Reserved bool + Reserved bool `json:"reserved"` }