package edgestacks import ( "net/http" "time" portainer "github.com/portainer/portainer/api" "github.com/portainer/portainer/api/dataservices" "github.com/portainer/portainer/api/internal/edge" "github.com/portainer/portainer/api/internal/set" httperror "github.com/portainer/portainer/pkg/libhttp/error" "github.com/portainer/portainer/pkg/libhttp/request" "github.com/portainer/portainer/pkg/libhttp/response" "github.com/pkg/errors" ) type updateEdgeStackPayload struct { StackFileContent string UpdateVersion bool EdgeGroups []portainer.EdgeGroupID DeploymentType portainer.EdgeStackDeploymentType // Uses the manifest's namespaces instead of the default one UseManifestNamespaces bool } func (payload *updateEdgeStackPayload) Validate(r *http.Request) error { if payload.StackFileContent == "" { return errors.New("invalid stack file content") } if len(payload.EdgeGroups) == 0 { return errors.New("edge Groups are mandatory for an Edge stack") } return nil } // @id EdgeStackUpdate // @summary Update an EdgeStack // @description **Access policy**: administrator // @tags edge_stacks // @security ApiKeyAuth // @security jwt // @accept json // @produce json // @param id path int true "EdgeStack Id" // @param body body updateEdgeStackPayload true "EdgeStack data" // @success 200 {object} portainer.EdgeStack // @failure 500 // @failure 400 // @failure 503 "Edge compute features are disabled" // @router /edge_stacks/{id} [put] func (handler *Handler) edgeStackUpdate(w http.ResponseWriter, r *http.Request) *httperror.HandlerError { stackID, err := request.RetrieveNumericRouteVariableValue(r, "id") if err != nil { return httperror.BadRequest("Invalid stack identifier route variable", err) } var payload updateEdgeStackPayload err = request.DecodeAndValidateJSONPayload(r, &payload) if err != nil { return httperror.BadRequest("Invalid request payload", err) } var stack *portainer.EdgeStack err = handler.DataStore.UpdateTx(func(tx dataservices.DataStoreTx) error { stack, err = handler.updateEdgeStack(tx, portainer.EdgeStackID(stackID), payload) return err }) if err != nil { var httpErr *httperror.HandlerError if errors.As(err, &httpErr) { return httpErr } return httperror.InternalServerError("Unexpected error", err) } return response.JSON(w, stack) } func (handler *Handler) updateEdgeStack(tx dataservices.DataStoreTx, stackID portainer.EdgeStackID, payload updateEdgeStackPayload) (*portainer.EdgeStack, error) { stack, err := tx.EdgeStack().EdgeStack(portainer.EdgeStackID(stackID)) if err != nil { return nil, handler.handlerDBErr(err, "Unable to find a stack with the specified identifier inside the database") } relationConfig, err := edge.FetchEndpointRelationsConfig(tx) if err != nil { return nil, httperror.InternalServerError("Unable to retrieve environments relations config from database", err) } relatedEndpointIds, err := edge.EdgeStackRelatedEndpoints(stack.EdgeGroups, relationConfig.Endpoints, relationConfig.EndpointGroups, relationConfig.EdgeGroups) if err != nil { return nil, httperror.InternalServerError("Unable to retrieve edge stack related environments from database", err) } groupsIds := stack.EdgeGroups if payload.EdgeGroups != nil { newRelated, _, err := handler.handleChangeEdgeGroups(tx, stack.ID, payload.EdgeGroups, relatedEndpointIds, relationConfig) if err != nil { return nil, httperror.InternalServerError("Unable to handle edge groups change", err) } groupsIds = payload.EdgeGroups relatedEndpointIds = newRelated } hasWrongType, err := hasWrongEnvironmentType(tx.Endpoint(), relatedEndpointIds, payload.DeploymentType) if err != nil { return nil, httperror.BadRequest("unable to check for existence of non fitting environments: %w", err) } if hasWrongType { return nil, httperror.BadRequest("edge stack with config do not match the environment type", nil) } stack.NumDeployments = len(relatedEndpointIds) stack.UseManifestNamespaces = payload.UseManifestNamespaces stack.EdgeGroups = groupsIds if payload.UpdateVersion { err := handler.updateStackVersion(stack, payload.DeploymentType, []byte(payload.StackFileContent), "", relatedEndpointIds) if err != nil { return nil, httperror.InternalServerError("Unable to update stack version", err) } } err = tx.EdgeStack().UpdateEdgeStack(stack.ID, stack) if err != nil { return nil, httperror.InternalServerError("Unable to persist the stack changes inside the database", err) } return stack, nil } func (handler *Handler) handleChangeEdgeGroups(tx dataservices.DataStoreTx, edgeStackID portainer.EdgeStackID, newEdgeGroupsIDs []portainer.EdgeGroupID, oldRelatedEnvironmentIDs []portainer.EndpointID, relationConfig *edge.EndpointRelationsConfig) ([]portainer.EndpointID, set.Set[portainer.EndpointID], error) { newRelatedEnvironmentIDs, err := edge.EdgeStackRelatedEndpoints(newEdgeGroupsIDs, relationConfig.Endpoints, relationConfig.EndpointGroups, relationConfig.EdgeGroups) if err != nil { return nil, nil, errors.WithMessage(err, "Unable to retrieve edge stack related environments from database") } oldRelatedSet := set.ToSet(oldRelatedEnvironmentIDs) newRelatedSet := set.ToSet(newRelatedEnvironmentIDs) endpointsToRemove := set.Set[portainer.EndpointID]{} for endpointID := range oldRelatedSet { if !newRelatedSet[endpointID] { endpointsToRemove[endpointID] = true } } for endpointID := range endpointsToRemove { relation, err := tx.EndpointRelation().EndpointRelation(endpointID) if err != nil { return nil, nil, errors.WithMessage(err, "Unable to find environment relation in database") } delete(relation.EdgeStacks, edgeStackID) err = tx.EndpointRelation().UpdateEndpointRelation(endpointID, relation) if err != nil { return nil, nil, errors.WithMessage(err, "Unable to persist environment relation in database") } } endpointsToAdd := set.Set[portainer.EndpointID]{} for endpointID := range newRelatedSet { if !oldRelatedSet[endpointID] { endpointsToAdd[endpointID] = true } } for endpointID := range endpointsToAdd { relation, err := tx.EndpointRelation().EndpointRelation(endpointID) if err != nil { return nil, nil, errors.WithMessage(err, "Unable to find environment relation in database") } relation.EdgeStacks[edgeStackID] = true err = tx.EndpointRelation().UpdateEndpointRelation(endpointID, relation) if err != nil { return nil, nil, errors.WithMessage(err, "Unable to persist environment relation in database") } } return newRelatedEnvironmentIDs, endpointsToAdd, nil } func newStatus(oldStatus map[portainer.EndpointID]portainer.EdgeStackStatus, relatedEnvironmentIds []portainer.EndpointID) map[portainer.EndpointID]portainer.EdgeStackStatus { newStatus := make(map[portainer.EndpointID]portainer.EdgeStackStatus) for _, endpointID := range relatedEnvironmentIds { newEnvStatus := portainer.EdgeStackStatus{} oldEnvStatus, ok := oldStatus[endpointID] if ok { newEnvStatus = oldEnvStatus } newEnvStatus.Status = []portainer.EdgeStackDeploymentStatus{ { Time: time.Now().Unix(), Type: portainer.EdgeStackStatusPending, }, } newStatus[endpointID] = newEnvStatus } return newStatus }