diff --git a/proxmox/client.go b/proxmox/client.go index b1edab45..9a3b4a73 100644 --- a/proxmox/client.go +++ b/proxmox/client.go @@ -138,7 +138,7 @@ func (c *Client) GetJsonRetryable(url string, data *map[string]interface{}, trie if strings.Contains(statErr.Error(), "500 no such resource") { return statErr } - //fmt.Printf("[DEBUG][GetJsonRetryable] Sleeping for %d seconds before asking url %s", ii+1, url) + // fmt.Printf("[DEBUG][GetJsonRetryable] Sleeping for %d seconds before asking url %s", ii+1, url) time.Sleep(time.Duration(ii+1) * time.Second) } return statErr @@ -153,7 +153,7 @@ func (c *Client) GetNodeList() (list map[string]interface{}, err error) { // For resource types that can be in a disabled state, disabled resources // will not be returned func (c *Client) GetResourceList(resourceType string) (list map[string]interface{}, err error) { - var endpoint = "/cluster/resources" + endpoint := "/cluster/resources" if resourceType != "" { endpoint = fmt.Sprintf("%s?type=%s", endpoint, resourceType) } @@ -443,8 +443,10 @@ func (c *Client) WaitForCompletion(taskResponse map[string]interface{}) (waitExi return "", fmt.Errorf("Wait timeout for:" + taskUpid) } -var rxTaskNode = regexp.MustCompile("UPID:(.*?):") -var rxExitStatusSuccess = regexp.MustCompile(`^(OK|WARNINGS)`) +var ( + rxTaskNode = regexp.MustCompile("UPID:(.*?):") + rxExitStatusSuccess = regexp.MustCompile(`^(OK|WARNINGS)`) +) func (c *Client) GetTaskExitstatus(taskUpid string) (exitStatus interface{}, err error) { node := rxTaskNode.FindStringSubmatch(taskUpid)[1] @@ -518,7 +520,7 @@ func (c *Client) DeleteVmParams(vmr *VmRef, params map[string]interface{}) (exit return "", err } - //Remove HA if required + // Remove HA if required if vmr.haState != "" { url := fmt.Sprintf("/cluster/ha/resources/%d", vmr.vmId) resp, err := c.session.Delete(url, nil, nil) @@ -616,7 +618,6 @@ func (c *Client) CreateLxcContainer(node string, vmParams map[string]interface{} } func (c *Client) CloneLxcContainer(vmr *VmRef, vmParams map[string]interface{}) (exitStatus string, err error) { - reqbody := ParamsToBody(vmParams) url := fmt.Sprintf("/nodes/%s/lxc/%s/clone", vmr.node, vmParams["vmid"]) resp, err := c.session.Post(url, nil, nil, &reqbody) @@ -880,7 +881,6 @@ func (c *Client) CreateVMDisk( fullDiskName string, diskParams map[string]interface{}, ) error { - reqbody := ParamsToBody(diskParams) url := fmt.Sprintf("/nodes/%s/storage/%s/content", nodeName, storageName) resp, err := c.session.Post(url, nil, nil, &reqbody) @@ -936,7 +936,6 @@ func (c *Client) createVMDisks( // CreateNewDisk - This method allows simpler disk creation for direct client users // It should work for any existing container and virtual machine func (c *Client) CreateNewDisk(vmr *VmRef, disk string, volume string) (exitStatus interface{}, err error) { - reqbody := ParamsToBody(map[string]interface{}{disk: volume}) url := fmt.Sprintf("/nodes/%s/%s/%d/config", vmr.node, vmr.vmType, vmr.vmId) resp, err := c.session.Put(url, nil, nil, &reqbody) @@ -1332,6 +1331,47 @@ func (c *Client) Upload(node string, storage string, contentType string, filenam return nil } +func (c *Client) UploadLargeFile(node string, storage string, contentType string, filename string, filesize int64, file io.Reader) error { + var contentLength int64 + + var body io.Reader + var mimetype string + var err error + body, mimetype, contentLength, err = createStreamedUploadBody(contentType, filename, filesize, file) + if err != nil { + return err + } + + url := fmt.Sprintf("%s/nodes/%s/storage/%s/upload", c.session.ApiUrl, node, storage) + headers := c.session.Headers.Clone() + headers.Add("Content-Type", mimetype) + headers.Add("Accept", "application/json") + req, err := c.session.NewRequest(http.MethodPost, url, &headers, body) + if err != nil { + return err + } + + req.ContentLength = contentLength + + resp, err := c.session.Do(req) + if err != nil { + return err + } + + taskResponse, err := ResponseJSON(resp) + if err != nil { + return err + } + exitStatus, err := c.WaitForCompletion(taskResponse) + if err != nil { + return err + } + if exitStatus != exitStatusSuccess { + return fmt.Errorf("moving file to destination failed: %v", exitStatus) + } + return nil +} + func createUploadBody(contentType string, filename string, r io.Reader) (io.Reader, string, error) { var buf bytes.Buffer w := multipart.NewWriter(&buf) @@ -1476,7 +1516,6 @@ func (c *Client) ReadVMHA(vmr *VmRef) (err error) { } } return - } func (c *Client) UpdateVMHA(vmr *VmRef, haState string, haGroup string) (exitStatus interface{}, err error) { @@ -1485,7 +1524,7 @@ func (c *Client) UpdateVMHA(vmr *VmRef, haState string, haGroup string) (exitSta return } - //Remove HA + // Remove HA if haState == "" { url := fmt.Sprintf("/cluster/ha/resources/%d", vmr.vmId) resp, err := c.session.Delete(url, nil, nil) @@ -1502,7 +1541,7 @@ func (c *Client) UpdateVMHA(vmr *VmRef, haState string, haGroup string) (exitSta return nil, err } - //Activate HA + // Activate HA if vmr.haState == "" { paramMap := map[string]interface{}{ "sid": vmr.vmId, @@ -1525,7 +1564,7 @@ func (c *Client) UpdateVMHA(vmr *VmRef, haState string, haGroup string) (exitSta } } - //Set wanted state + // Set wanted state paramMap := map[string]interface{}{ "state": haState, "group": haGroup, @@ -1573,8 +1612,7 @@ func (c *Client) DeletePool(poolid string) error { return c.Delete("/pools/" + poolid) } -//permissions check - +// permissions check func (c *Client) GetUserPermissions(id UserID, path string) (permissions []string, err error) { existence, err := CheckUserExistence(id, c) if err != nil {