refactor(grafanabackuper): add restore function
This commit is contained in:
parent
83a15b5a87
commit
1f0d76ba9e
9 changed files with 214 additions and 8 deletions
|
@ -11,7 +11,10 @@ func main() {
|
||||||
cfg := &config.Config{}
|
cfg := &config.Config{}
|
||||||
|
|
||||||
rootCmd := cmd.NewRootCommand(cfg)
|
rootCmd := cmd.NewRootCommand(cfg)
|
||||||
rootCmd.AddCommand(cmd.NewBackupCommand(cfg))
|
rootCmd.AddCommand(
|
||||||
|
cmd.NewBackupCommand(cfg),
|
||||||
|
cmd.NewRestoreCommand(cfg),
|
||||||
|
)
|
||||||
|
|
||||||
if err := rootCmd.Execute(); err != nil {
|
if err := rootCmd.Execute(); err != nil {
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
|
|
|
@ -10,6 +10,7 @@ import (
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
//nolint:dupl // This function may be the same as or similar to other cmd functions.
|
||||||
func NewBackupCommand(c *config.Config) *cobra.Command {
|
func NewBackupCommand(c *config.Config) *cobra.Command {
|
||||||
backupCmd := &cobra.Command{
|
backupCmd := &cobra.Command{
|
||||||
Use: "backup",
|
Use: "backup",
|
||||||
|
|
54
internal/cmd/restore.go
Normal file
54
internal/cmd/restore.go
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"git.ar21.de/yolokube/grafana-backuper/internal/config"
|
||||||
|
"git.ar21.de/yolokube/grafana-backuper/pkg/grafanabackuper"
|
||||||
|
"github.com/rs/zerolog/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
//nolint:dupl // This function may be the same as or similar to other cmd functions.
|
||||||
|
func NewRestoreCommand(c *config.Config) *cobra.Command {
|
||||||
|
restoreCmd := &cobra.Command{
|
||||||
|
Use: "restore",
|
||||||
|
Short: "Restore the dashboards from a git repository to grafana.",
|
||||||
|
Long: "Restore the dashboards from a git repository to grafana.",
|
||||||
|
Run: func(cmd *cobra.Command, _ []string) {
|
||||||
|
if err := restore(cmd.Context(), c); err != nil {
|
||||||
|
log.Fatal().Err(err).Send()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
PreRun: func(cmd *cobra.Command, _ []string) {
|
||||||
|
errs := c.Validate()
|
||||||
|
for _, err := range errs {
|
||||||
|
log.Error().Err(err).Send()
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(errs) > 0 {
|
||||||
|
if err := cmd.Help(); err != nil {
|
||||||
|
log.Error().Err(err).Send()
|
||||||
|
}
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
restoreCmd.PersistentFlags().BoolVar(&c.ForceCommits, "force", false, "Force dashboards / ignore existence check")
|
||||||
|
|
||||||
|
return restoreCmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func restore(ctx context.Context, cfg *config.Config) error {
|
||||||
|
client := grafanabackuper.NewClient(
|
||||||
|
grafanabackuper.WithZerologLogger(cfg.Logger),
|
||||||
|
)
|
||||||
|
|
||||||
|
if err := client.Prepare(ctx, cfg); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return client.Restore.Start(ctx, cfg)
|
||||||
|
}
|
|
@ -4,6 +4,8 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/go-git/go-billy/v5"
|
"github.com/go-git/go-billy/v5"
|
||||||
"github.com/go-git/go-billy/v5/memfs"
|
"github.com/go-git/go-billy/v5/memfs"
|
||||||
|
@ -214,3 +216,36 @@ func (p *Project) Push(ctx context.Context) error {
|
||||||
|
|
||||||
return p.repository.PushContext(ctx, &pushOpts)
|
return p.repository.PushContext(ctx, &pushOpts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Project) ListJSONFiles(directory string) ([]string, error) {
|
||||||
|
files, err := p.fs.ReadDir(directory)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var allFiles []string
|
||||||
|
for _, file := range files {
|
||||||
|
if file.IsDir() {
|
||||||
|
var childFiles []string
|
||||||
|
childFiles, err = p.ListJSONFiles(filepath.Join(directory, file.Name()))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
allFiles = append(allFiles, childFiles...)
|
||||||
|
} else if strings.HasSuffix(file.Name(), ".json") {
|
||||||
|
allFiles = append(allFiles, filepath.Join(directory, file.Name()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return allFiles, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Project) ReadFile(filepath string) ([]byte, error) {
|
||||||
|
file, err := p.fs.Open(filepath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
return io.ReadAll(file)
|
||||||
|
}
|
||||||
|
|
|
@ -79,9 +79,6 @@ type DashboardCreateOpts struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (o DashboardCreateOpts) Validate() error {
|
func (o DashboardCreateOpts) Validate() error {
|
||||||
if o.FolderUID == "" && o.FolderID == 0 {
|
|
||||||
return errors.New("folder ID or UID missing")
|
|
||||||
}
|
|
||||||
if o.Dashboard == nil {
|
if o.Dashboard == nil {
|
||||||
return errors.New("dashboard is nil")
|
return errors.New("dashboard is nil")
|
||||||
}
|
}
|
||||||
|
@ -111,7 +108,7 @@ func (c *DashboardClient) Create(
|
||||||
reqBody.Message = opts.Message
|
reqBody.Message = opts.Message
|
||||||
if opts.FolderUID != "" {
|
if opts.FolderUID != "" {
|
||||||
reqBody.FolderUID = opts.FolderUID
|
reqBody.FolderUID = opts.FolderUID
|
||||||
} else {
|
} else if opts.FolderID > 0 {
|
||||||
reqBody.FolderID = opts.FolderID
|
reqBody.FolderID = opts.FolderID
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -142,3 +139,12 @@ func (c *DashboardClient) Delete(ctx context.Context, uid string) (*Response, er
|
||||||
|
|
||||||
return c.client.do(req, nil)
|
return c.client.do(req, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *DashboardClient) ParseRaw(raw []byte) (*Dashboard, error) {
|
||||||
|
var body schema.Dashboard
|
||||||
|
if err := json.Unmarshal(raw, &body); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return DashboardFromSchema(body), nil
|
||||||
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ package schema
|
||||||
import "time"
|
import "time"
|
||||||
|
|
||||||
type DashboardCreateRequest struct {
|
type DashboardCreateRequest struct {
|
||||||
Dashboard any `json:"dasboard"`
|
Dashboard any `json:"dashboard"`
|
||||||
FolderID uint `json:"folderId,omitempty"`
|
FolderID uint `json:"folderId,omitempty"`
|
||||||
FolderUID string `json:"folderUid"`
|
FolderUID string `json:"folderUid"`
|
||||||
Message string `json:"message,omitempty"`
|
Message string `json:"message,omitempty"`
|
||||||
|
|
|
@ -23,7 +23,8 @@ type Client struct {
|
||||||
logger zerolog.Logger
|
logger zerolog.Logger
|
||||||
signer *git.SignKey
|
signer *git.SignKey
|
||||||
|
|
||||||
Backup BackupClient
|
Backup BackupClient
|
||||||
|
Restore RestoreClient
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewClient(options ...ClientOption) *Client {
|
func NewClient(options ...ClientOption) *Client {
|
||||||
|
@ -34,6 +35,7 @@ func NewClient(options ...ClientOption) *Client {
|
||||||
}
|
}
|
||||||
|
|
||||||
client.Backup = BackupClient{client: client}
|
client.Backup = BackupClient{client: client}
|
||||||
|
client.Restore = RestoreClient{client: client}
|
||||||
|
|
||||||
return client
|
return client
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
package grafanabackuper
|
package grafanabackuper
|
||||||
|
|
||||||
import "fmt"
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
type Message struct {
|
type Message struct {
|
||||||
ID uint
|
ID uint
|
||||||
|
|
103
pkg/grafanabackuper/restore.go
Normal file
103
pkg/grafanabackuper/restore.go
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
package grafanabackuper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"git.ar21.de/yolokube/grafana-backuper/internal/config"
|
||||||
|
"git.ar21.de/yolokube/grafana-backuper/pkg/grafana"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RestoreClient struct {
|
||||||
|
client *Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RestoreClient) Start(ctx context.Context, cfg *config.Config) error {
|
||||||
|
files, err := r.client.git.ListJSONFiles("./")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
r.client.logger.Debug().Int("files", len(files)).Msg("Collected all files")
|
||||||
|
|
||||||
|
for _, filename := range files {
|
||||||
|
r.client.logger.Debug().Str("file", filename).Msg("Reading data")
|
||||||
|
var data []byte
|
||||||
|
data, err = r.client.git.ReadFile(filename)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
r.client.logger.Debug().Msg("Parsing raw dashboard data")
|
||||||
|
var dashboard *grafana.Dashboard
|
||||||
|
dashboard, err = r.client.grafana.Dashboard.ParseRaw(data)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
content, ok := dashboard.Dashboard.(map[string]any)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
uid := fmt.Sprint(content["uid"])
|
||||||
|
title := fmt.Sprint(content["title"])
|
||||||
|
|
||||||
|
r.client.logger.Debug().
|
||||||
|
Str("dashboard-uid", uid).
|
||||||
|
Str("title", title).
|
||||||
|
Msg("Fetching current dashboard version from Grafana")
|
||||||
|
var current *grafana.Dashboard
|
||||||
|
current, _, err = r.client.grafana.Dashboard.Get(ctx, uid)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if current.Version == dashboard.Version && !cfg.ForceCommits {
|
||||||
|
r.client.logger.Info().
|
||||||
|
Any("dashboard-uid", uid).
|
||||||
|
Str("folder", dashboard.FolderTitle).
|
||||||
|
Str("dashboard", title).
|
||||||
|
Msg("Dashboard already up to date")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
r.client.logger.Debug().Str("dashboard-uid", uid).Str("title", title).Msg("Syncing dashboard with Grafana")
|
||||||
|
var createResp *grafana.DashboardCreateResponse
|
||||||
|
createResp, err = r.syncDashboard(ctx, dashboard, cfg.ForceCommits)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
r.client.logger.Info().Any("resp", createResp).Msg("Created / Updated dashboard successfully")
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RestoreClient) syncDashboard(
|
||||||
|
ctx context.Context,
|
||||||
|
d *grafana.Dashboard,
|
||||||
|
force bool,
|
||||||
|
) (*grafana.DashboardCreateResponse, error) {
|
||||||
|
createOpts := grafana.DashboardCreateOpts{
|
||||||
|
Dashboard: d.Dashboard,
|
||||||
|
FolderUID: d.FolderUID,
|
||||||
|
Message: "sync git repository to grafana",
|
||||||
|
Overwrite: force,
|
||||||
|
}
|
||||||
|
|
||||||
|
r.client.logger.Debug().Msg("Validating create options")
|
||||||
|
if err := createOpts.Validate(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
createResp, resp, err := r.client.grafana.Dashboard.Create(ctx, createOpts)
|
||||||
|
if err != nil {
|
||||||
|
body, _ := io.ReadAll(resp.Body)
|
||||||
|
r.client.logger.Debug().Str("resp", string(body)).Msg("Got error during dashboard creation / update")
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return createResp, nil
|
||||||
|
}
|
Loading…
Reference in a new issue