main: format exit errors as JSON when using --json

This commit is contained in:
Michael Terry 2024-07-28 09:08:10 -04:00
parent 2d250a9135
commit 6283915f86
2 changed files with 42 additions and 6 deletions

View file

@ -0,0 +1,6 @@
Enhancement: Format exit errors as JSON with --json
Restic now prints any exit error messages as JSON when requested.
https://github.com/restic/restic/issues/4948
https://github.com/restic/restic/pull/4952

View file

@ -4,6 +4,7 @@ import (
"bufio" "bufio"
"bytes" "bytes"
"context" "context"
"encoding/json"
"fmt" "fmt"
"log" "log"
"os" "os"
@ -102,6 +103,30 @@ func tweakGoGC() {
} }
} }
func printExitError(code int, message string) {
if globalOptions.JSON {
type jsonExitError struct {
MessageType string `json:"message_type"` // exit_error
Code int `json:"code"`
Message string `json:"message"`
}
jsonS := jsonExitError{
MessageType: "exit_error",
Code: code,
Message: message,
}
err := json.NewEncoder(globalOptions.stderr).Encode(jsonS)
if err != nil {
Warnf("JSON encode failed: %v\n", err)
return
}
} else {
fmt.Fprintf(globalOptions.stderr, "%v\n", message)
}
}
func main() { func main() {
tweakGoGC() tweakGoGC()
// install custom global logger into a buffer, if an error occurs // install custom global logger into a buffer, if an error occurs
@ -131,21 +156,22 @@ func main() {
err = nil err = nil
} }
var exitMessage string
switch { switch {
case restic.IsAlreadyLocked(err): case restic.IsAlreadyLocked(err):
fmt.Fprintf(os.Stderr, "%v\nthe `unlock` command can be used to remove stale locks\n", err) exitMessage = fmt.Sprintf("%v\nthe `unlock` command can be used to remove stale locks", err)
case err == ErrInvalidSourceData: case err == ErrInvalidSourceData:
fmt.Fprintf(os.Stderr, "Warning: %v\n", err) exitMessage = fmt.Sprintf("Warning: %v", err)
case errors.IsFatal(err): case errors.IsFatal(err):
fmt.Fprintf(os.Stderr, "%v\n", err) exitMessage = err.Error()
case err != nil: case err != nil:
fmt.Fprintf(os.Stderr, "%+v\n", err) exitMessage = fmt.Sprintf("%+v", err)
if logBuffer.Len() > 0 { if logBuffer.Len() > 0 {
fmt.Fprintf(os.Stderr, "also, the following messages were logged by a library:\n") exitMessage += "also, the following messages were logged by a library:\n"
sc := bufio.NewScanner(logBuffer) sc := bufio.NewScanner(logBuffer)
for sc.Scan() { for sc.Scan() {
fmt.Fprintln(os.Stderr, sc.Text()) exitMessage += fmt.Sprintln(sc.Text())
} }
} }
} }
@ -165,5 +191,9 @@ func main() {
default: default:
exitCode = 1 exitCode = 1
} }
if exitCode != 0 {
printExitError(exitCode, exitMessage)
}
Exit(exitCode) Exit(exitCode)
} }