feat: working update
This commit is contained in:
121
proxmox.go
121
proxmox.go
@@ -9,6 +9,7 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
@@ -123,6 +124,9 @@ func (p *Plugin) Create(ctx context.Context, req *resource.CreateRequest) (*reso
|
|||||||
client := &http.Client{}
|
client := &http.Client{}
|
||||||
|
|
||||||
arguments := "vmid=" + props.VMID + "&ostemplate=" + props.OSTemplate + "&hostname=" + props.Hostname
|
arguments := "vmid=" + props.VMID + "&ostemplate=" + props.OSTemplate + "&hostname=" + props.Hostname
|
||||||
|
if props.Description != "" {
|
||||||
|
arguments += "&description=" + props.Description
|
||||||
|
}
|
||||||
|
|
||||||
request, err := http.NewRequest("POST", config.URL+"/api2/json/nodes/"+config.NODE+"/lxc", bytes.NewBuffer([]byte(arguments)))
|
request, err := http.NewRequest("POST", config.URL+"/api2/json/nodes/"+config.NODE+"/lxc", bytes.NewBuffer([]byte(arguments)))
|
||||||
request.Header.Set("Authorization", "PVEAPIToken="+username+"="+token)
|
request.Header.Set("Authorization", "PVEAPIToken="+username+"="+token)
|
||||||
@@ -223,22 +227,115 @@ func (p *Plugin) Read(ctx context.Context, req *resource.ReadRequest) (*resource
|
|||||||
|
|
||||||
// Update modifies an existing resource.
|
// Update modifies an existing resource.
|
||||||
func (p *Plugin) Update(ctx context.Context, req *resource.UpdateRequest) (*resource.UpdateResult, error) {
|
func (p *Plugin) Update(ctx context.Context, req *resource.UpdateRequest) (*resource.UpdateResult, error) {
|
||||||
// TODO: Implement resource update
|
|
||||||
//
|
prior, err := parseLXCProperties(req.PriorProperties)
|
||||||
// 1. Use req.NativeID to identify the resource
|
if err != nil {
|
||||||
// 2. Use req.PatchDocument for changes (JSON Patch format)
|
return &resource.UpdateResult{
|
||||||
// Or compare req.PriorProperties with req.DesiredProperties
|
ProgressResult: &resource.ProgressResult{
|
||||||
// 3. Call your provider's API to apply changes
|
Operation: resource.OperationUpdate,
|
||||||
// 4. Return ProgressResult with status
|
OperationStatus: resource.OperationStatusFailure,
|
||||||
|
ErrorCode: resource.OperationErrorCodeInvalidRequest,
|
||||||
|
StatusMessage: err.Error(),
|
||||||
|
},
|
||||||
|
}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
desir, err := parseLXCProperties(req.DesiredProperties)
|
||||||
|
if err != nil {
|
||||||
|
return &resource.UpdateResult{
|
||||||
|
ProgressResult: &resource.ProgressResult{
|
||||||
|
Operation: resource.OperationUpdate,
|
||||||
|
OperationStatus: resource.OperationStatusFailure,
|
||||||
|
ErrorCode: resource.OperationErrorCodeInvalidRequest,
|
||||||
|
StatusMessage: err.Error(),
|
||||||
|
},
|
||||||
|
}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if prior == nil {
|
||||||
|
p.Create(ctx, &resource.CreateRequest{
|
||||||
|
ResourceType: req.ResourceType,
|
||||||
|
Label: req.Label,
|
||||||
|
Properties: req.DesiredProperties,
|
||||||
|
TargetConfig: req.TargetConfig,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if prior.VMID != desir.VMID {
|
||||||
|
return &resource.UpdateResult{
|
||||||
|
ProgressResult: &resource.ProgressResult{
|
||||||
|
Operation: resource.OperationUpdate,
|
||||||
|
OperationStatus: resource.OperationStatusFailure,
|
||||||
|
ErrorCode: resource.OperationErrorCodeInvalidRequest,
|
||||||
|
StatusMessage: "can't change vmid",
|
||||||
|
},
|
||||||
|
}, fmt.Errorf("can't change vmid")
|
||||||
|
}
|
||||||
|
|
||||||
|
if prior.Hostname != desir.Hostname || prior.Description != desir.Description {
|
||||||
|
config, err := parseTargetConfig(req.TargetConfig)
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err.Error())
|
||||||
|
return &resource.UpdateResult{
|
||||||
|
ProgressResult: &resource.ProgressResult{
|
||||||
|
Operation: resource.OperationCreate,
|
||||||
|
OperationStatus: resource.OperationStatusFailure,
|
||||||
|
ErrorCode: resource.OperationErrorCodeInternalFailure,
|
||||||
|
StatusMessage: err.Error(),
|
||||||
|
},
|
||||||
|
}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
username, token, err := getCredentials()
|
||||||
|
if err != nil {
|
||||||
|
return &resource.UpdateResult{
|
||||||
|
ProgressResult: &resource.ProgressResult{
|
||||||
|
Operation: resource.OperationUpdate,
|
||||||
|
OperationStatus: resource.OperationStatusFailure,
|
||||||
|
ErrorCode: resource.OperationErrorCodeAccessDenied,
|
||||||
|
StatusMessage: err.Error(),
|
||||||
|
},
|
||||||
|
}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
client := &http.Client{}
|
||||||
|
|
||||||
|
url := config.URL + "/api2/json/nodes/" + config.NODE + "/lxc/" + desir.VMID + "/config"
|
||||||
|
arguments := "vmid=" + desir.VMID + "&hostname=" + desir.Hostname + "&description=" + desir.Description
|
||||||
|
|
||||||
|
argumentBuffer := bytes.NewBuffer([]byte(arguments))
|
||||||
|
request, err := http.NewRequest("PUT", url, argumentBuffer)
|
||||||
|
request.Header.Set("Authorization", "PVEAPIToken="+username+"="+token)
|
||||||
|
|
||||||
|
resp, err := client.Do(request)
|
||||||
|
if err != nil {
|
||||||
|
return &resource.UpdateResult{
|
||||||
|
ProgressResult: &resource.ProgressResult{
|
||||||
|
Operation: resource.OperationCreate,
|
||||||
|
OperationStatus: resource.OperationStatusFailure,
|
||||||
|
ErrorCode: resource.OperationErrorCodeInternalFailure,
|
||||||
|
StatusMessage: err.Error(),
|
||||||
|
},
|
||||||
|
}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Println("Response StatusCode: ", resp.Status)
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := p.Read(ctx, &resource.ReadRequest{
|
||||||
|
NativeID: req.NativeID,
|
||||||
|
ResourceType: req.ResourceType,
|
||||||
|
TargetConfig: req.TargetConfig,
|
||||||
|
})
|
||||||
|
|
||||||
return &resource.UpdateResult{
|
return &resource.UpdateResult{
|
||||||
ProgressResult: &resource.ProgressResult{
|
ProgressResult: &resource.ProgressResult{
|
||||||
Operation: resource.OperationUpdate,
|
Operation: resource.OperationUpdate,
|
||||||
OperationStatus: resource.OperationStatusFailure,
|
OperationStatus: resource.OperationStatusSuccess,
|
||||||
ErrorCode: resource.OperationErrorCodeInternalFailure,
|
NativeID: req.NativeID,
|
||||||
StatusMessage: "Update not implemented",
|
ResourceProperties: json.RawMessage(result.Properties),
|
||||||
},
|
},
|
||||||
}, ErrNotImplemented
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete removes a resource.
|
// Delete removes a resource.
|
||||||
|
|||||||
@@ -107,3 +107,48 @@ func TestRead(t *testing.T) {
|
|||||||
require.Equal(t, "ntfy", props["hostname"], "hostname should match")
|
require.Equal(t, "ntfy", props["hostname"], "hostname should match")
|
||||||
require.Equal(t, strconv.Itoa(120), props["vmid"], "vmid should match")
|
require.Equal(t, strconv.Itoa(120), props["vmid"], "vmid should match")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestUpdate(t *testing.T) {
|
||||||
|
ctx := context.Background()
|
||||||
|
plugin := &Plugin{}
|
||||||
|
|
||||||
|
priorProperties, _ := json.Marshal(map[string]any{
|
||||||
|
"vmid": "200",
|
||||||
|
"hostname": "testlxc",
|
||||||
|
"description": "none",
|
||||||
|
"ostemplate": "local:vztmpl/alpine-3.22-default_20250617_amd64.tar.xz",
|
||||||
|
})
|
||||||
|
|
||||||
|
desiredProperties, _ := json.Marshal(map[string]any{
|
||||||
|
"vmid": "200",
|
||||||
|
"hostname": "testlxc-updated",
|
||||||
|
"description": "none",
|
||||||
|
"ostemplate": "local:vztmpl/alpine-3.22-default_20250617_amd64.tar.xz",
|
||||||
|
})
|
||||||
|
|
||||||
|
req := &resource.UpdateRequest{
|
||||||
|
NativeID: "200",
|
||||||
|
ResourceType: "PROXMOX::Service::LXC",
|
||||||
|
PriorProperties: priorProperties,
|
||||||
|
DesiredProperties: desiredProperties,
|
||||||
|
TargetConfig: testTargetConfig(),
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := plugin.Update(ctx, req)
|
||||||
|
require.NoError(t, err, "Update should not return error")
|
||||||
|
require.NotNil(t, result.ProgressResult, "Update should return ProgressResult")
|
||||||
|
require.Equal(t, resource.OperationStatusSuccess, result.ProgressResult.OperationStatus, "Update should return Success status")
|
||||||
|
|
||||||
|
readReq := &resource.ReadRequest{
|
||||||
|
NativeID: strconv.Itoa(200),
|
||||||
|
ResourceType: "PROXMOX::Service::LXC",
|
||||||
|
TargetConfig: testTargetConfig(),
|
||||||
|
}
|
||||||
|
|
||||||
|
readResult, err := plugin.Read(ctx, readReq)
|
||||||
|
var props map[string]any
|
||||||
|
|
||||||
|
err = json.Unmarshal([]byte(readResult.Properties), &props)
|
||||||
|
require.Equal(t, "testlxc-updated", props["hostname"], "hostname should have changed")
|
||||||
|
// test if update has happened
|
||||||
|
}
|
||||||
|
|||||||
8
types.go
8
types.go
@@ -20,6 +20,14 @@ type ReadRequest struct {
|
|||||||
TargetConfig json.RawMessage
|
TargetConfig json.RawMessage
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type UpdateRequest struct {
|
||||||
|
NativeID string
|
||||||
|
ResourceType string
|
||||||
|
PriorProperties json.RawMessage
|
||||||
|
DesiredProperties json.RawMessage
|
||||||
|
TargetConfig json.RawMessage
|
||||||
|
}
|
||||||
|
|
||||||
type StatusLXCGeneral struct {
|
type StatusLXCGeneral struct {
|
||||||
Status string `json:"status"`
|
Status string `json:"status"`
|
||||||
NetIn int `json:"netin"`
|
NetIn int `json:"netin"`
|
||||||
|
|||||||
Reference in New Issue
Block a user