mirror of https://github.com/portainer/portainer
148 lines
4.5 KiB
Go
148 lines
4.5 KiB
Go
package endpointrelation
|
|
|
|
import (
|
|
portainer "github.com/portainer/portainer/api"
|
|
"github.com/portainer/portainer/api/dataservices"
|
|
"github.com/portainer/portainer/api/internal/edge/cache"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
type ServiceTx struct {
|
|
service *Service
|
|
tx portainer.Transaction
|
|
}
|
|
|
|
func (service ServiceTx) BucketName() string {
|
|
return BucketName
|
|
}
|
|
|
|
// EndpointRelations returns an array of all EndpointRelations
|
|
func (service ServiceTx) EndpointRelations() ([]portainer.EndpointRelation, error) {
|
|
var all = make([]portainer.EndpointRelation, 0)
|
|
|
|
return all, service.tx.GetAll(
|
|
BucketName,
|
|
&portainer.EndpointRelation{},
|
|
dataservices.AppendFn(&all),
|
|
)
|
|
}
|
|
|
|
// EndpointRelation returns an Environment(Endpoint) relation object by EndpointID
|
|
func (service ServiceTx) EndpointRelation(endpointID portainer.EndpointID) (*portainer.EndpointRelation, error) {
|
|
var endpointRelation portainer.EndpointRelation
|
|
identifier := service.service.connection.ConvertToKey(int(endpointID))
|
|
|
|
err := service.tx.GetObject(BucketName, identifier, &endpointRelation)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &endpointRelation, nil
|
|
}
|
|
|
|
// CreateEndpointRelation saves endpointRelation
|
|
func (service ServiceTx) Create(endpointRelation *portainer.EndpointRelation) error {
|
|
err := service.tx.CreateObjectWithId(BucketName, int(endpointRelation.EndpointID), endpointRelation)
|
|
cache.Del(endpointRelation.EndpointID)
|
|
|
|
return err
|
|
}
|
|
|
|
// UpdateEndpointRelation updates an Environment(Endpoint) relation object
|
|
func (service ServiceTx) UpdateEndpointRelation(endpointID portainer.EndpointID, endpointRelation *portainer.EndpointRelation) error {
|
|
previousRelationState, _ := service.EndpointRelation(endpointID)
|
|
|
|
identifier := service.service.connection.ConvertToKey(int(endpointID))
|
|
err := service.tx.UpdateObject(BucketName, identifier, endpointRelation)
|
|
cache.Del(endpointID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
updatedRelationState, _ := service.EndpointRelation(endpointID)
|
|
|
|
service.updateEdgeStacksAfterRelationChange(previousRelationState, updatedRelationState)
|
|
|
|
return nil
|
|
}
|
|
|
|
// DeleteEndpointRelation deletes an Environment(Endpoint) relation object
|
|
func (service ServiceTx) DeleteEndpointRelation(endpointID portainer.EndpointID) error {
|
|
deletedRelation, _ := service.EndpointRelation(endpointID)
|
|
|
|
identifier := service.service.connection.ConvertToKey(int(endpointID))
|
|
err := service.tx.DeleteObject(BucketName, identifier)
|
|
cache.Del(endpointID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
service.updateEdgeStacksAfterRelationChange(deletedRelation, nil)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (service ServiceTx) InvalidateEdgeCacheForEdgeStack(edgeStackID portainer.EdgeStackID) {
|
|
rels, err := service.EndpointRelations()
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("cannot retrieve endpoint relations")
|
|
return
|
|
}
|
|
|
|
for _, rel := range rels {
|
|
for id := range rel.EdgeStacks {
|
|
if edgeStackID == id {
|
|
cache.Del(rel.EndpointID)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (service ServiceTx) updateEdgeStacksAfterRelationChange(previousRelationState *portainer.EndpointRelation, updatedRelationState *portainer.EndpointRelation) {
|
|
relations, _ := service.EndpointRelations()
|
|
|
|
stacksToUpdate := map[portainer.EdgeStackID]bool{}
|
|
|
|
if previousRelationState != nil {
|
|
for stackId, enabled := range previousRelationState.EdgeStacks {
|
|
// flag stack for update if stack is not in the updated relation state
|
|
// = stack has been removed for this relation
|
|
// or this relation has been deleted
|
|
if enabled && (updatedRelationState == nil || !updatedRelationState.EdgeStacks[stackId]) {
|
|
stacksToUpdate[stackId] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
if updatedRelationState != nil {
|
|
for stackId, enabled := range updatedRelationState.EdgeStacks {
|
|
// flag stack for update if stack is not in the previous relation state
|
|
// = stack has been added for this relation
|
|
if enabled && (previousRelationState == nil || !previousRelationState.EdgeStacks[stackId]) {
|
|
stacksToUpdate[stackId] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
// for each stack referenced by the updated relation
|
|
// list how many time this stack is referenced in all relations
|
|
// in order to update the stack deployments count
|
|
for refStackId, refStackEnabled := range stacksToUpdate {
|
|
if refStackEnabled {
|
|
numDeployments := 0
|
|
for _, r := range relations {
|
|
for sId, enabled := range r.EdgeStacks {
|
|
if enabled && sId == refStackId {
|
|
numDeployments += 1
|
|
}
|
|
}
|
|
}
|
|
|
|
service.service.updateStackFnTx(service.tx, refStackId, func(edgeStack *portainer.EdgeStack) {
|
|
edgeStack.NumDeployments = numDeployments
|
|
})
|
|
}
|
|
}
|
|
}
|