2019-08-03 00:25:22 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"io"
|
|
|
|
"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
|
|
|
// 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 06:34:01 +00:00
|
|
|
// AuthenticateUser a user to get an auth token
|
2019-08-26 06:53:42 +00:00
|
|
|
AuthenticateUser(options AuthenticateUserOptions) (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:40:15 +00:00
|
|
|
StackList(options StackListOptions) ([]portainer.Stack, error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Create swarm stack
|
2019-08-26 05:46:28 +00:00
|
|
|
StackCreateSwarm(options StackCreateSwarmOptions) (stack portainer.Stack, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Create compose stack
|
2019-08-26 05:48:59 +00:00
|
|
|
StackCreateCompose(options StackCreateComposeOptions) (stack portainer.Stack, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Update stack
|
2019-08-26 05:52:41 +00:00
|
|
|
StackUpdate(options StackUpdateOptions) 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-26 05:24:53 +00:00
|
|
|
StackFileInspect(stackID portainer.StackID) (content string, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Get endpoint Docker info
|
2019-08-26 05:25:54 +00:00
|
|
|
EndpointDockerInfo(endpointID portainer.EndpointID) (info map[string]interface{}, err error)
|
2019-08-06 23:31:30 +00:00
|
|
|
|
|
|
|
// Get Portainer status info
|
2019-08-26 05:29:21 +00:00
|
|
|
Status() (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-23 06:18:18 +00:00
|
|
|
|
|
|
|
// Proxy proxies a request to /endpoint/{id}/docker and returns its result
|
2019-08-27 03:35:34 +00:00
|
|
|
Proxy(endpointID portainer.EndpointID, req *http.Request) (resp *http.Response, err error)
|
2019-09-09 04:13:00 +00:00
|
|
|
|
2019-09-09 04:15:46 +00:00
|
|
|
// UserList retrieves a list of users
|
|
|
|
UserList() (users []portainer.User, err error)
|
|
|
|
|
2019-09-09 04:13:00 +00:00
|
|
|
// GetUsername returns the user name used by the client
|
|
|
|
GetUsername() string
|
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
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-27 03:35:34 +00:00
|
|
|
func (n *portainerClientImp) doWithToken(uri, method string, requestBody io.Reader, headers http.Header) (resp *http.Response, err error) {
|
|
|
|
// Ensure there is an auth token
|
|
|
|
if n.token == "" {
|
|
|
|
n.token, err = n.AuthenticateUser(AuthenticateUserOptions{
|
|
|
|
Username: n.user,
|
|
|
|
Password: n.password,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
headers.Set("Authorization", "Bearer "+n.token)
|
|
|
|
|
|
|
|
return n.do(uri, method, requestBody, headers)
|
|
|
|
}
|
|
|
|
|
2019-08-03 00:25:22 +00:00
|
|
|
// 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-26 08:20:45 +00:00
|
|
|
// Encode request body, if any
|
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-26 08:20:45 +00:00
|
|
|
// Set content type header
|
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-09-01 22:07:28 +00:00
|
|
|
err = checkResponseForErrors(resp)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-26 08:20:45 +00:00
|
|
|
// Decode response body, if any
|
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 06:53:42 +00:00
|
|
|
n.token, err = n.AuthenticateUser(AuthenticateUserOptions{
|
|
|
|
Username: n.user,
|
|
|
|
Password: n.password,
|
|
|
|
})
|
2019-08-24 13:01:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
2019-08-24 13:01:27 +00:00
|
|
|
headers.Set("Authorization", "Bearer "+n.token)
|
2019-08-03 00:25:22 +00:00
|
|
|
|
2019-08-24 13:01:27 +00:00
|
|
|
return n.doJSON(uri, method, headers, request, response)
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
|
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)
|
2019-08-03 00:25:22 +00:00
|
|
|
}
|
|
|
|
|
2019-08-06 03:19:35 +00:00
|
|
|
func (n *portainerClientImp) AfterResponse(hook func(resp *http.Response) (err error)) {
|
|
|
|
n.afterResponseHooks = append(n.afterResponseHooks, hook)
|
2019-08-23 06:18:18 +00:00
|
|
|
}
|
|
|
|
|
2019-09-09 04:13:00 +00:00
|
|
|
func (n *portainerClientImp) GetUsername() string {
|
|
|
|
return n.user
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|