diff --git a/api/log/client.go b/api/log/client.go index 57238a0..274b4aa 100644 --- a/api/log/client.go +++ b/api/log/client.go @@ -3,6 +3,7 @@ package log import ( "context" "fmt" + "log" "os" "sort" "time" @@ -16,6 +17,8 @@ import ( "github.com/grafana/loki/pkg/logqlmodel" "github.com/grafana/loki/pkg/util/unmarshal" "github.com/prometheus/common/config" + "k8s.io/apimachinery/pkg/util/wait" + "k8s.io/client-go/util/retry" ) type Client struct { @@ -89,6 +92,39 @@ func (c *Client) QueryRange(ctx context.Context, out output.LogOutput, q Query) return printResult(resp.Data.Result, out) } +// QueryRangeWithRetry queries logs within a specific time range with a retry +// in case of an error or not finding any logs. +func (c *Client) QueryRangeWithRetry(ctx context.Context, out output.LogOutput, q Query) error { + return retry.OnError( + wait.Backoff{ + Steps: 5, + Duration: 200 * time.Millisecond, + Factor: 2.0, + Jitter: 0.1, + Cap: 10 * time.Second, + }, + func(err error) bool { + // retry regardless of the error + return true + }, + func() error { + resp, err := c.Client.QueryRange(q.QueryString, q.Limit, q.Start, q.End, q.Direction, q.Step, q.Interval, q.Quiet) + if err != nil { + log.Println(err) + return err + } + + switch streams := resp.Data.Result.(type) { + case loghttp.Streams: + if len(streams) == 0 { + return fmt.Errorf("received no log streams") + } + } + + return printResult(resp.Data.Result, out) + }) +} + func printResult(value loghttp.ResultValue, out output.LogOutput) error { switch value.Type() { case logqlmodel.ValueTypeStreams: diff --git a/create/application.go b/create/application.go index fa35a94..9c6a8e1 100644 --- a/create/application.go +++ b/create/application.go @@ -420,14 +420,14 @@ func printUnverifiedHostsMessage(app *apps.Application) { } func printBuildLogs(ctx context.Context, client *api.Client, build *apps.Build) error { - return client.Log.QueryRange( + return client.Log.QueryRangeWithRetry( ctx, client.Log.StdOut, errorLogQuery(logs.BuildQuery(build.Name, build.Namespace)), ) } func printReleaseLogs(ctx context.Context, client *api.Client, release *apps.Release) error { - return client.Log.QueryRange( + return client.Log.QueryRangeWithRetry( ctx, client.Log.StdOut, errorLogQuery(logs.ApplicationQuery(release.Labels[util.ApplicationNameLabel], release.Namespace)), )