2021-11-29 13:06:50 +00:00
|
|
|
package openamt
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
|
|
|
"encoding/pem"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
portainer "github.com/portainer/portainer/api"
|
|
|
|
)
|
|
|
|
|
|
|
|
type CIRAConfig struct {
|
|
|
|
ConfigName string `json:"configName"`
|
|
|
|
MPSServerAddress string `json:"mpsServerAddress"`
|
|
|
|
ServerAddressFormat int `json:"serverAddressFormat"`
|
|
|
|
CommonName string `json:"commonName"`
|
|
|
|
MPSPort int `json:"mpsPort"`
|
|
|
|
Username string `json:"username"`
|
|
|
|
MPSRootCertificate string `json:"mpsRootCertificate"`
|
|
|
|
RegeneratePassword bool `json:"regeneratePassword"`
|
|
|
|
AuthMethod int `json:"authMethod"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (service *Service) createOrUpdateCIRAConfig(configuration portainer.OpenAMTConfiguration, configName string) (*CIRAConfig, error) {
|
|
|
|
ciraConfig, err := service.getCIRAConfig(configuration, configName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
method := http.MethodPost
|
|
|
|
if ciraConfig != nil {
|
|
|
|
method = http.MethodPatch
|
|
|
|
}
|
|
|
|
|
|
|
|
ciraConfig, err = service.saveCIRAConfig(method, configuration, configName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return ciraConfig, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (service *Service) getCIRAConfig(configuration portainer.OpenAMTConfiguration, configName string) (*CIRAConfig, error) {
|
2021-11-30 23:35:47 +00:00
|
|
|
url := fmt.Sprintf("https://%s/rps/api/v1/admin/ciraconfigs/%s", configuration.MPSServer, configName)
|
2021-11-29 13:06:50 +00:00
|
|
|
|
2022-01-23 19:48:04 +00:00
|
|
|
responseBody, err := service.executeGetRequest(url, configuration.MPSToken)
|
2021-11-29 13:06:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if responseBody == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var result CIRAConfig
|
|
|
|
err = json.Unmarshal(responseBody, &result)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (service *Service) saveCIRAConfig(method string, configuration portainer.OpenAMTConfiguration, configName string) (*CIRAConfig, error) {
|
2021-11-30 23:35:47 +00:00
|
|
|
url := fmt.Sprintf("https://%s/rps/api/v1/admin/ciraconfigs", configuration.MPSServer)
|
2021-11-29 13:06:50 +00:00
|
|
|
|
|
|
|
certificate, err := service.getCIRACertificate(configuration)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-11-30 23:35:47 +00:00
|
|
|
addressFormat, err := addressFormat(configuration.MPSServer)
|
2021-11-29 13:06:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
config := CIRAConfig{
|
|
|
|
ConfigName: configName,
|
2021-11-30 23:35:47 +00:00
|
|
|
MPSServerAddress: configuration.MPSServer,
|
|
|
|
CommonName: configuration.MPSServer,
|
2021-11-29 13:06:50 +00:00
|
|
|
ServerAddressFormat: addressFormat,
|
|
|
|
MPSPort: 4433,
|
|
|
|
Username: "admin",
|
|
|
|
MPSRootCertificate: certificate,
|
|
|
|
RegeneratePassword: false,
|
|
|
|
AuthMethod: 2,
|
|
|
|
}
|
|
|
|
payload, _ := json.Marshal(config)
|
|
|
|
|
2022-01-23 19:48:04 +00:00
|
|
|
responseBody, err := service.executeSaveRequest(method, url, configuration.MPSToken, payload)
|
2021-11-29 13:06:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var result CIRAConfig
|
|
|
|
err = json.Unmarshal(responseBody, &result)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func addressFormat(url string) (int, error) {
|
|
|
|
ip := net.ParseIP(url)
|
|
|
|
if ip == nil {
|
|
|
|
return 201, nil // FQDN
|
|
|
|
}
|
|
|
|
if strings.Contains(url, ".") {
|
|
|
|
return 3, nil // IPV4
|
|
|
|
}
|
|
|
|
if strings.Contains(url, ":") {
|
|
|
|
return 4, nil // IPV6
|
|
|
|
}
|
|
|
|
return 0, fmt.Errorf("could not determine server address format for %s", url)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (service *Service) getCIRACertificate(configuration portainer.OpenAMTConfiguration) (string, error) {
|
2021-11-30 23:35:47 +00:00
|
|
|
loginURL := fmt.Sprintf("https://%s/mps/api/v1/ciracert", configuration.MPSServer)
|
2021-11-29 13:06:50 +00:00
|
|
|
|
|
|
|
req, err := http.NewRequest(http.MethodGet, loginURL, nil)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2022-01-23 19:48:04 +00:00
|
|
|
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", configuration.MPSToken))
|
2021-11-29 13:06:50 +00:00
|
|
|
|
|
|
|
response, err := service.httpsClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2023-03-27 18:14:16 +00:00
|
|
|
defer response.Body.Close()
|
2021-11-29 13:06:50 +00:00
|
|
|
|
|
|
|
if response.StatusCode != http.StatusOK {
|
2022-01-23 19:48:04 +00:00
|
|
|
return "", fmt.Errorf("unexpected status code %s", response.Status)
|
2021-11-29 13:06:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
certificate, err := io.ReadAll(response.Body)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2023-03-27 18:14:16 +00:00
|
|
|
|
2021-11-29 13:06:50 +00:00
|
|
|
block, _ := pem.Decode(certificate)
|
2023-03-27 18:14:16 +00:00
|
|
|
|
2021-11-29 13:06:50 +00:00
|
|
|
return base64.StdEncoding.EncodeToString(block.Bytes), nil
|
|
|
|
}
|