2018-06-11 13:13:19 +00:00
|
|
|
package endpoints
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
2019-10-07 02:42:01 +00:00
|
|
|
"reflect"
|
2018-06-11 13:13:19 +00:00
|
|
|
"strconv"
|
|
|
|
|
2018-09-10 10:01:38 +00:00
|
|
|
httperror "github.com/portainer/libhttp/error"
|
|
|
|
"github.com/portainer/libhttp/request"
|
|
|
|
"github.com/portainer/libhttp/response"
|
2019-03-21 01:20:14 +00:00
|
|
|
"github.com/portainer/portainer/api"
|
|
|
|
"github.com/portainer/portainer/api/http/client"
|
2018-06-11 13:13:19 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type endpointUpdatePayload struct {
|
2018-10-28 09:27:06 +00:00
|
|
|
Name *string
|
|
|
|
URL *string
|
|
|
|
PublicURL *string
|
|
|
|
GroupID *int
|
|
|
|
TLS *bool
|
|
|
|
TLSSkipVerify *bool
|
|
|
|
TLSSkipClientVerify *bool
|
|
|
|
Status *int
|
|
|
|
AzureApplicationID *string
|
|
|
|
AzureTenantID *string
|
|
|
|
AzureAuthenticationKey *string
|
2018-06-15 07:18:25 +00:00
|
|
|
Tags []string
|
2019-05-24 06:04:58 +00:00
|
|
|
UserAccessPolicies portainer.UserAccessPolicies
|
|
|
|
TeamAccessPolicies portainer.TeamAccessPolicies
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (payload *endpointUpdatePayload) Validate(r *http.Request) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// PUT request on /api/endpoints/:id
|
|
|
|
func (handler *Handler) endpointUpdate(w http.ResponseWriter, r *http.Request) *httperror.HandlerError {
|
|
|
|
if !handler.authorizeEndpointManagement {
|
|
|
|
return &httperror.HandlerError{http.StatusServiceUnavailable, "Endpoint management is disabled", ErrEndpointManagementDisabled}
|
|
|
|
}
|
|
|
|
|
|
|
|
endpointID, err := request.RetrieveNumericRouteVariableValue(r, "id")
|
|
|
|
if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusBadRequest, "Invalid endpoint identifier route variable", err}
|
|
|
|
}
|
|
|
|
|
|
|
|
var payload endpointUpdatePayload
|
|
|
|
err = request.DecodeAndValidateJSONPayload(r, &payload)
|
|
|
|
if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusBadRequest, "Invalid request payload", err}
|
|
|
|
}
|
|
|
|
|
|
|
|
endpoint, err := handler.EndpointService.Endpoint(portainer.EndpointID(endpointID))
|
2018-06-19 11:15:10 +00:00
|
|
|
if err == portainer.ErrObjectNotFound {
|
2018-06-11 13:13:19 +00:00
|
|
|
return &httperror.HandlerError{http.StatusNotFound, "Unable to find an endpoint with the specified identifier inside the database", err}
|
|
|
|
} else if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to find an endpoint with the specified identifier inside the database", err}
|
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.Name != nil {
|
|
|
|
endpoint.Name = *payload.Name
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.URL != nil {
|
|
|
|
endpoint.URL = *payload.URL
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.PublicURL != nil {
|
|
|
|
endpoint.PublicURL = *payload.PublicURL
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.GroupID != nil {
|
|
|
|
endpoint.GroupID = portainer.EndpointGroupID(*payload.GroupID)
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
2018-06-15 07:18:25 +00:00
|
|
|
if payload.Tags != nil {
|
|
|
|
endpoint.Tags = payload.Tags
|
|
|
|
}
|
|
|
|
|
2019-09-09 22:58:26 +00:00
|
|
|
updateAuthorizations := false
|
2019-10-07 02:42:01 +00:00
|
|
|
if payload.UserAccessPolicies != nil && !reflect.DeepEqual(payload.UserAccessPolicies, endpoint.UserAccessPolicies) {
|
2019-05-24 06:04:58 +00:00
|
|
|
endpoint.UserAccessPolicies = payload.UserAccessPolicies
|
2019-09-09 22:58:26 +00:00
|
|
|
updateAuthorizations = true
|
2019-05-24 06:04:58 +00:00
|
|
|
}
|
|
|
|
|
2019-10-07 02:42:01 +00:00
|
|
|
if payload.TeamAccessPolicies != nil && !reflect.DeepEqual(payload.TeamAccessPolicies, endpoint.TeamAccessPolicies) {
|
2019-05-24 06:04:58 +00:00
|
|
|
endpoint.TeamAccessPolicies = payload.TeamAccessPolicies
|
2019-09-09 22:58:26 +00:00
|
|
|
updateAuthorizations = true
|
2019-05-24 06:04:58 +00:00
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.Status != nil {
|
|
|
|
switch *payload.Status {
|
|
|
|
case 1:
|
|
|
|
endpoint.Status = portainer.EndpointStatusUp
|
|
|
|
break
|
|
|
|
case 2:
|
|
|
|
endpoint.Status = portainer.EndpointStatusDown
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-11 13:13:19 +00:00
|
|
|
if endpoint.Type == portainer.AzureEnvironment {
|
|
|
|
credentials := endpoint.AzureCredentials
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.AzureApplicationID != nil {
|
|
|
|
credentials.ApplicationID = *payload.AzureApplicationID
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.AzureTenantID != nil {
|
|
|
|
credentials.TenantID = *payload.AzureTenantID
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.AzureAuthenticationKey != nil {
|
|
|
|
credentials.AuthenticationKey = *payload.AzureAuthenticationKey
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
httpClient := client.NewHTTPClient()
|
|
|
|
_, authErr := httpClient.ExecuteAzureAuthenticationRequest(&credentials)
|
|
|
|
if authErr != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to authenticate against Azure", authErr}
|
|
|
|
}
|
|
|
|
endpoint.AzureCredentials = credentials
|
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.TLS != nil {
|
|
|
|
folder := strconv.Itoa(endpointID)
|
|
|
|
|
|
|
|
if *payload.TLS {
|
|
|
|
endpoint.TLSConfig.TLS = true
|
|
|
|
if payload.TLSSkipVerify != nil {
|
|
|
|
endpoint.TLSConfig.TLSSkipVerify = *payload.TLSSkipVerify
|
|
|
|
|
|
|
|
if !*payload.TLSSkipVerify {
|
|
|
|
caCertPath, _ := handler.FileService.GetPathForTLSFile(folder, portainer.TLSFileCA)
|
|
|
|
endpoint.TLSConfig.TLSCACertPath = caCertPath
|
|
|
|
} else {
|
|
|
|
endpoint.TLSConfig.TLSCACertPath = ""
|
|
|
|
handler.FileService.DeleteTLSFile(folder, portainer.TLSFileCA)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if payload.TLSSkipClientVerify != nil {
|
|
|
|
if !*payload.TLSSkipClientVerify {
|
|
|
|
certPath, _ := handler.FileService.GetPathForTLSFile(folder, portainer.TLSFileCert)
|
|
|
|
endpoint.TLSConfig.TLSCertPath = certPath
|
|
|
|
keyPath, _ := handler.FileService.GetPathForTLSFile(folder, portainer.TLSFileKey)
|
|
|
|
endpoint.TLSConfig.TLSKeyPath = keyPath
|
|
|
|
} else {
|
|
|
|
endpoint.TLSConfig.TLSCertPath = ""
|
|
|
|
handler.FileService.DeleteTLSFile(folder, portainer.TLSFileCert)
|
|
|
|
endpoint.TLSConfig.TLSKeyPath = ""
|
|
|
|
handler.FileService.DeleteTLSFile(folder, portainer.TLSFileKey)
|
|
|
|
}
|
|
|
|
}
|
2018-06-11 13:13:19 +00:00
|
|
|
|
|
|
|
} else {
|
2018-10-28 09:27:06 +00:00
|
|
|
endpoint.TLSConfig.TLS = false
|
|
|
|
endpoint.TLSConfig.TLSSkipVerify = false
|
|
|
|
endpoint.TLSConfig.TLSCACertPath = ""
|
2018-06-11 13:13:19 +00:00
|
|
|
endpoint.TLSConfig.TLSCertPath = ""
|
|
|
|
endpoint.TLSConfig.TLSKeyPath = ""
|
2018-10-28 09:27:06 +00:00
|
|
|
err = handler.FileService.DeleteTLSFiles(folder)
|
|
|
|
if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to remove TLS files from disk", err}
|
|
|
|
}
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-28 09:27:06 +00:00
|
|
|
if payload.URL != nil || payload.TLS != nil || endpoint.Type == portainer.AzureEnvironment {
|
2019-11-12 23:41:42 +00:00
|
|
|
_, err = handler.ProxyManager.CreateAndRegisterEndpointProxy(endpoint)
|
2018-10-28 09:27:06 +00:00
|
|
|
if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to register HTTP proxy for the endpoint", err}
|
|
|
|
}
|
2018-06-11 13:13:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err = handler.EndpointService.UpdateEndpoint(endpoint.ID, endpoint)
|
|
|
|
if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to persist endpoint changes inside the database", err}
|
|
|
|
}
|
|
|
|
|
2019-09-09 22:58:26 +00:00
|
|
|
if updateAuthorizations {
|
2019-10-07 02:42:01 +00:00
|
|
|
err = handler.AuthorizationService.UpdateUsersAuthorizations()
|
2019-09-09 22:58:26 +00:00
|
|
|
if err != nil {
|
|
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to update user authorizations", err}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-11 13:13:19 +00:00
|
|
|
return response.JSON(w, endpoint)
|
|
|
|
}
|