2018-05-06 07:15:57 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
2020-06-09 02:43:32 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2018-07-03 18:31:02 +00:00
|
|
|
"io/ioutil"
|
2019-11-20 05:16:40 +00:00
|
|
|
"log"
|
2018-05-06 07:15:57 +00:00
|
|
|
"net/http"
|
2020-06-09 02:43:32 +00:00
|
|
|
"net/url"
|
2018-05-06 07:15:57 +00:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2019-03-21 01:20:14 +00:00
|
|
|
"github.com/portainer/portainer/api"
|
2018-05-06 07:15:57 +00:00
|
|
|
)
|
|
|
|
|
2018-08-21 18:40:42 +00:00
|
|
|
const (
|
|
|
|
errInvalidResponseStatus = portainer.Error("Invalid response status (expecting 200)")
|
2018-09-24 00:09:12 +00:00
|
|
|
defaultHTTPTimeout = 5
|
2018-08-21 18:40:42 +00:00
|
|
|
)
|
|
|
|
|
2020-06-09 02:43:32 +00:00
|
|
|
// HTTPClient represents a client to send HTTP requests.
|
|
|
|
type HTTPClient struct {
|
|
|
|
*http.Client
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewHTTPClient is used to build a new HTTPClient.
|
|
|
|
func NewHTTPClient() *HTTPClient {
|
|
|
|
return &HTTPClient{
|
|
|
|
&http.Client{
|
|
|
|
Timeout: time.Second * time.Duration(defaultHTTPTimeout),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AzureAuthenticationResponse represents an Azure API authentication response.
|
|
|
|
type AzureAuthenticationResponse struct {
|
|
|
|
AccessToken string `json:"access_token"`
|
|
|
|
ExpiresOn string `json:"expires_on"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExecuteAzureAuthenticationRequest is used to execute an authentication request
|
|
|
|
// against the Azure API. It re-uses the same http.Client.
|
|
|
|
func (client *HTTPClient) ExecuteAzureAuthenticationRequest(credentials *portainer.AzureCredentials) (*AzureAuthenticationResponse, error) {
|
|
|
|
loginURL := fmt.Sprintf("https://login.microsoftonline.com/%s/oauth2/token", credentials.TenantID)
|
|
|
|
params := url.Values{
|
|
|
|
"grant_type": {"client_credentials"},
|
|
|
|
"client_id": {credentials.ApplicationID},
|
|
|
|
"client_secret": {credentials.AuthenticationKey},
|
|
|
|
"resource": {"https://management.azure.com/"},
|
|
|
|
}
|
|
|
|
|
|
|
|
response, err := client.PostForm(loginURL, params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if response.StatusCode != http.StatusOK {
|
|
|
|
return nil, portainer.ErrAzureInvalidCredentials
|
|
|
|
}
|
|
|
|
|
|
|
|
var token AzureAuthenticationResponse
|
|
|
|
err = json.NewDecoder(response.Body).Decode(&token)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &token, nil
|
|
|
|
}
|
|
|
|
|
2018-07-03 18:31:02 +00:00
|
|
|
// Get executes a simple HTTP GET to the specified URL and returns
|
2018-09-24 00:09:12 +00:00
|
|
|
// the content of the response body. Timeout can be specified via the timeout parameter,
|
|
|
|
// will default to defaultHTTPTimeout if set to 0.
|
|
|
|
func Get(url string, timeout int) ([]byte, error) {
|
|
|
|
|
|
|
|
if timeout == 0 {
|
|
|
|
timeout = defaultHTTPTimeout
|
|
|
|
}
|
|
|
|
|
2018-07-03 18:31:02 +00:00
|
|
|
client := &http.Client{
|
2018-09-24 00:09:12 +00:00
|
|
|
Timeout: time.Second * time.Duration(timeout),
|
2018-07-03 18:31:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
response, err := client.Get(url)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer response.Body.Close()
|
|
|
|
|
2018-08-21 18:40:42 +00:00
|
|
|
if response.StatusCode != http.StatusOK {
|
2019-11-20 05:16:40 +00:00
|
|
|
log.Printf("[ERROR] [http,client] [message: unexpected status code] [status_code: %d]", response.StatusCode)
|
2018-08-21 18:40:42 +00:00
|
|
|
return nil, errInvalidResponseStatus
|
|
|
|
}
|
|
|
|
|
2018-07-03 18:31:02 +00:00
|
|
|
body, err := ioutil.ReadAll(response.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return body, nil
|
|
|
|
}
|
|
|
|
|
2018-05-06 07:15:57 +00:00
|
|
|
// ExecutePingOperation will send a SystemPing operation HTTP request to a Docker environment
|
|
|
|
// using the specified host and optional TLS configuration.
|
2018-05-28 14:40:33 +00:00
|
|
|
// It uses a new Http.Client for each operation.
|
2018-05-06 07:15:57 +00:00
|
|
|
func ExecutePingOperation(host string, tlsConfig *tls.Config) (bool, error) {
|
|
|
|
transport := &http.Transport{}
|
|
|
|
|
|
|
|
scheme := "http"
|
|
|
|
if tlsConfig != nil {
|
|
|
|
transport.TLSClientConfig = tlsConfig
|
|
|
|
scheme = "https"
|
|
|
|
}
|
|
|
|
|
|
|
|
client := &http.Client{
|
|
|
|
Timeout: time.Second * 3,
|
|
|
|
Transport: transport,
|
|
|
|
}
|
|
|
|
|
|
|
|
target := strings.Replace(host, "tcp://", scheme+"://", 1)
|
|
|
|
return pingOperation(client, target)
|
|
|
|
}
|
|
|
|
|
|
|
|
func pingOperation(client *http.Client, target string) (bool, error) {
|
|
|
|
pingOperationURL := target + "/_ping"
|
|
|
|
|
|
|
|
response, err := client.Get(pingOperationURL)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
agentOnDockerEnvironment := false
|
|
|
|
if response.Header.Get(portainer.PortainerAgentHeader) != "" {
|
|
|
|
agentOnDockerEnvironment = true
|
|
|
|
}
|
|
|
|
|
|
|
|
return agentOnDockerEnvironment, nil
|
|
|
|
}
|