forked from Azure/azure-extensions-cli
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sdk.go
135 lines (117 loc) · 4.59 KB
/
sdk.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
package main
import (
"encoding/xml"
"fmt"
"time"
"github.com/Azure/azure-sdk-for-go/management"
log "github.com/Sirupsen/logrus"
)
const (
operationStatusPollingInterval = time.Second * 10
apiVersion = "2015-04-01"
)
// ExtensionsClient builds a new Azure Service Management Client with Extension
// Publishing operations.
type ExtensionsClient struct {
client management.Client
}
// NewClient constructs an ExtensionsClient.
func NewClient(mgtURL string, subscriptionID string, cert []byte) (ExtensionsClient, error) {
cfg := management.DefaultConfig()
cfg.APIVersion = apiVersion
cfg.ManagementURL = mgtURL
cl, err := management.NewClientFromConfig(subscriptionID, cert, cfg)
return ExtensionsClient{cl}, err
}
// ListVersionsResponse is response returned from Publisher Extensions endpoint.
type ListVersionsResponse struct {
XMLName xml.Name `xml:"ExtensionImages"`
Extensions []struct {
Ns string `xml:"ProviderNameSpace"`
Name string `xml:"Type"`
Version string `xml:"Version"`
ReplicationCompleted bool `xml:"ReplicationCompleted"`
Regions string `xml:"Regions"`
IsInternal bool `xml:"IsInternalExtension"`
} `xml:"ExtensionImage"`
}
// ListVersions returns all the published extensions and their versions from the
// publisher subscription.
func (c ExtensionsClient) ListVersions() (ListVersionsResponse, error) {
var l ListVersionsResponse
response, err := c.client.SendAzureGetRequest("services/publisherextensions")
if err != nil {
return l, err
}
err = xml.Unmarshal(response, &l)
return l, err
}
// ReplicationStatusResponse is the response contents of the Get Replication
// Status endpoint.
type ReplicationStatusResponse struct {
XMLName xml.Name `xml:"ReplicationStatusList"`
Statuses []struct {
Location string `xml:"Location"`
Status string `xml:"Status"`
} `xml:"ReplicationStatus"`
}
// GetReplicationStatus retrieves the replication status of the specified
// extension handler.
func (c ExtensionsClient) GetReplicationStatus(publisherNamespace, extension,
version string) (ReplicationStatusResponse, error) {
var l ReplicationStatusResponse
response, err := c.client.SendAzureGetRequest(fmt.Sprintf("services/extensions/%s/%s/%s/replicationstatus",
publisherNamespace, extension, version))
if err != nil {
return l, err
}
err = xml.Unmarshal(response, &l)
return l, err
}
// CreateExtension sends the given extension handler definition XML to create a
// brand new extension (not a version). Returned operation ID should be polled for result.
func (c ExtensionsClient) CreateExtension(data []byte) (management.OperationID, error) {
return c.client.SendAzurePostRequest("services/extensions", data)
}
// UpdateExtension sends the given extension handler definition XML to issue and update
// request. Returned operation ID should be polled for result.
func (c ExtensionsClient) UpdateExtension(data []byte) (management.OperationID, error) {
return c.client.SendAzurePutRequest("services/extensions?action=update", "text/xml", data)
}
// DeleteExtension deletes the extension version. It should be marked as internal first.
// Returned operation ID should be polled for result.
func (c ExtensionsClient) DeleteExtension(namespace, name, version string) (management.OperationID, error) {
return c.client.SendAzureDeleteRequest(fmt.Sprintf("services/extensions/%s/%s/%s", namespace, name, version))
}
// WaitForOperation polls indefinitely until the specified Azure Service
// Management REST API operation ID reaches a terminal state. If operation
// fails, it wraps the error and returns it.
func (c ExtensionsClient) WaitForOperation(opID management.OperationID) error {
lg := log.WithField("x-ms-operation-id", opID)
lg.Debug("Waiting for operation to complete.")
for {
op, err := c.client.GetOperationStatus(opID)
if err != nil {
log.Errorf("Error fetching operation status: %v", err)
continue // don't return because of GetOperationStatus flakiness.
}
switch op.Status {
case management.OperationStatusSucceeded:
lg.Debug("Operation successful.")
return nil
case management.OperationStatusFailed:
lg.Debug("Operation failed.")
if op.Error != nil {
return op.Error
}
return fmt.Errorf("Azure Operation (x-ms-request-id=%s) has failed", opID)
case management.OperationStatusInProgress:
lg.Debug("Operation in progress...")
time.Sleep(operationStatusPollingInterval)
continue
default:
lg.Errorf("Encoutered unhandled operation status: %v", op.Status)
return fmt.Errorf("Unhandled operation status returned from API: %s", op.Status)
}
}
}