-
Notifications
You must be signed in to change notification settings - Fork 2.2k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add a new
silence import
command to amtool
This command read silences data from a query JSON output and import to alertmanager. It allows `amtool` to be used as a backup/restore tool for silences, i.e. #1000 Backup / export: ``` amtool silence -o json > silences.json ``` Restore / import: ``` amtool silence import silences.json ```
- Loading branch information
Showing
3 changed files
with
145 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,127 @@ | ||
package cli | ||
|
||
import ( | ||
"encoding/json" | ||
"fmt" | ||
"os" | ||
|
||
"github.com/pkg/errors" | ||
"github.com/prometheus/alertmanager/types" | ||
"github.com/prometheus/common/log" | ||
"github.com/spf13/cobra" | ||
flag "github.com/spf13/pflag" | ||
) | ||
|
||
var importFlags *flag.FlagSet | ||
var importCmd = &cobra.Command{ | ||
Use: "import [JSON file]", | ||
Short: "Import silences", | ||
Long: `Import alertmanager silences from JSON file or stdin | ||
This command can be used to bulk import silences from a JSON file | ||
created by query command. For example: | ||
amtool silence query -o json foo > foo.json | ||
amtool silence import foo.json | ||
JSON data can also come from stdin if no param is specified. | ||
`, | ||
Args: cobra.MaximumNArgs(1), | ||
Run: CommandWrapper(bulkImport), | ||
} | ||
|
||
func init() { | ||
importCmd.Flags().BoolP("force", "f", false, "Force adding new silences even if it already exists") | ||
importCmd.Flags().IntP("worker", "w", 8, "Number of concurrent workers to use for import") | ||
importFlags = importCmd.Flags() | ||
} | ||
|
||
func addSilenceWorker(silences <-chan *types.Silence, errs chan<- error) { | ||
for s := range silences { | ||
silenceId, err := addSilence(s) | ||
if err != nil && err.Error() == "[bad_data] not found" { | ||
// silence doesn't exists yet, retry to create as a new one | ||
s.ID = "" | ||
silenceId, err = addSilence(s) | ||
} | ||
|
||
if err != nil { | ||
log.Errorf("error adding silence %v: %v", s.ID, err) | ||
} else { | ||
fmt.Println(silenceId) | ||
} | ||
errs <- err | ||
} | ||
} | ||
|
||
func bulkImport(cmd *cobra.Command, args []string) error { | ||
force, err := importFlags.GetBool("force") | ||
if err != nil { | ||
return err | ||
} | ||
|
||
input := os.Stdin | ||
if len(args) == 1 { | ||
input, err = os.Open(args[0]) | ||
if err != nil { | ||
return err | ||
} | ||
defer input.Close() | ||
} | ||
|
||
dec := json.NewDecoder(input) | ||
|
||
// read open square bracket | ||
_, err = dec.Token() | ||
if err != nil { | ||
return errors.Wrap(err, "couldn't unmarshal input data, is it JSON?") | ||
} | ||
|
||
silences := make(chan *types.Silence, 100) | ||
errs := make(chan error, 100) | ||
workers, err := importFlags.GetInt("worker") | ||
if err != nil { | ||
return err | ||
} | ||
for w := 0; w < workers; w++ { | ||
go addSilenceWorker(silences, errs) | ||
} | ||
|
||
count := 0 | ||
for dec.More() { | ||
var s types.Silence | ||
err := dec.Decode(&s) | ||
if err != nil { | ||
return errors.Wrap(err, "couldn't unmarshal input data, is it JSON?") | ||
} | ||
|
||
if force { | ||
// reset the silence ID so Alertmanager will always create new silence | ||
s.ID = "" | ||
} | ||
|
||
silences <- &s | ||
count++ | ||
} | ||
close(silences) | ||
|
||
// read closing bracket | ||
_, err = dec.Token() | ||
if err != nil { | ||
return errors.Wrap(err, "invalid JSON") | ||
} | ||
|
||
errCount := 0 | ||
for i := 0; i < count; i++ { | ||
err = <-errs | ||
if err != nil { | ||
errCount++ | ||
} | ||
} | ||
|
||
if errCount > 0 { | ||
return fmt.Errorf("couldn't import %v out of %v silences", errCount, count) | ||
} | ||
|
||
return nil | ||
} |