* EE-319: backup endpoint (#193) * feat(backup): * add an orbiter to block writes while backup * add backup handler * add an ability to tar.gz a dir * add aes encryption support * EE-320: restore endpoint (#196) * feat(backup): * add restore handler * re-init system state after restore * feat(backup): Update server to respect readonly lock (#199) * feat(backup): EE-322 Add backup and restore screen (#198) Co-authored-by: Simon Meng <simon.meng@portainer.io> * name archive as portainer-backup_yyyy-mm-dd_hh-mm-ss * backup custom templates and edge jobs * restart http and proxy servers after restore to re-init internal state * feat(backup): EE-322 hide password field if password protect toggle is off * feat(backup): EE-322 add tooltip for password field of restore backup * feat(backup): EE-322 wait for backend restart after restoring * Shutdown background go-routines * changed restore err message when cannot extract * fix: symlinks are ignored from backups * replace single admin check with a restartable monitor (#238) * clean log Co-authored-by: Maxime Bajeux <max.bajeux@gmail.com> Co-authored-by: cong meng <mcpacino@gmail.com> Co-authored-by: Simon Meng <simon.meng@portainer.io>
132 lines
3.8 KiB
Go
132 lines
3.8 KiB
Go
package resourcecontrol
|
|
|
|
import (
|
|
portainer "github.com/portainer/portainer/api"
|
|
"github.com/portainer/portainer/api/bolt/internal"
|
|
|
|
"github.com/boltdb/bolt"
|
|
)
|
|
|
|
const (
|
|
// BucketName represents the name of the bucket where this service stores data.
|
|
BucketName = "resource_control"
|
|
)
|
|
|
|
// Service represents a service for managing endpoint data.
|
|
type Service struct {
|
|
connection *internal.DbConnection
|
|
}
|
|
|
|
// NewService creates a new instance of a service.
|
|
func NewService(connection *internal.DbConnection) (*Service, error) {
|
|
err := internal.CreateBucket(connection, BucketName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Service{
|
|
connection: connection,
|
|
}, nil
|
|
}
|
|
|
|
// ResourceControl returns a ResourceControl object by ID
|
|
func (service *Service) ResourceControl(ID portainer.ResourceControlID) (*portainer.ResourceControl, error) {
|
|
var resourceControl portainer.ResourceControl
|
|
identifier := internal.Itob(int(ID))
|
|
|
|
err := internal.GetObject(service.connection, BucketName, identifier, &resourceControl)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &resourceControl, nil
|
|
}
|
|
|
|
// ResourceControlByResourceIDAndType returns a ResourceControl object by checking if the resourceID is equal
|
|
// to the main ResourceID or in SubResourceIDs. It also performs a check on the resource type. Return nil
|
|
// if no ResourceControl was found.
|
|
func (service *Service) ResourceControlByResourceIDAndType(resourceID string, resourceType portainer.ResourceControlType) (*portainer.ResourceControl, error) {
|
|
var resourceControl *portainer.ResourceControl
|
|
|
|
err := service.connection.View(func(tx *bolt.Tx) error {
|
|
bucket := tx.Bucket([]byte(BucketName))
|
|
cursor := bucket.Cursor()
|
|
|
|
for k, v := cursor.First(); k != nil; k, v = cursor.Next() {
|
|
var rc portainer.ResourceControl
|
|
err := internal.UnmarshalObject(v, &rc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if rc.ResourceID == resourceID && rc.Type == resourceType {
|
|
resourceControl = &rc
|
|
break
|
|
}
|
|
|
|
for _, subResourceID := range rc.SubResourceIDs {
|
|
if subResourceID == resourceID {
|
|
resourceControl = &rc
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
})
|
|
|
|
return resourceControl, err
|
|
}
|
|
|
|
// ResourceControls returns all the ResourceControl objects
|
|
func (service *Service) ResourceControls() ([]portainer.ResourceControl, error) {
|
|
var rcs = make([]portainer.ResourceControl, 0)
|
|
|
|
err := service.connection.View(func(tx *bolt.Tx) error {
|
|
bucket := tx.Bucket([]byte(BucketName))
|
|
|
|
cursor := bucket.Cursor()
|
|
for k, v := cursor.First(); k != nil; k, v = cursor.Next() {
|
|
var resourceControl portainer.ResourceControl
|
|
err := internal.UnmarshalObject(v, &resourceControl)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rcs = append(rcs, resourceControl)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
|
|
return rcs, err
|
|
}
|
|
|
|
// CreateResourceControl creates a new ResourceControl object
|
|
func (service *Service) CreateResourceControl(resourceControl *portainer.ResourceControl) error {
|
|
return service.connection.Update(func(tx *bolt.Tx) error {
|
|
bucket := tx.Bucket([]byte(BucketName))
|
|
|
|
id, _ := bucket.NextSequence()
|
|
resourceControl.ID = portainer.ResourceControlID(id)
|
|
|
|
data, err := internal.MarshalObject(resourceControl)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return bucket.Put(internal.Itob(int(resourceControl.ID)), data)
|
|
})
|
|
}
|
|
|
|
// UpdateResourceControl saves a ResourceControl object.
|
|
func (service *Service) UpdateResourceControl(ID portainer.ResourceControlID, resourceControl *portainer.ResourceControl) error {
|
|
identifier := internal.Itob(int(ID))
|
|
return internal.UpdateObject(service.connection, BucketName, identifier, resourceControl)
|
|
}
|
|
|
|
// DeleteResourceControl deletes a ResourceControl object by ID
|
|
func (service *Service) DeleteResourceControl(ID portainer.ResourceControlID) error {
|
|
identifier := internal.Itob(int(ID))
|
|
return internal.DeleteObject(service.connection, BucketName, identifier)
|
|
}
|