influxdb/pkger/clone_resource.go

1022 lines
27 KiB
Go
Raw Normal View History

package pkger
import (
"context"
"errors"
"fmt"
"regexp"
"sort"
"strings"
"github.com/influxdata/influxdb"
ierrors "github.com/influxdata/influxdb/kit/errors"
"github.com/influxdata/influxdb/notification"
icheck "github.com/influxdata/influxdb/notification/check"
"github.com/influxdata/influxdb/notification/endpoint"
"github.com/influxdata/influxdb/notification/rule"
"github.com/influxdata/influxdb/snowflake"
)
var idGenerator influxdb.IDGenerator = snowflake.NewDefaultIDGenerator()
// ResourceToClone is a resource that will be cloned.
type ResourceToClone struct {
Kind Kind `json:"kind"`
ID influxdb.ID `json:"id"`
Name string `json:"name"`
}
// OK validates a resource clone is viable.
func (r ResourceToClone) OK() error {
if err := r.Kind.OK(); err != nil {
return err
}
if r.ID == influxdb.ID(0) {
return errors.New("must provide an ID")
}
return nil
}
var kindPriorities = map[Kind]int{
KindLabel: 1,
KindBucket: 2,
KindCheck: 3,
KindCheckDeadman: 4,
KindCheckThreshold: 5,
KindNotificationEndpoint: 6,
KindNotificationEndpointHTTP: 7,
KindNotificationEndpointPagerDuty: 8,
KindNotificationEndpointSlack: 9,
KindNotificationRule: 10,
KindTask: 11,
KindVariable: 12,
KindDashboard: 13,
KindTelegraf: 14,
}
type exportKey struct {
orgID influxdb.ID
name string
kind Kind
}
func newExportKey(orgID influxdb.ID, k Kind, name string) exportKey {
return exportKey{
orgID: orgID,
name: name,
kind: k,
}
}
type exportCoordinator struct {
bucketSVC influxdb.BucketService
checkSVC influxdb.CheckService
dashSVC influxdb.DashboardService
labelSVC influxdb.LabelService
endpointSVC influxdb.NotificationEndpointService
ruleSVC influxdb.NotificationRuleStore
taskSVC influxdb.TaskService
teleSVC influxdb.TelegrafConfigStore
varSVC influxdb.VariableService
mObjects map[exportKey]Object
}
func (ex *exportCoordinator) Export(ctx context.Context, resourcesToClone []ResourceToClone, labelNames ...string) error {
cloneAssFn, err := ex.resourceCloneAssociationsGen(ctx, labelNames...)
if err != nil {
return err
}
resourcesToClone = uniqResourcesToClone(resourcesToClone)
// sorting this in priority order guarantees that the dependencies/associations
// for a resource are handled prior to the resource being processed.
// i.e. if a bucket depends on a label, then labels need to be run first
// to guarantee they are available before a bucket is exported.
sort.Slice(resourcesToClone, func(i, j int) bool {
iName, jName := resourcesToClone[i].Name, resourcesToClone[j].Name
iKind, jKind := resourcesToClone[i].Kind, resourcesToClone[j].Kind
if iKind.is(jKind) {
return iName < jName
}
return kindPriorities[iKind] < kindPriorities[jKind]
})
for _, r := range resourcesToClone {
err := ex.resourceCloneToKind(ctx, r, cloneAssFn)
if err != nil {
return internalErr(fmt.Errorf("failed to clone resource: resource_id=%s resource_kind=%s err=%q", r.ID, r.Kind, err))
}
}
return nil
}
func (ex *exportCoordinator) Objects() []Object {
objects := make([]Object, 0, len(ex.mObjects))
for _, obj := range ex.mObjects {
objects = append(objects, obj)
}
sort.Slice(objects, func(i, j int) bool {
iName, jName := objects[i].Name(), objects[j].Name()
iKind, jKind := objects[i].Kind, objects[j].Kind
if iKind.is(jKind) {
return iName < jName
}
return kindPriorities[iKind] < kindPriorities[jKind]
})
return objects
}
type cloneAssociationsFn func(context.Context, ResourceToClone) (associations []Resource, skipResource bool, err error)
func (ex *exportCoordinator) resourceCloneToKind(ctx context.Context, r ResourceToClone, cFn cloneAssociationsFn) (e error) {
defer func() {
if e != nil {
e = ierrors.Wrap(e, "cloning resource")
}
}()
ass, skipResource, err := cFn(ctx, r)
if err != nil {
return err
}
if skipResource {
return nil
}
mapResource := func(orgID influxdb.ID, k Kind, object Object) {
if len(ass) > 0 {
object.Spec[fieldAssociations] = ass
}
key := newExportKey(orgID, k, object.Spec.stringShort(fieldName))
ex.mObjects[key] = object
}
switch {
case r.Kind.is(KindBucket):
bkt, err := ex.bucketSVC.FindBucketByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(bkt.OrgID, KindBucket, bucketToObject(*bkt, r.Name))
case r.Kind.is(KindCheck),
r.Kind.is(KindCheckDeadman),
r.Kind.is(KindCheckThreshold):
ch, err := ex.checkSVC.FindCheckByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(ch.GetOrgID(), KindCheck, checkToObject(ch, r.Name))
case r.Kind.is(KindDashboard):
dash, err := ex.findDashboardByIDFull(ctx, r.ID)
if err != nil {
return err
}
mapResource(dash.OrganizationID, KindDashboard, DashboardToObject(*dash, r.Name))
case r.Kind.is(KindLabel):
l, err := ex.labelSVC.FindLabelByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(l.OrgID, KindLabel, labelToObject(*l, r.Name))
case r.Kind.is(KindNotificationEndpoint),
r.Kind.is(KindNotificationEndpointHTTP),
r.Kind.is(KindNotificationEndpointPagerDuty),
r.Kind.is(KindNotificationEndpointSlack):
e, err := ex.endpointSVC.FindNotificationEndpointByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(e.GetOrgID(), KindNotificationEndpoint, endpointKind(e, r.Name))
case r.Kind.is(KindNotificationRule):
rule, ruleEndpoint, err := ex.getEndpointRule(ctx, r.ID)
if err != nil {
return err
}
endpointKey := newExportKey(ruleEndpoint.GetOrgID(), KindNotificationEndpoint, ruleEndpoint.GetName())
object, ok := ex.mObjects[endpointKey]
if !ok {
mapResource(ruleEndpoint.GetOrgID(), KindNotificationEndpoint, endpointKind(ruleEndpoint, ""))
object = ex.mObjects[endpointKey]
}
endpointObjectName := object.Name()
mapResource(rule.GetOrgID(), KindNotificationRule, ruleToObject(rule, endpointObjectName, r.Name))
case r.Kind.is(KindTask):
t, err := ex.taskSVC.FindTaskByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(t.OrganizationID, KindTask, taskToObject(*t, r.Name))
case r.Kind.is(KindTelegraf):
t, err := ex.teleSVC.FindTelegrafConfigByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(t.OrgID, KindTelegraf, telegrafToObject(*t, r.Name))
case r.Kind.is(KindVariable):
v, err := ex.varSVC.FindVariableByID(ctx, r.ID)
if err != nil {
return err
}
mapResource(v.OrganizationID, KindVariable, VariableToObject(*v, r.Name))
default:
return errors.New("unsupported kind provided: " + string(r.Kind))
}
return nil
}
func (ex *exportCoordinator) resourceCloneAssociationsGen(ctx context.Context, labelNames ...string) (cloneAssociationsFn, error) {
mLabelNames := make(map[string]bool)
for _, labelName := range labelNames {
mLabelNames[labelName] = true
}
mLabelIDs, err := getLabelIDMap(ctx, ex.labelSVC, labelNames)
if err != nil {
return nil, err
}
cloneFn := func(ctx context.Context, r ResourceToClone) ([]Resource, bool, error) {
if r.Kind.is(KindUnknown) {
return nil, true, nil
}
if r.Kind.is(KindLabel) {
// check here verifies the label maps to an id of a valid label name
shouldSkip := len(mLabelIDs) > 0 && !mLabelIDs[r.ID]
return nil, shouldSkip, nil
}
labels, err := ex.labelSVC.FindResourceLabels(ctx, influxdb.LabelMappingFilter{
ResourceID: r.ID,
ResourceType: r.Kind.ResourceType(),
})
if err != nil {
return nil, false, ierrors.Wrap(err, "finding resource labels")
}
if len(mLabelNames) > 0 {
shouldSkip := true
for _, l := range labels {
if _, ok := mLabelNames[l.Name]; ok {
shouldSkip = false
break
}
}
if shouldSkip {
return nil, true, nil
}
}
var associations []Resource
for _, l := range labels {
if len(mLabelNames) > 0 {
if _, ok := mLabelNames[l.Name]; !ok {
continue
}
}
labelObject := labelToObject(*l, "")
k := newExportKey(l.OrgID, KindLabel, l.Name)
existing, ok := ex.mObjects[k]
if ok {
associations = append(associations, Resource{
fieldKind: KindLabel.String(),
fieldName: existing.Name(),
})
continue
}
associations = append(associations, Resource{
fieldKind: KindLabel.String(),
fieldName: labelObject.Name(),
})
ex.mObjects[k] = labelObject
}
return associations, false, nil
}
return cloneFn, nil
}
func (ex *exportCoordinator) getEndpointRule(ctx context.Context, id influxdb.ID) (influxdb.NotificationRule, influxdb.NotificationEndpoint, error) {
rule, err := ex.ruleSVC.FindNotificationRuleByID(ctx, id)
if err != nil {
return nil, nil, err
}
ruleEndpoint, err := ex.endpointSVC.FindNotificationEndpointByID(ctx, rule.GetEndpointID())
if err != nil {
return nil, nil, err
}
return rule, ruleEndpoint, nil
}
func (ex *exportCoordinator) findDashboardByIDFull(ctx context.Context, id influxdb.ID) (*influxdb.Dashboard, error) {
dash, err := ex.dashSVC.FindDashboardByID(ctx, id)
if err != nil {
return nil, err
}
for _, cell := range dash.Cells {
v, err := ex.dashSVC.GetDashboardCellView(ctx, id, cell.ID)
if err != nil {
return nil, err
}
cell.View = v
}
return dash, nil
}
func uniqResourcesToClone(resources []ResourceToClone) []ResourceToClone {
type key struct {
kind Kind
id influxdb.ID
}
m := make(map[key]ResourceToClone)
for i := range resources {
r := resources[i]
rKey := key{kind: r.Kind, id: r.ID}
kr, ok := m[rKey]
switch {
case ok && kr.Name == r.Name:
case ok && kr.Name != "" && r.Name == "":
default:
m[rKey] = r
}
}
out := make([]ResourceToClone, 0, len(resources))
for _, r := range m {
out = append(out, r)
}
return out
}
func bucketToObject(bkt influxdb.Bucket, name string) Object {
if name == "" {
name = bkt.Name
}
o := newObject(KindBucket, name)
assignNonZeroStrings(o.Spec, map[string]string{fieldDescription: bkt.Description})
if bkt.RetentionPeriod != 0 {
o.Spec[fieldBucketRetentionRules] = retentionRules{newRetentionRule(bkt.RetentionPeriod)}
}
return o
}
func checkToObject(ch influxdb.Check, name string) Object {
if name == "" {
name = ch.GetName()
}
o := newObject(KindUnknown, name)
assignNonZeroStrings(o.Spec, map[string]string{
fieldDescription: ch.GetDescription(),
fieldStatus: influxdb.TaskStatusActive,
})
assignBase := func(base icheck.Base) {
o.Spec[fieldQuery] = strings.TrimSpace(base.Query.Text)
o.Spec[fieldCheckStatusMessageTemplate] = base.StatusMessageTemplate
assignNonZeroFluxDurs(o.Spec, map[string]*notification.Duration{
fieldEvery: base.Every,
fieldOffset: base.Offset,
})
var tags []Resource
for _, t := range base.Tags {
if t.Valid() != nil {
continue
}
tags = append(tags, Resource{
fieldKey: t.Key,
fieldValue: t.Value,
})
}
if len(tags) > 0 {
o.Spec[fieldCheckTags] = tags
}
}
switch cT := ch.(type) {
case *icheck.Deadman:
o.Kind = KindCheckDeadman
assignBase(cT.Base)
assignNonZeroFluxDurs(o.Spec, map[string]*notification.Duration{
fieldCheckTimeSince: cT.TimeSince,
fieldCheckStaleTime: cT.StaleTime,
})
o.Spec[fieldLevel] = cT.Level.String()
assignNonZeroBools(o.Spec, map[string]bool{fieldCheckReportZero: cT.ReportZero})
case *icheck.Threshold:
o.Kind = KindCheckThreshold
assignBase(cT.Base)
var thresholds []Resource
for _, th := range cT.Thresholds {
thresholds = append(thresholds, convertThreshold(th))
}
o.Spec[fieldCheckThresholds] = thresholds
}
return o
}
func convertThreshold(th icheck.ThresholdConfig) Resource {
r := Resource{fieldLevel: th.GetLevel().String()}
assignLesser := func(threshType thresholdType, allValues bool, val float64) {
r[fieldType] = string(threshType)
assignNonZeroBools(r, map[string]bool{fieldCheckAllValues: allValues})
r[fieldValue] = val
}
switch realType := th.(type) {
case icheck.Lesser:
assignLesser(thresholdTypeLesser, realType.AllValues, realType.Value)
case *icheck.Lesser:
assignLesser(thresholdTypeLesser, realType.AllValues, realType.Value)
case icheck.Greater:
assignLesser(thresholdTypeGreater, realType.AllValues, realType.Value)
case *icheck.Greater:
assignLesser(thresholdTypeGreater, realType.AllValues, realType.Value)
case icheck.Range:
assignRangeThreshold(r, realType)
case *icheck.Range:
assignRangeThreshold(r, *realType)
}
return r
}
func assignRangeThreshold(r Resource, rangeThreshold icheck.Range) {
thType := thresholdTypeOutsideRange
if rangeThreshold.Within {
thType = thresholdTypeInsideRange
}
r[fieldType] = string(thType)
assignNonZeroBools(r, map[string]bool{fieldCheckAllValues: rangeThreshold.AllValues})
r[fieldMax] = rangeThreshold.Max
r[fieldMin] = rangeThreshold.Min
}
func convertCellView(cell influxdb.Cell) chart {
var name string
if cell.View != nil {
name = cell.View.Name
}
ch := chart{
Name: name,
Height: int(cell.H),
Width: int(cell.W),
XPos: int(cell.X),
YPos: int(cell.Y),
}
setCommon := func(k chartKind, iColors []influxdb.ViewColor, dec influxdb.DecimalPlaces, iQueries []influxdb.DashboardQuery) {
ch.Kind = k
ch.Colors = convertColors(iColors)
ch.DecimalPlaces = int(dec.Digits)
ch.EnforceDecimals = dec.IsEnforced
ch.Queries = convertQueries(iQueries)
}
setNoteFixes := func(note string, noteOnEmpty bool, prefix, suffix string) {
ch.Note = note
ch.NoteOnEmpty = noteOnEmpty
ch.Prefix = prefix
ch.Suffix = suffix
}
setLegend := func(l influxdb.Legend) {
ch.Legend.Orientation = l.Orientation
ch.Legend.Type = l.Type
}
props := cell.View.Properties
switch p := props.(type) {
case influxdb.GaugeViewProperties:
setCommon(chartKindGauge, p.ViewColors, p.DecimalPlaces, p.Queries)
setNoteFixes(p.Note, p.ShowNoteWhenEmpty, p.Prefix, p.Suffix)
ch.TickPrefix = p.TickPrefix
ch.TickSuffix = p.TickSuffix
case influxdb.HeatmapViewProperties:
ch.Kind = chartKindHeatMap
ch.Queries = convertQueries(p.Queries)
ch.Colors = stringsToColors(p.ViewColors)
ch.XCol = p.XColumn
ch.YCol = p.YColumn
ch.Axes = []axis{
{Label: p.XAxisLabel, Prefix: p.XPrefix, Suffix: p.XSuffix, Name: "x", Domain: p.XDomain},
{Label: p.YAxisLabel, Prefix: p.YPrefix, Suffix: p.YSuffix, Name: "y", Domain: p.YDomain},
}
ch.Note = p.Note
ch.NoteOnEmpty = p.ShowNoteWhenEmpty
ch.BinSize = int(p.BinSize)
case influxdb.HistogramViewProperties:
ch.Kind = chartKindHistogram
ch.Queries = convertQueries(p.Queries)
ch.Colors = convertColors(p.ViewColors)
ch.XCol = p.XColumn
ch.Axes = []axis{{Label: p.XAxisLabel, Name: "x", Domain: p.XDomain}}
ch.Note = p.Note
ch.NoteOnEmpty = p.ShowNoteWhenEmpty
ch.BinCount = p.BinCount
ch.Position = p.Position
case influxdb.MarkdownViewProperties:
ch.Kind = chartKindMarkdown
ch.Note = p.Note
case influxdb.LinePlusSingleStatProperties:
setCommon(chartKindSingleStatPlusLine, p.ViewColors, p.DecimalPlaces, p.Queries)
setNoteFixes(p.Note, p.ShowNoteWhenEmpty, p.Prefix, p.Suffix)
setLegend(p.Legend)
ch.Axes = convertAxes(p.Axes)
ch.Shade = p.ShadeBelow
ch.XCol = p.XColumn
ch.YCol = p.YColumn
ch.Position = p.Position
case influxdb.SingleStatViewProperties:
setCommon(chartKindSingleStat, p.ViewColors, p.DecimalPlaces, p.Queries)
setNoteFixes(p.Note, p.ShowNoteWhenEmpty, p.Prefix, p.Suffix)
2020-01-16 20:55:30 +00:00
ch.TickPrefix = p.TickPrefix
ch.TickSuffix = p.TickSuffix
case influxdb.ScatterViewProperties:
ch.Kind = chartKindScatter
ch.Queries = convertQueries(p.Queries)
ch.Colors = stringsToColors(p.ViewColors)
ch.XCol = p.XColumn
ch.YCol = p.YColumn
ch.Axes = []axis{
{Label: p.XAxisLabel, Prefix: p.XPrefix, Suffix: p.XSuffix, Name: "x", Domain: p.XDomain},
{Label: p.YAxisLabel, Prefix: p.YPrefix, Suffix: p.YSuffix, Name: "y", Domain: p.YDomain},
}
ch.Note = p.Note
ch.NoteOnEmpty = p.ShowNoteWhenEmpty
case influxdb.TableViewProperties:
setCommon(chartKindTable, p.ViewColors, p.DecimalPlaces, p.Queries)
setNoteFixes(p.Note, p.ShowNoteWhenEmpty, "", "")
ch.TimeFormat = p.TimeFormat
ch.TableOptions = tableOptions{
VerticalTimeAxis: p.TableOptions.VerticalTimeAxis,
SortByField: p.TableOptions.SortBy.InternalName,
Wrapping: p.TableOptions.Wrapping,
FixFirstColumn: p.TableOptions.FixFirstColumn,
}
for _, fieldOpt := range p.FieldOptions {
ch.FieldOptions = append(ch.FieldOptions, fieldOption{
FieldName: fieldOpt.InternalName,
DisplayName: fieldOpt.DisplayName,
Visible: fieldOpt.Visible,
})
}
case influxdb.XYViewProperties:
setCommon(chartKindXY, p.ViewColors, influxdb.DecimalPlaces{}, p.Queries)
setNoteFixes(p.Note, p.ShowNoteWhenEmpty, "", "")
setLegend(p.Legend)
ch.Axes = convertAxes(p.Axes)
ch.Geom = p.Geom
ch.Shade = p.ShadeBelow
ch.XCol = p.XColumn
ch.YCol = p.YColumn
ch.Position = p.Position
}
return ch
}
func convertChartToResource(ch chart) Resource {
r := Resource{
fieldKind: ch.Kind.title(),
fieldName: ch.Name,
fieldChartQueries: ch.Queries,
fieldChartHeight: ch.Height,
fieldChartWidth: ch.Width,
}
if len(ch.Colors) > 0 {
r[fieldChartColors] = ch.Colors
}
if len(ch.Axes) > 0 {
r[fieldChartAxes] = ch.Axes
}
if ch.EnforceDecimals {
r[fieldChartDecimalPlaces] = ch.DecimalPlaces
}
if ch.Legend.Type != "" {
r[fieldChartLegend] = ch.Legend
}
if zero := new(tableOptions); ch.TableOptions != *zero {
tRes := make(Resource)
assignNonZeroBools(tRes, map[string]bool{
fieldChartTableOptionVerticalTimeAxis: ch.TableOptions.VerticalTimeAxis,
fieldChartTableOptionFixFirstColumn: ch.TableOptions.VerticalTimeAxis,
})
assignNonZeroStrings(tRes, map[string]string{
fieldChartTableOptionSortBy: ch.TableOptions.SortByField,
fieldChartTableOptionWrapping: ch.TableOptions.Wrapping,
})
r[fieldChartTableOptions] = tRes
}
if len(ch.FieldOptions) > 0 {
fieldOpts := make([]Resource, 0, len(ch.FieldOptions))
for _, fo := range ch.FieldOptions {
fRes := make(Resource)
assignNonZeroBools(fRes, map[string]bool{
fieldChartFieldOptionVisible: fo.Visible,
})
assignNonZeroStrings(fRes, map[string]string{
fieldChartFieldOptionDisplayName: fo.DisplayName,
fieldChartFieldOptionFieldName: fo.FieldName,
})
fieldOpts = append(fieldOpts, fRes)
}
r[fieldChartFieldOptions] = fieldOpts
}
assignNonZeroBools(r, map[string]bool{
fieldChartNoteOnEmpty: ch.NoteOnEmpty,
fieldChartShade: ch.Shade,
})
assignNonZeroStrings(r, map[string]string{
2020-01-16 21:53:28 +00:00
fieldChartNote: ch.Note,
fieldPrefix: ch.Prefix,
fieldSuffix: ch.Suffix,
fieldChartGeom: ch.Geom,
fieldChartXCol: ch.XCol,
fieldChartYCol: ch.YCol,
fieldChartPosition: ch.Position,
fieldChartTickPrefix: ch.TickPrefix,
fieldChartTickSuffix: ch.TickSuffix,
fieldChartTimeFormat: ch.TimeFormat,
})
assignNonZeroInts(r, map[string]int{
fieldChartXPos: ch.XPos,
fieldChartYPos: ch.YPos,
fieldChartBinCount: ch.BinCount,
fieldChartBinSize: ch.BinSize,
})
return r
}
func convertAxes(iAxes map[string]influxdb.Axis) axes {
out := make(axes, 0, len(iAxes))
for name, a := range iAxes {
out = append(out, axis{
Base: a.Base,
Label: a.Label,
Name: name,
Prefix: a.Prefix,
Scale: a.Scale,
Suffix: a.Suffix,
})
}
return out
}
func convertColors(iColors []influxdb.ViewColor) colors {
out := make(colors, 0, len(iColors))
for _, ic := range iColors {
out = append(out, &color{
Name: ic.Name,
Type: ic.Type,
Hex: ic.Hex,
Value: flt64Ptr(ic.Value),
})
}
return out
}
func convertQueries(iQueries []influxdb.DashboardQuery) queries {
out := make(queries, 0, len(iQueries))
for _, iq := range iQueries {
out = append(out, query{Query: strings.TrimSpace(iq.Text)})
}
return out
}
// DashboardToObject converts an influxdb.Dashboard to a pkger.Resource.
func DashboardToObject(dash influxdb.Dashboard, name string) Object {
if name == "" {
name = dash.Name
}
sort.Slice(dash.Cells, func(i, j int) bool {
ic, jc := dash.Cells[i], dash.Cells[j]
if ic.X == jc.X {
return ic.Y < jc.Y
}
return ic.X < jc.X
})
charts := make([]Resource, 0, len(dash.Cells))
for _, cell := range dash.Cells {
if cell.ID == influxdb.ID(0) {
continue
}
ch := convertCellView(*cell)
if !ch.Kind.ok() {
continue
}
charts = append(charts, convertChartToResource(ch))
}
return Object{
APIVersion: APIVersion,
Kind: KindDashboard,
Metadata: convertToMetadataResource(name),
Spec: Resource{
fieldDescription: dash.Description,
fieldDashCharts: charts,
},
}
}
func labelToObject(l influxdb.Label, name string) Object {
if name == "" {
name = l.Name
}
o := newObject(KindLabel, name)
assignNonZeroStrings(o.Spec, map[string]string{
fieldDescription: l.Properties["description"],
fieldLabelColor: l.Properties["color"],
})
return o
}
func endpointKind(e influxdb.NotificationEndpoint, name string) Object {
if name == "" {
name = e.GetName()
}
k := Object{
APIVersion: APIVersion,
Metadata: convertToMetadataResource(name),
Spec: make(Resource),
}
assignNonZeroStrings(k.Spec, map[string]string{
fieldName: name,
fieldDescription: e.GetDescription(),
fieldStatus: string(e.GetStatus()),
})
switch actual := e.(type) {
case *endpoint.HTTP:
k.Kind = KindNotificationEndpointHTTP
k.Spec[fieldNotificationEndpointHTTPMethod] = actual.Method
k.Spec[fieldNotificationEndpointURL] = actual.URL
k.Spec[fieldType] = actual.AuthMethod
assignNonZeroSecrets(k.Spec, map[string]influxdb.SecretField{
fieldNotificationEndpointPassword: actual.Password,
fieldNotificationEndpointToken: actual.Token,
fieldNotificationEndpointUsername: actual.Username,
})
case *endpoint.PagerDuty:
k.Kind = KindNotificationEndpointPagerDuty
k.Spec[fieldNotificationEndpointURL] = actual.ClientURL
assignNonZeroSecrets(k.Spec, map[string]influxdb.SecretField{
fieldNotificationEndpointRoutingKey: actual.RoutingKey,
})
case *endpoint.Slack:
k.Kind = KindNotificationEndpointSlack
k.Spec[fieldNotificationEndpointURL] = actual.URL
assignNonZeroSecrets(k.Spec, map[string]influxdb.SecretField{
fieldNotificationEndpointToken: actual.Token,
})
}
return k
}
func ruleToObject(iRule influxdb.NotificationRule, endpointName, name string) Object {
if name == "" {
name = iRule.GetName()
}
k := Object{
APIVersion: APIVersion,
Kind: KindNotificationRule,
Metadata: convertToMetadataResource(name),
Spec: Resource{
fieldNotificationRuleEndpointName: endpointName,
},
}
assignNonZeroStrings(k.Spec, map[string]string{
fieldDescription: iRule.GetDescription(),
})
assignBase := func(base rule.Base) {
assignNonZeroFluxDurs(k.Spec, map[string]*notification.Duration{
fieldEvery: base.Every,
fieldOffset: base.Offset,
})
var tagRes []Resource
for _, tRule := range base.TagRules {
tagRes = append(tagRes, Resource{
fieldKey: tRule.Key,
fieldValue: tRule.Value,
fieldOperator: tRule.Operator.String(),
})
}
if len(tagRes) > 0 {
k.Spec[fieldNotificationRuleTagRules] = tagRes
}
var statusRuleRes []Resource
for _, sRule := range base.StatusRules {
sRes := Resource{
fieldNotificationRuleCurrentLevel: sRule.CurrentLevel.String(),
}
if sRule.PreviousLevel != nil {
sRes[fieldNotificationRulePreviousLevel] = sRule.PreviousLevel.String()
}
statusRuleRes = append(statusRuleRes, sRes)
}
if len(statusRuleRes) > 0 {
k.Spec[fieldNotificationRuleStatusRules] = statusRuleRes
}
}
switch t := iRule.(type) {
case *rule.HTTP:
assignBase(t.Base)
case *rule.PagerDuty:
assignBase(t.Base)
k.Spec[fieldNotificationRuleMessageTemplate] = t.MessageTemplate
case *rule.Slack:
assignBase(t.Base)
k.Spec[fieldNotificationRuleMessageTemplate] = t.MessageTemplate
assignNonZeroStrings(k.Spec, map[string]string{fieldNotificationRuleChannel: t.Channel})
}
return k
}
// regex used to rip out the hard coded task option stuffs
var taskFluxRegex = regexp.MustCompile(`option task = {(.|\n)*?}`)
func taskToObject(t influxdb.Task, name string) Object {
if name == "" {
name = t.Name
}
query := strings.TrimSpace(taskFluxRegex.ReplaceAllString(t.Flux, ""))
k := Object{
APIVersion: APIVersion,
Kind: KindTask,
Metadata: convertToMetadataResource(name),
Spec: Resource{
fieldQuery: strings.TrimSpace(query),
},
}
assignNonZeroStrings(k.Spec, map[string]string{
fieldTaskCron: t.Cron,
fieldDescription: t.Description,
fieldEvery: t.Every,
fieldOffset: durToStr(t.Offset),
})
return k
}
func telegrafToObject(t influxdb.TelegrafConfig, name string) Object {
if name == "" {
name = t.Name
}
o := newObject(KindTelegraf, name)
assignNonZeroStrings(o.Spec, map[string]string{
fieldTelegrafConfig: t.Config,
fieldDescription: t.Description,
})
return o
}
// VariableToObject converts an influxdb.Variable to a pkger.Object.
func VariableToObject(v influxdb.Variable, name string) Object {
if name == "" {
name = v.Name
}
o := newObject(KindVariable, name)
assignNonZeroStrings(o.Spec, map[string]string{fieldDescription: v.Description})
args := v.Arguments
if args == nil {
return o
}
o.Spec[fieldType] = args.Type
switch args.Type {
case fieldArgTypeConstant:
vals, ok := args.Values.(influxdb.VariableConstantValues)
if ok {
o.Spec[fieldValues] = []string(vals)
}
case fieldArgTypeMap:
vals, ok := args.Values.(influxdb.VariableMapValues)
if ok {
o.Spec[fieldValues] = map[string]string(vals)
}
case fieldArgTypeQuery:
vals, ok := args.Values.(influxdb.VariableQueryValues)
if ok {
o.Spec[fieldLanguage] = vals.Language
o.Spec[fieldQuery] = strings.TrimSpace(vals.Query)
}
}
return o
}
func convertToMetadataResource(name string) Resource {
return Resource{
fieldName: name,
}
}
func newObject(kind Kind, name string) Object {
return Object{
APIVersion: APIVersion,
Kind: kind,
Metadata: Resource{
// this timestamp is added to make the resource unique. Should also indicate
// to the end user that this is machine readable and the spec.name field is
// the one they want to edit when a name change is desired.
fieldName: idGenerator.ID().String(),
},
Spec: Resource{
fieldName: name,
},
}
}
func assignNonZeroFluxDurs(r Resource, m map[string]*notification.Duration) {
for field, dur := range m {
if dur == nil {
continue
}
if dur.TimeDuration() == 0 {
continue
}
r[field] = dur.TimeDuration().String()
}
}
func assignNonZeroBools(r Resource, m map[string]bool) {
for k, v := range m {
if v {
r[k] = v
}
}
}
func assignNonZeroInts(r Resource, m map[string]int) {
for k, v := range m {
if v != 0 {
r[k] = v
}
}
}
func assignNonZeroStrings(r Resource, m map[string]string) {
for k, v := range m {
if v != "" {
r[k] = v
}
}
}
func assignNonZeroSecrets(r Resource, m map[string]influxdb.SecretField) {
for field, secret := range m {
if secret.Key == "" {
continue
}
r[field] = Resource{
fieldReferencesSecret: Resource{
fieldKey: secret.Key,
},
}
}
}
func stringsToColors(clrs []string) colors {
newColors := make(colors, 0)
for _, x := range clrs {
newColors = append(newColors, &color{Hex: x})
}
return newColors
}