2019-01-08 00:37:16 +00:00
|
|
|
package influxdb
|
2018-07-10 21:56:57 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2018-08-08 19:48:46 +00:00
|
|
|
"strconv"
|
2018-07-31 01:36:22 +00:00
|
|
|
"strings"
|
|
|
|
"unicode"
|
2021-03-30 18:10:02 +00:00
|
|
|
|
|
|
|
"github.com/influxdata/influxdb/v2/kit/platform"
|
|
|
|
"github.com/influxdata/influxdb/v2/kit/platform/errors"
|
2018-07-10 21:56:57 +00:00
|
|
|
)
|
|
|
|
|
2021-08-30 22:27:11 +00:00
|
|
|
// DBRPMappingService provides CRUD to DBRPMappingV2s.
|
|
|
|
type DBRPMappingService interface {
|
2020-04-20 16:55:23 +00:00
|
|
|
// FindBy returns the dbrp mapping for the specified ID.
|
|
|
|
// Requires orgID because every resource will be org-scoped.
|
2021-08-30 22:27:11 +00:00
|
|
|
FindByID(ctx context.Context, orgID, id platform.ID) (*DBRPMapping, error)
|
2020-04-20 16:55:23 +00:00
|
|
|
// FindMany returns a list of dbrp mappings that match filter and the total count of matching dbrp mappings.
|
2021-08-30 22:27:11 +00:00
|
|
|
FindMany(ctx context.Context, dbrp DBRPMappingFilter, opts ...FindOptions) ([]*DBRPMapping, int, error)
|
2020-04-20 16:55:23 +00:00
|
|
|
// Create creates a new dbrp mapping, if a different mapping exists an error is returned.
|
2021-08-30 22:27:11 +00:00
|
|
|
Create(ctx context.Context, dbrp *DBRPMapping) error
|
2020-04-20 16:55:23 +00:00
|
|
|
// Update a new dbrp mapping
|
2021-08-30 22:27:11 +00:00
|
|
|
Update(ctx context.Context, dbrp *DBRPMapping) error
|
2020-04-20 16:55:23 +00:00
|
|
|
// Delete removes a dbrp mapping.
|
|
|
|
// Deleting a mapping that does not exists is not an error.
|
|
|
|
// Requires orgID because every resource will be org-scoped.
|
2021-03-30 18:10:02 +00:00
|
|
|
Delete(ctx context.Context, orgID, id platform.ID) error
|
2020-04-20 16:55:23 +00:00
|
|
|
}
|
|
|
|
|
2021-08-30 22:27:11 +00:00
|
|
|
// DBRPMapping represents a mapping of a database and retention policy to an organization ID and bucket ID.
|
|
|
|
type DBRPMapping struct {
|
2021-03-30 18:10:02 +00:00
|
|
|
ID platform.ID `json:"id"`
|
|
|
|
Database string `json:"database"`
|
|
|
|
RetentionPolicy string `json:"retention_policy"`
|
2020-04-20 16:55:23 +00:00
|
|
|
|
|
|
|
// Default indicates if this mapping is the default for the cluster and database.
|
|
|
|
Default bool `json:"default"`
|
2022-08-03 21:44:04 +00:00
|
|
|
// Virtual indicates if this is a virtual mapping (tied to bucket name) or physical
|
|
|
|
Virtual bool `json:"virtual"`
|
2020-04-20 16:55:23 +00:00
|
|
|
|
2021-03-30 18:10:02 +00:00
|
|
|
OrganizationID platform.ID `json:"orgID"`
|
|
|
|
BucketID platform.ID `json:"bucketID"`
|
2020-04-20 16:55:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Validate reports any validation errors for the mapping.
|
2021-08-30 22:27:11 +00:00
|
|
|
func (m DBRPMapping) Validate() error {
|
2020-04-20 16:55:23 +00:00
|
|
|
if !validName(m.Database) {
|
2021-03-30 18:10:02 +00:00
|
|
|
return &errors.Error{
|
|
|
|
Code: errors.EInvalid,
|
2020-04-20 16:55:23 +00:00
|
|
|
Msg: "database must contain at least one character and only be letters, numbers, '_', '-', and '.'",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !validName(m.RetentionPolicy) {
|
2021-03-30 18:10:02 +00:00
|
|
|
return &errors.Error{
|
|
|
|
Code: errors.EInvalid,
|
2020-04-20 16:55:23 +00:00
|
|
|
Msg: "retentionPolicy must contain at least one character and only be letters, numbers, '_', '-', and '.'",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !m.OrganizationID.Valid() {
|
2021-03-30 18:10:02 +00:00
|
|
|
return &errors.Error{
|
|
|
|
Code: errors.EInvalid,
|
2020-04-20 16:55:23 +00:00
|
|
|
Msg: "organizationID is required",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !m.BucketID.Valid() {
|
2021-03-30 18:10:02 +00:00
|
|
|
return &errors.Error{
|
|
|
|
Code: errors.EInvalid,
|
2020-04-20 16:55:23 +00:00
|
|
|
Msg: "bucketID is required",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Equal checks if the two mappings are identical.
|
2021-08-30 22:27:11 +00:00
|
|
|
func (m *DBRPMapping) Equal(o *DBRPMapping) bool {
|
2020-04-20 16:55:23 +00:00
|
|
|
if m == o {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if m == nil || o == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return m.Database == o.Database &&
|
|
|
|
m.RetentionPolicy == o.RetentionPolicy &&
|
|
|
|
m.Default == o.Default &&
|
|
|
|
m.OrganizationID.Valid() &&
|
|
|
|
o.OrganizationID.Valid() &&
|
|
|
|
m.BucketID.Valid() &&
|
|
|
|
o.BucketID.Valid() &&
|
|
|
|
o.ID.Valid() &&
|
|
|
|
m.ID == o.ID &&
|
|
|
|
m.OrganizationID == o.OrganizationID &&
|
|
|
|
m.BucketID == o.BucketID
|
|
|
|
}
|
|
|
|
|
2021-08-30 22:27:11 +00:00
|
|
|
// DBRPMappingFilter represents a set of filters that restrict the returned results.
|
|
|
|
type DBRPMappingFilter struct {
|
2021-03-30 18:10:02 +00:00
|
|
|
ID *platform.ID
|
|
|
|
OrgID *platform.ID
|
|
|
|
BucketID *platform.ID
|
2020-04-20 16:55:23 +00:00
|
|
|
|
|
|
|
Database *string
|
|
|
|
RetentionPolicy *string
|
|
|
|
Default *bool
|
2022-08-12 17:16:37 +00:00
|
|
|
Virtual *bool
|
2020-04-20 16:55:23 +00:00
|
|
|
}
|
|
|
|
|
2021-08-30 22:27:11 +00:00
|
|
|
func (f DBRPMappingFilter) String() string {
|
2020-04-20 16:55:23 +00:00
|
|
|
var s strings.Builder
|
|
|
|
|
|
|
|
s.WriteString("{ id:")
|
|
|
|
if f.ID != nil {
|
|
|
|
s.WriteString(f.ID.String())
|
|
|
|
} else {
|
|
|
|
s.WriteString("<nil>")
|
|
|
|
}
|
|
|
|
|
|
|
|
s.WriteString(" org_id:")
|
|
|
|
if f.ID != nil {
|
|
|
|
s.WriteString(f.OrgID.String())
|
|
|
|
} else {
|
|
|
|
s.WriteString("<nil>")
|
|
|
|
}
|
|
|
|
|
|
|
|
s.WriteString(" bucket_id:")
|
|
|
|
if f.ID != nil {
|
|
|
|
s.WriteString(f.OrgID.String())
|
|
|
|
} else {
|
|
|
|
s.WriteString("<nil>")
|
|
|
|
}
|
|
|
|
|
|
|
|
s.WriteString(" db:")
|
|
|
|
if f.Database != nil {
|
|
|
|
s.WriteString(*f.Database)
|
|
|
|
} else {
|
|
|
|
s.WriteString("<nil>")
|
|
|
|
}
|
|
|
|
|
|
|
|
s.WriteString(" rp:")
|
|
|
|
if f.RetentionPolicy != nil {
|
|
|
|
s.WriteString(*f.RetentionPolicy)
|
|
|
|
} else {
|
|
|
|
s.WriteString("<nil>")
|
|
|
|
}
|
|
|
|
|
|
|
|
s.WriteString(" default:")
|
|
|
|
if f.Default != nil {
|
|
|
|
s.WriteString(strconv.FormatBool(*f.Default))
|
|
|
|
} else {
|
|
|
|
s.WriteString("<nil>")
|
|
|
|
}
|
|
|
|
s.WriteString("}")
|
|
|
|
return s.String()
|
|
|
|
}
|
|
|
|
|
2018-07-31 01:36:22 +00:00
|
|
|
// validName checks to see if the given name can would be valid for DB/RP name
|
|
|
|
func validName(name string) bool {
|
|
|
|
for _, r := range name {
|
|
|
|
if !unicode.IsPrint(r) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return name != "" &&
|
|
|
|
name != "." &&
|
|
|
|
name != ".." &&
|
|
|
|
!strings.ContainsAny(name, `/\`)
|
|
|
|
}
|