2019-08-03 00:25:22 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2019-08-10 16:14:55 +00:00
|
|
|
|
|
|
|
portainer "github.com/portainer/portainer/api"
|
2019-08-03 00:25:22 +00:00
|
|
|
)
|
|
|
|
|
2019-08-23 17:08:08 +00:00
|
|
|
// StackListFilter represents a filter for a stack list
|
2019-08-03 00:25:22 +00:00
|
|
|
type StackListFilter struct {
|
2019-08-23 16:11:03 +00:00
|
|
|
SwarmID string `json:"SwarmId,omitempty"`
|
|
|
|
EndpointID portainer.EndpointID `json:"EndpointId,omitempty"`
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
|
2019-08-23 17:08:08 +00:00
|
|
|
// Config represents a Portainer client configuration
|
2019-08-03 00:28:48 +00:00
|
|
|
type Config struct {
|
2019-08-23 16:11:03 +00:00
|
|
|
URL *url.URL
|
2019-08-03 00:25:22 +00:00
|
|
|
User string
|
|
|
|
Password string
|
|
|
|
Token string
|
2019-08-09 19:16:12 +00:00
|
|
|
UserAgent string
|
2019-08-03 00:25:22 +00:00
|
|
|
DoNotUseToken bool
|
|
|
|
}
|
|
|
|
|
2019-08-23 17:08:08 +00:00
|
|
|
// PortainerClient represents a Portainer API client
|
2019-08-03 00:25:22 +00:00
|
|
|
type PortainerClient interface {
|
2019-08-26 05:13:43 +00:00
|
|
|
// Auth a user to get an auth token
|
|
|
|
Auth() (token string, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Get endpoints
|
2019-08-26 05:15:44 +00:00
|
|
|
EndpointList() ([]portainer.Endpoint, error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
2019-08-09 17:17:40 +00:00
|
|
|
// Get endpoint groups
|
2019-08-26 05:16:40 +00:00
|
|
|
EndpointGroupList() ([]portainer.EndpointGroup, error)
|
2019-08-09 17:17:40 +00:00
|
|
|
|
2019-08-06 23:31:30 +00:00
|
|
|
// Get stacks, optionally filtered by swarmId and endpointId
|
2019-08-26 05:17:41 +00:00
|
|
|
StackList(swarmID string, endpointID portainer.EndpointID) ([]portainer.Stack, error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Create swarm stack
|
2019-08-26 05:18:46 +00:00
|
|
|
StackCreateSwarm(stackName string, environmentVariables []portainer.Pair, stackFileContent string, swarmClusterID string, endpointID portainer.EndpointID) (stack portainer.Stack, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Create compose stack
|
2019-08-26 05:19:41 +00:00
|
|
|
StackCreateCompose(stackName string, environmentVariables []portainer.Pair, stackFileContent string, endpointID portainer.EndpointID) (stack portainer.Stack, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Update stack
|
2019-08-26 05:19:41 +00:00
|
|
|
StackUpdate(stack portainer.Stack, environmentVariables []portainer.Pair, stackFileContent string, prune bool, endpointID portainer.EndpointID) error
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Delete stack
|
2019-08-26 05:21:14 +00:00
|
|
|
StackDelete(stackID portainer.StackID) error
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Get stack file content
|
2019-08-23 16:11:03 +00:00
|
|
|
GetStackFileContent(stackID portainer.StackID) (content string, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Get endpoint Docker info
|
2019-08-23 16:11:03 +00:00
|
|
|
GetEndpointDockerInfo(endpointID portainer.EndpointID) (info map[string]interface{}, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Get Portainer status info
|
2019-08-10 16:14:55 +00:00
|
|
|
GetStatus() (portainer.Status, error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Run a function before sending a request to Portainer
|
2019-08-06 03:19:35 +00:00
|
|
|
BeforeRequest(hook func(req *http.Request) (err error))
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Run a function after receiving a response from Portainer
|
2019-08-06 03:19:35 +00:00
|
|
|
AfterResponse(hook func(resp *http.Response) (err error))
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
|
2019-08-03 00:56:19 +00:00
|
|
|
type portainerClientImp struct {
|
2019-08-06 03:19:35 +00:00
|
|
|
httpClient *http.Client
|
|
|
|
url *url.URL
|
|
|
|
user string
|
|
|
|
password string
|
|
|
|
token string
|
2019-08-09 19:16:12 +00:00
|
|
|
userAgent string
|
2019-08-06 03:19:35 +00:00
|
|
|
beforeRequestHooks []func(req *http.Request) (err error)
|
|
|
|
afterResponseHooks []func(resp *http.Response) (err error)
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if an http.Response object has errors
|
|
|
|
func checkResponseForErrors(resp *http.Response) error {
|
|
|
|
if 300 <= resp.StatusCode {
|
|
|
|
// Guess it's a GenericError
|
|
|
|
respBody := GenericError{}
|
|
|
|
err := json.NewDecoder(resp.Body).Decode(&respBody)
|
|
|
|
if err != nil {
|
|
|
|
// It's not a GenericError
|
|
|
|
bodyBytes, err := ioutil.ReadAll(resp.Body)
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
resp.Body = ioutil.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
return errors.New(string(bodyBytes))
|
|
|
|
}
|
|
|
|
return &respBody
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do an http request
|
2019-08-24 19:06:16 +00:00
|
|
|
func (n *portainerClientImp) do(uri, method string, requestBody io.Reader, headers http.Header) (resp *http.Response, err error) {
|
2019-08-23 16:11:03 +00:00
|
|
|
requestURL, err := n.url.Parse(uri)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-24 19:06:16 +00:00
|
|
|
req, err := http.NewRequest(method, requestURL.String(), requestBody)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if headers != nil {
|
|
|
|
req.Header = headers
|
|
|
|
}
|
|
|
|
|
2019-08-24 19:39:56 +00:00
|
|
|
// Set user agent header
|
|
|
|
req.Header.Set("User-Agent", n.userAgent)
|
2019-08-03 00:25:22 +00:00
|
|
|
|
2019-08-06 03:19:35 +00:00
|
|
|
// Run all "before request" hooks
|
|
|
|
for i := 0; i < len(n.beforeRequestHooks); i++ {
|
|
|
|
err = n.beforeRequestHooks[i](req)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-03 00:25:22 +00:00
|
|
|
resp, err = n.httpClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-06 03:19:35 +00:00
|
|
|
// Run all "after response" hooks
|
|
|
|
for i := 0; i < len(n.afterResponseHooks); i++ {
|
|
|
|
err = n.afterResponseHooks[i](resp)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-03 00:25:22 +00:00
|
|
|
err = checkResponseForErrors(resp)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do a JSON http request
|
2019-08-24 19:06:16 +00:00
|
|
|
func (n *portainerClientImp) doJSON(uri, method string, headers http.Header, requestBody interface{}, responseBody interface{}) error {
|
2019-08-03 00:25:22 +00:00
|
|
|
var body io.Reader
|
|
|
|
|
2019-08-24 19:06:16 +00:00
|
|
|
if requestBody != nil {
|
|
|
|
reqBodyBytes, err := json.Marshal(requestBody)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
body = bytes.NewReader(reqBodyBytes)
|
|
|
|
}
|
|
|
|
|
2019-08-24 13:05:34 +00:00
|
|
|
headers.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
resp, err := n.do(uri, method, body, headers)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-24 19:06:16 +00:00
|
|
|
if responseBody != nil {
|
2019-08-03 00:25:22 +00:00
|
|
|
d := json.NewDecoder(resp.Body)
|
2019-08-24 19:06:16 +00:00
|
|
|
err := d.Decode(responseBody)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
// Do a JSON http request with an auth token
|
|
|
|
func (n *portainerClientImp) doJSONWithToken(uri, method string, headers http.Header, request interface{}, response interface{}) (err error) {
|
|
|
|
// Ensure there is an auth token
|
|
|
|
if n.token == "" {
|
2019-08-26 05:13:43 +00:00
|
|
|
n.token, err = n.Auth()
|
2019-08-24 13:01:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
headers.Set("Authorization", "Bearer "+n.token)
|
|
|
|
|
|
|
|
return n.doJSON(uri, method, headers, request, response)
|
|
|
|
}
|
|
|
|
|
2019-08-06 03:19:35 +00:00
|
|
|
func (n *portainerClientImp) BeforeRequest(hook func(req *http.Request) (err error)) {
|
|
|
|
n.beforeRequestHooks = append(n.beforeRequestHooks, hook)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *portainerClientImp) AfterResponse(hook func(resp *http.Response) (err error)) {
|
|
|
|
n.afterResponseHooks = append(n.afterResponseHooks, hook)
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:13:43 +00:00
|
|
|
func (n *portainerClientImp) Auth() (token string, err error) {
|
2019-08-03 00:25:22 +00:00
|
|
|
reqBody := AuthenticateUserRequest{
|
|
|
|
Username: n.user,
|
|
|
|
Password: n.password,
|
|
|
|
}
|
|
|
|
|
|
|
|
respBody := AuthenticateUserResponse{}
|
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSON("auth", http.MethodPost, http.Header{}, &reqBody, &respBody)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
token = respBody.Jwt
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:15:44 +00:00
|
|
|
func (n *portainerClientImp) EndpointList() (endpoints []portainer.Endpoint, err error) {
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken("endpoints", http.MethodGet, http.Header{}, nil, &endpoints)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:16:40 +00:00
|
|
|
func (n *portainerClientImp) EndpointGroupList() (endpointGroups []portainer.EndpointGroup, err error) {
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken("endpoint_groups", http.MethodGet, http.Header{}, nil, &endpointGroups)
|
2019-08-09 17:17:40 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:17:41 +00:00
|
|
|
func (n *portainerClientImp) StackList(swarmID string, endpointID portainer.EndpointID) (stacks []portainer.Stack, err error) {
|
2019-08-03 00:25:22 +00:00
|
|
|
filter := StackListFilter{
|
2019-08-23 16:11:03 +00:00
|
|
|
SwarmID: swarmID,
|
|
|
|
EndpointID: endpointID,
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
|
2019-08-23 16:11:03 +00:00
|
|
|
filterJSONBytes, _ := json.Marshal(filter)
|
|
|
|
filterJSONString := string(filterJSONBytes)
|
2019-08-03 00:25:22 +00:00
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("stacks?filters=%s", filterJSONString), http.MethodGet, http.Header{}, nil, &stacks)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:18:46 +00:00
|
|
|
func (n *portainerClientImp) StackCreateSwarm(stackName string, environmentVariables []portainer.Pair, stackFileContent string, swarmClusterID string, endpointID portainer.EndpointID) (stack portainer.Stack, err error) {
|
2019-08-03 00:25:22 +00:00
|
|
|
reqBody := StackCreateRequest{
|
|
|
|
Name: stackName,
|
|
|
|
Env: environmentVariables,
|
2019-08-23 16:11:03 +00:00
|
|
|
SwarmID: swarmClusterID,
|
2019-08-03 00:25:22 +00:00
|
|
|
StackFileContent: stackFileContent,
|
|
|
|
}
|
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("stacks?type=%v&method=%s&endpointId=%v", 1, "string", endpointID), http.MethodPost, http.Header{}, &reqBody, &stack)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:19:41 +00:00
|
|
|
func (n *portainerClientImp) StackCreateCompose(stackName string, environmentVariables []portainer.Pair, stackFileContent string, endpointID portainer.EndpointID) (stack portainer.Stack, err error) {
|
2019-08-03 00:25:22 +00:00
|
|
|
reqBody := StackCreateRequest{
|
|
|
|
Name: stackName,
|
|
|
|
Env: environmentVariables,
|
|
|
|
StackFileContent: stackFileContent,
|
|
|
|
}
|
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("stacks?type=%v&method=%s&endpointId=%v", 2, "string", endpointID), http.MethodPost, http.Header{}, &reqBody, &stack)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:19:41 +00:00
|
|
|
func (n *portainerClientImp) StackUpdate(stack portainer.Stack, environmentVariables []portainer.Pair, stackFileContent string, prune bool, endpointID portainer.EndpointID) (err error) {
|
2019-08-03 00:25:22 +00:00
|
|
|
reqBody := StackUpdateRequest{
|
|
|
|
Env: environmentVariables,
|
|
|
|
StackFileContent: stackFileContent,
|
|
|
|
Prune: prune,
|
|
|
|
}
|
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("stacks/%v?endpointId=%v", stack.ID, endpointID), http.MethodPut, http.Header{}, &reqBody, nil)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-26 05:21:14 +00:00
|
|
|
func (n *portainerClientImp) StackDelete(stackID portainer.StackID) (err error) {
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("stacks/%d", stackID), http.MethodDelete, http.Header{}, nil, nil)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-23 16:11:03 +00:00
|
|
|
func (n *portainerClientImp) GetStackFileContent(stackID portainer.StackID) (content string, err error) {
|
2019-08-03 00:25:22 +00:00
|
|
|
var respBody StackFileInspectResponse
|
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("stacks/%v/file", stackID), http.MethodGet, http.Header{}, nil, &respBody)
|
2019-08-03 00:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
content = respBody.StackFileContent
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-23 16:11:03 +00:00
|
|
|
func (n *portainerClientImp) GetEndpointDockerInfo(endpointID portainer.EndpointID) (info map[string]interface{}, err error) {
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken(fmt.Sprintf("endpoints/%v/docker/info", endpointID), http.MethodGet, http.Header{}, nil, &info)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-10 16:14:55 +00:00
|
|
|
func (n *portainerClientImp) GetStatus() (status portainer.Status, err error) {
|
2019-08-24 13:01:27 +00:00
|
|
|
err = n.doJSONWithToken("status", http.MethodGet, http.Header{}, nil, &status)
|
2019-08-03 00:25:22 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-23 17:08:08 +00:00
|
|
|
// NewClient creates a new Portainer API client
|
2019-08-09 16:39:40 +00:00
|
|
|
func NewClient(httpClient *http.Client, config Config) PortainerClient {
|
|
|
|
return &portainerClientImp{
|
2019-08-03 00:25:22 +00:00
|
|
|
httpClient: httpClient,
|
2019-08-23 16:11:03 +00:00
|
|
|
url: config.URL,
|
2019-08-03 00:25:22 +00:00
|
|
|
user: config.User,
|
|
|
|
password: config.Password,
|
|
|
|
token: config.Token,
|
2019-08-09 19:16:12 +00:00
|
|
|
userAgent: config.UserAgent,
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
}
|