2019-08-02 02:48:38 +00:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/tls"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2019-08-02 20:09:54 +00:00
|
|
|
"strings"
|
2019-08-02 02:48:38 +00:00
|
|
|
|
|
|
|
"github.com/spf13/viper"
|
|
|
|
)
|
|
|
|
|
2019-08-02 23:37:32 +00:00
|
|
|
var cachedClient PortainerClient
|
2019-08-02 02:48:38 +00:00
|
|
|
|
2019-08-02 18:21:50 +00:00
|
|
|
type ClientConfig struct {
|
|
|
|
Url string
|
|
|
|
User string
|
|
|
|
Password string
|
|
|
|
Token string
|
|
|
|
DoNotUseToken bool
|
2019-08-02 17:26:51 +00:00
|
|
|
}
|
|
|
|
|
2019-08-02 23:37:32 +00:00
|
|
|
type PortainerClient interface {
|
|
|
|
Authenticate() (token string, err error)
|
|
|
|
GetEndpoints() ([]EndpointSubset, error)
|
|
|
|
GetStacks(swarmId string, endpointId uint32) ([]Stack, error)
|
|
|
|
CreateSwarmStack(stackName string, environmentVariables []StackEnv, stackFileContent string, swarmClusterId string, endpointId string) error
|
|
|
|
CreateComposeStack(stackName string, environmentVariables []StackEnv, stackFileContent string, endpointId string) error
|
|
|
|
UpdateStack(stack Stack, environmentVariables []StackEnv, stackFileContent string, prune bool, endpointId string) error
|
|
|
|
DeleteStack(stackId uint32) error
|
|
|
|
GetStackFileContent(stackId uint32) (content string, err error)
|
|
|
|
GetEndpointDockerInfo(endpointId string) (info map[string]interface{}, err error)
|
|
|
|
GetStatus() (Status, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
type PortainerClientImp struct {
|
2019-08-02 22:20:18 +00:00
|
|
|
httpClient *http.Client
|
2019-08-02 18:21:50 +00:00
|
|
|
url *url.URL
|
|
|
|
user string
|
|
|
|
password string
|
|
|
|
token string
|
|
|
|
doNotUseToken bool
|
2019-08-02 02:48:38 +00:00
|
|
|
}
|
|
|
|
|
2019-08-02 17:21:29 +00:00
|
|
|
// Check if an http.Response object has errors
|
2019-08-02 02:48:38 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:21:29 +00:00
|
|
|
// Do an http request
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) do(uri, method string, request io.Reader, requestType string, headers http.Header) (resp *http.Response, err error) {
|
2019-08-02 02:48:38 +00:00
|
|
|
requestUrl, err := n.url.Parse(uri)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest(method, requestUrl.String(), request)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if headers != nil {
|
|
|
|
req.Header = headers
|
|
|
|
}
|
|
|
|
|
|
|
|
if request != nil {
|
|
|
|
req.Header.Set("Content-Type", requestType)
|
|
|
|
}
|
|
|
|
|
2019-08-02 18:21:50 +00:00
|
|
|
if !n.doNotUseToken {
|
|
|
|
if n.token == "" {
|
2019-08-02 23:30:49 +00:00
|
|
|
n.token, err = n.Authenticate()
|
2019-08-02 18:21:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
PrintDebug(fmt.Sprintf("Auth token: %s", n.token))
|
|
|
|
}
|
2019-08-02 02:48:38 +00:00
|
|
|
req.Header.Set("Authorization", "Bearer "+n.token)
|
|
|
|
}
|
|
|
|
|
|
|
|
PrintDebugRequest("Request", req)
|
|
|
|
|
2019-08-02 22:20:18 +00:00
|
|
|
resp, err = n.httpClient.Do(req)
|
2019-08-02 02:48:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err = checkResponseForErrors(resp)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
PrintDebugResponse("Response", resp)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:21:29 +00:00
|
|
|
// Do a JSON http request
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) doJSON(uri, method string, request interface{}, response interface{}) error {
|
2019-08-02 02:48:38 +00:00
|
|
|
var body io.Reader
|
|
|
|
|
|
|
|
if request != nil {
|
|
|
|
reqBodyBytes, err := json.Marshal(request)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
body = bytes.NewReader(reqBodyBytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := n.do(uri, method, body, "application/json", nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if response != nil {
|
|
|
|
d := json.NewDecoder(resp.Body)
|
|
|
|
err := d.Decode(response)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:21:29 +00:00
|
|
|
// Authenticate a user to get an auth token
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) Authenticate() (token string, err error) {
|
2019-08-02 02:48:38 +00:00
|
|
|
PrintVerbose("Getting auth token...")
|
|
|
|
|
|
|
|
reqBody := AuthenticateUserRequest{
|
2019-08-02 18:21:50 +00:00
|
|
|
Username: n.user,
|
|
|
|
Password: n.password,
|
2019-08-02 02:48:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
respBody := AuthenticateUserResponse{}
|
|
|
|
|
2019-08-02 23:30:49 +00:00
|
|
|
previousDoNotUseTokenValue := n.doNotUseToken
|
|
|
|
n.doNotUseToken = true
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON("auth", http.MethodPost, &reqBody, &respBody)
|
2019-08-02 02:48:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-02 23:30:49 +00:00
|
|
|
n.doNotUseToken = previousDoNotUseTokenValue
|
|
|
|
|
2019-08-02 02:48:38 +00:00
|
|
|
token = respBody.Jwt
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:10:59 +00:00
|
|
|
// Get endpoints
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) GetEndpoints() (endpoints []EndpointSubset, err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Getting endpoints...")
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON("endpoints", http.MethodGet, nil, &endpoints)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get stacks, optionally filtered by swarmId and endpointId
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) GetStacks(swarmId string, endpointId uint32) (stacks []Stack, err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Getting stacks...")
|
|
|
|
|
|
|
|
filter := StackListFilter{
|
|
|
|
SwarmId: swarmId,
|
|
|
|
EndpointId: endpointId,
|
|
|
|
}
|
|
|
|
|
|
|
|
filterJsonBytes, _ := json.Marshal(filter)
|
|
|
|
filterJsonString := string(filterJsonBytes)
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("stacks?filters=%s", filterJsonString), http.MethodGet, nil, &stacks)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create swarm stack
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) CreateSwarmStack(stackName string, environmentVariables []StackEnv, stackFileContent string, swarmClusterId string, endpointId string) (err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Deploying stack...")
|
|
|
|
|
|
|
|
reqBody := StackCreateRequest{
|
|
|
|
Name: stackName,
|
|
|
|
Env: environmentVariables,
|
|
|
|
SwarmID: swarmClusterId,
|
|
|
|
StackFileContent: stackFileContent,
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("stacks?type=%v&method=%s&endpointId=%s", 1, "string", endpointId), http.MethodPost, &reqBody, nil)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create compose stack
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) CreateComposeStack(stackName string, environmentVariables []StackEnv, stackFileContent string, endpointId string) (err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Deploying stack...")
|
|
|
|
|
|
|
|
reqBody := StackCreateRequest{
|
|
|
|
Name: stackName,
|
|
|
|
Env: environmentVariables,
|
|
|
|
StackFileContent: stackFileContent,
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("stacks?type=%v&method=%s&endpointId=%s", 2, "string", endpointId), http.MethodPost, &reqBody, nil)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update stack
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) UpdateStack(stack Stack, environmentVariables []StackEnv, stackFileContent string, prune bool, endpointId string) (err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Updating stack...")
|
|
|
|
|
|
|
|
reqBody := StackUpdateRequest{
|
|
|
|
Env: environmentVariables,
|
|
|
|
StackFileContent: stackFileContent,
|
|
|
|
Prune: prune,
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("stacks/%v?endpointId=%s", stack.Id, endpointId), http.MethodPut, &reqBody, nil)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete stack
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) DeleteStack(stackId uint32) (err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Deleting stack...")
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("stacks/%d", stackId), http.MethodDelete, nil, nil)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get stack file content
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) GetStackFileContent(stackId uint32) (content string, err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Getting stack file content...")
|
|
|
|
|
|
|
|
var respBody StackFileInspectResponse
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("stacks/%v/file", stackId), http.MethodGet, nil, &respBody)
|
2019-08-02 17:10:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
content = respBody.StackFileContent
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get endpoint Docker info
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) GetEndpointDockerInfo(endpointId string) (info map[string]interface{}, err error) {
|
2019-08-02 17:10:59 +00:00
|
|
|
PrintVerbose("Getting endpoint Docker info...")
|
|
|
|
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON(fmt.Sprintf("endpoints/%v/docker/info", endpointId), http.MethodGet, nil, &info)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get Portainer status info
|
2019-08-02 23:37:32 +00:00
|
|
|
func (n *PortainerClientImp) GetStatus() (status Status, err error) {
|
2019-08-02 17:12:25 +00:00
|
|
|
err = n.doJSON("status", http.MethodGet, nil, &status)
|
2019-08-02 17:10:59 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:21:29 +00:00
|
|
|
// Create a new client
|
2019-08-02 23:37:32 +00:00
|
|
|
func NewClient(httpClient *http.Client, config ClientConfig) (c PortainerClient, err error) {
|
2019-08-02 20:09:54 +00:00
|
|
|
apiUrl, err := url.Parse(strings.TrimRight(config.Url, "/") + "/api/")
|
2019-08-02 02:48:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-02 23:37:32 +00:00
|
|
|
c = &PortainerClientImp{
|
2019-08-02 22:20:18 +00:00
|
|
|
httpClient: httpClient,
|
|
|
|
url: apiUrl,
|
|
|
|
user: config.User,
|
|
|
|
password: config.Password,
|
|
|
|
token: config.Token,
|
2019-08-02 02:48:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:21:29 +00:00
|
|
|
// Get the cached client or a new one
|
2019-08-02 23:37:32 +00:00
|
|
|
func GetClient() (c PortainerClient, err error) {
|
2019-08-02 20:32:26 +00:00
|
|
|
if cachedClient == nil {
|
2019-08-02 22:20:18 +00:00
|
|
|
cachedClient, err = GetDefaultClient()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return cachedClient, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the default client
|
2019-08-02 23:37:32 +00:00
|
|
|
func GetDefaultClient() (c PortainerClient, err error) {
|
2019-08-02 22:20:18 +00:00
|
|
|
return NewClient(GetDefaultHttpClient(), GetDefaultClientConfig())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the default config for a client
|
|
|
|
func GetDefaultClientConfig() ClientConfig {
|
|
|
|
return ClientConfig{
|
|
|
|
Url: viper.GetString("url"),
|
|
|
|
User: viper.GetString("user"),
|
|
|
|
Password: viper.GetString("password"),
|
|
|
|
Token: viper.GetString("auth-token"),
|
|
|
|
DoNotUseToken: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the default http client for a Portainer client
|
|
|
|
func GetDefaultHttpClient() *http.Client {
|
|
|
|
return &http.Client{
|
|
|
|
Timeout: viper.GetDuration("timeout"),
|
|
|
|
Transport: &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
InsecureSkipVerify: viper.GetBool("insecure"),
|
|
|
|
},
|
|
|
|
},
|
2019-08-02 02:48:38 +00:00
|
|
|
}
|
|
|
|
}
|