mirror of https://github.com/portainer/portainer
177 lines
4.3 KiB
Go
177 lines
4.3 KiB
Go
package endpoint
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
|
|
portainer "github.com/portainer/portainer/api"
|
|
"github.com/portainer/portainer/api/dataservices"
|
|
)
|
|
|
|
// BucketName represents the name of the bucket where this service stores data.
|
|
const BucketName = "endpoints"
|
|
|
|
// Service represents a service for managing environment(endpoint) data.
|
|
type Service struct {
|
|
connection portainer.Connection
|
|
mu sync.RWMutex
|
|
idxEdgeID map[string]portainer.EndpointID
|
|
heartbeats sync.Map
|
|
}
|
|
|
|
func (service *Service) BucketName() string {
|
|
return BucketName
|
|
}
|
|
|
|
// NewService creates a new instance of a service.
|
|
func NewService(connection portainer.Connection) (*Service, error) {
|
|
err := connection.SetServiceName(BucketName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
s := &Service{
|
|
connection: connection,
|
|
idxEdgeID: make(map[string]portainer.EndpointID),
|
|
}
|
|
|
|
es, err := s.endpoints()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, e := range es {
|
|
if len(e.EdgeID) > 0 {
|
|
s.idxEdgeID[e.EdgeID] = e.ID
|
|
}
|
|
|
|
s.heartbeats.Store(e.ID, e.LastCheckInDate)
|
|
}
|
|
|
|
return s, nil
|
|
}
|
|
|
|
func (service *Service) Tx(tx portainer.Transaction) ServiceTx {
|
|
return ServiceTx{
|
|
service: service,
|
|
tx: tx,
|
|
}
|
|
}
|
|
|
|
// Endpoint returns an environment(endpoint) by ID.
|
|
func (service *Service) Endpoint(ID portainer.EndpointID) (*portainer.Endpoint, error) {
|
|
var endpoint *portainer.Endpoint
|
|
var err error
|
|
|
|
err = service.connection.ViewTx(func(tx portainer.Transaction) error {
|
|
endpoint, err = service.Tx(tx).Endpoint(ID)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
endpoint.LastCheckInDate, _ = service.Heartbeat(ID)
|
|
|
|
return endpoint, nil
|
|
}
|
|
|
|
// UpdateEndpoint updates an environment(endpoint).
|
|
func (service *Service) UpdateEndpoint(ID portainer.EndpointID, endpoint *portainer.Endpoint) error {
|
|
return service.connection.UpdateTx(func(tx portainer.Transaction) error {
|
|
return service.Tx(tx).UpdateEndpoint(ID, endpoint)
|
|
})
|
|
}
|
|
|
|
// DeleteEndpoint deletes an environment(endpoint).
|
|
func (service *Service) DeleteEndpoint(ID portainer.EndpointID) error {
|
|
return service.connection.UpdateTx(func(tx portainer.Transaction) error {
|
|
return service.Tx(tx).DeleteEndpoint(ID)
|
|
})
|
|
}
|
|
|
|
func (service *Service) endpoints() ([]portainer.Endpoint, error) {
|
|
var endpoints []portainer.Endpoint
|
|
var err error
|
|
|
|
err = service.connection.ViewTx(func(tx portainer.Transaction) error {
|
|
endpoints, err = service.Tx(tx).Endpoints()
|
|
return err
|
|
})
|
|
|
|
return endpoints, err
|
|
}
|
|
|
|
// Endpoints return an array containing all the environments(endpoints).
|
|
func (service *Service) Endpoints() ([]portainer.Endpoint, error) {
|
|
endpoints, err := service.endpoints()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for i, e := range endpoints {
|
|
t, _ := service.Heartbeat(e.ID)
|
|
endpoints[i].LastCheckInDate = t
|
|
}
|
|
|
|
return endpoints, nil
|
|
}
|
|
|
|
// EndpointIDByEdgeID returns the EndpointID from the given EdgeID using an in-memory index
|
|
func (service *Service) EndpointIDByEdgeID(edgeID string) (portainer.EndpointID, bool) {
|
|
service.mu.RLock()
|
|
endpointID, ok := service.idxEdgeID[edgeID]
|
|
service.mu.RUnlock()
|
|
|
|
return endpointID, ok
|
|
}
|
|
|
|
func (service *Service) Heartbeat(endpointID portainer.EndpointID) (int64, bool) {
|
|
if t, ok := service.heartbeats.Load(endpointID); ok {
|
|
return t.(int64), true
|
|
}
|
|
|
|
return 0, false
|
|
}
|
|
|
|
func (service *Service) UpdateHeartbeat(endpointID portainer.EndpointID) {
|
|
service.heartbeats.Store(endpointID, time.Now().Unix())
|
|
}
|
|
|
|
// CreateEndpoint assign an ID to a new environment(endpoint) and saves it.
|
|
func (service *Service) Create(endpoint *portainer.Endpoint) error {
|
|
return service.connection.UpdateTx(func(tx portainer.Transaction) error {
|
|
return service.Tx(tx).Create(endpoint)
|
|
})
|
|
}
|
|
|
|
func (service *Service) EndpointsByTeamID(teamID portainer.TeamID) ([]portainer.Endpoint, error) {
|
|
var endpoints = make([]portainer.Endpoint, 0)
|
|
|
|
return endpoints, service.connection.GetAll(
|
|
BucketName,
|
|
&portainer.Endpoint{},
|
|
dataservices.FilterFn(&endpoints, func(e portainer.Endpoint) bool {
|
|
for t := range e.TeamAccessPolicies {
|
|
if t == teamID {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}),
|
|
)
|
|
}
|
|
|
|
// GetNextIdentifier returns the next identifier for an environment(endpoint).
|
|
func (service *Service) GetNextIdentifier() int {
|
|
var identifier int
|
|
|
|
service.connection.UpdateTx(func(tx portainer.Transaction) error {
|
|
identifier = service.Tx(tx).GetNextIdentifier()
|
|
|
|
return nil
|
|
})
|
|
|
|
return identifier
|
|
}
|