Improve the presentation of the measurement listing from the CLI

This commit is contained in:
Arturo Filastò 2018-09-13 14:54:56 +02:00
parent 251f136b53
commit 4ed94dfc53
8 changed files with 177 additions and 25 deletions

View File

@ -3,6 +3,7 @@ package main
import ( import (
// commands // commands
"github.com/apex/log"
_ "github.com/ooni/probe-cli/internal/cli/geoip" _ "github.com/ooni/probe-cli/internal/cli/geoip"
_ "github.com/ooni/probe-cli/internal/cli/info" _ "github.com/ooni/probe-cli/internal/cli/info"
_ "github.com/ooni/probe-cli/internal/cli/list" _ "github.com/ooni/probe-cli/internal/cli/list"
@ -19,5 +20,9 @@ import (
) )
func main() { func main() {
crashreport.CapturePanicAndWait(app.Run, nil) err, _ := crashreport.CapturePanic(app.Run, nil)
if err != nil {
log.WithError(err.(error)).Error("panic in app.Run")
crashreport.Wait()
}
} }

View File

@ -32,22 +32,36 @@ func init() {
DataUsageUp: 0.0, DataUsageUp: 0.0,
DataUsageDown: 0.0, DataUsageDown: 0.0,
TotalRuntime: 0, TotalRuntime: 0,
ASN: 0,
NetworkName: "",
NetworkCountryCode: "ZZ",
} }
for _, msmt := range measurements { isFirst := true
isLast := false
for idx, msmt := range measurements {
if idx > 0 {
isFirst = false
}
if idx == len(measurements)-1 {
isLast = true
}
// We assume that since these are summary level information the first // We assume that since these are summary level information the first
// item will contain the information necessary. // item will contain the information necessary.
if msmtSummary.TotalRuntime == 0 { if isFirst {
msmtSummary.TotalRuntime = msmt.ResultRuntime msmtSummary.TotalRuntime = msmt.ResultRuntime
}
if msmtSummary.DataUsageUp == 0 {
msmtSummary.DataUsageUp = msmt.DataUsageUp msmtSummary.DataUsageUp = msmt.DataUsageUp
msmtSummary.DataUsageDown = msmt.DataUsageDown msmtSummary.DataUsageDown = msmt.DataUsageDown
msmtSummary.NetworkName = msmt.NetworkName
msmtSummary.NetworkCountryCode = msmt.NetworkCountryCode
msmtSummary.ASN = msmt.ASN
msmtSummary.StartTime = msmt.MeasurementStartTime
} }
if msmt.IsAnomaly.Bool == true { if msmt.IsAnomaly.Bool == true {
msmtSummary.AnomalyCount++ msmtSummary.AnomalyCount++
} }
msmtSummary.TotalCount++ msmtSummary.TotalCount++
output.MeasurementItem(msmt) output.MeasurementItem(msmt, isFirst, isLast)
} }
output.MeasurementSummary(msmtSummary) output.MeasurementSummary(msmtSummary)
} else { } else {

View File

@ -1,6 +1,7 @@
package crashreport package crashreport
import ( import (
"github.com/apex/log"
"github.com/getsentry/raven-go" "github.com/getsentry/raven-go"
) )
@ -8,13 +9,15 @@ import (
// crash reporting logic a no-op. // crash reporting logic a no-op.
var Disabled = false var Disabled = false
var client *raven.Client
// CapturePanic is a wrapper around raven.CapturePanic that becomes a noop if // CapturePanic is a wrapper around raven.CapturePanic that becomes a noop if
// `Disabled` is set to true. // `Disabled` is set to true.
func CapturePanic(f func(), tags map[string]string) (interface{}, string) { func CapturePanic(f func(), tags map[string]string) (interface{}, string) {
if Disabled == true { if Disabled == true {
return nil, "" return nil, ""
} }
return raven.CapturePanic(f, tags) return client.CapturePanic(f, tags)
} }
// CapturePanicAndWait is a wrapper around raven.CapturePanicAndWait that becomes a noop if // CapturePanicAndWait is a wrapper around raven.CapturePanicAndWait that becomes a noop if
@ -23,7 +26,7 @@ func CapturePanicAndWait(f func(), tags map[string]string) (interface{}, string)
if Disabled == true { if Disabled == true {
return nil, "" return nil, ""
} }
return raven.CapturePanicAndWait(f, tags) return client.CapturePanicAndWait(f, tags)
} }
// CaptureError is a wrapper around raven.CaptureError // CaptureError is a wrapper around raven.CaptureError
@ -31,7 +34,7 @@ func CaptureError(err error, tags map[string]string) string {
if Disabled == true { if Disabled == true {
return "" return ""
} }
return raven.CaptureError(err, tags) return client.CaptureError(err, tags)
} }
// CaptureErrorAndWait is a wrapper around raven.CaptureErrorAndWait // CaptureErrorAndWait is a wrapper around raven.CaptureErrorAndWait
@ -39,9 +42,21 @@ func CaptureErrorAndWait(err error, tags map[string]string) string {
if Disabled == true { if Disabled == true {
return "" return ""
} }
return raven.CaptureErrorAndWait(err, tags) return client.CaptureErrorAndWait(err, tags)
}
// Wait will block on sending messages to the sentry server
func Wait() {
if Disabled == false {
log.Info("sending exception backtrace")
client.Wait()
}
} }
func init() { func init() {
raven.SetDSN("https://cb4510e090f64382ac371040c19b2258:8448daeebfa643c289ef398f8645980b@sentry.io/1234954") var err error
client, err = raven.NewClient("https://cb4510e090f64382ac371040c19b2258:8448daeebfa643c289ef398f8645980b@sentry.io/1234954", nil)
if err != nil {
log.WithError(err).Error("failed to create a raven client")
}
} }

View File

@ -114,6 +114,10 @@ func (h *Handler) TypedLog(t string, e *log.Entry) error {
return nil return nil
case "table": case "table":
return logTable(h.Writer, e.Fields) return logTable(h.Writer, e.Fields)
case "measurement_item":
return logMeasurementItem(h.Writer, e.Fields)
case "measurement_summary":
return logMeasurementSummary(h.Writer, e.Fields)
case "result_item": case "result_item":
return logResultItem(h.Writer, e.Fields) return logResultItem(h.Writer, e.Fields)
case "result_summary": case "result_summary":

View File

@ -0,0 +1,103 @@
package cli
import (
"fmt"
"io"
"strings"
"time"
"github.com/apex/log"
"github.com/ooni/probe-cli/internal/util"
)
func statusIcon(ok bool) string {
if ok {
return "✓"
}
return "❌"
}
func logMeasurementItem(w io.Writer, f log.Fields) error {
colWidth := 24
rID := f.Get("id").(int64)
testName := f.Get("test_name").(string)
// We currently don't use these fields in the view
//testGroupName := f.Get("test_group_name").(string)
//networkName := f.Get("network_name").(string)
//asn := fmt.Sprintf("AS%d (%s)", f.Get("asn").(uint), f.Get("network_country_code").(string))
testKeys := f.Get("test_keys").(string)
isAnomaly := f.Get("is_anomaly").(bool)
isFailed := f.Get("is_failed").(bool)
isUploaded := f.Get("is_uploaded").(bool)
url := f.Get("url").(string)
urlCategoryCode := f.Get("url_category_code").(string)
isFirst := f.Get("is_first").(bool)
isLast := f.Get("is_last").(bool)
if isFirst {
fmt.Fprintf(w, "┏"+strings.Repeat("━", colWidth*2+2)+"┓\n")
} else {
fmt.Fprintf(w, "┢"+strings.Repeat("━", colWidth*2+2)+"┪\n")
}
anomalyStr := fmt.Sprintf("ok: %s", statusIcon(!isAnomaly))
uploadStr := fmt.Sprintf("uploaded: %s", statusIcon(isUploaded))
failureStr := fmt.Sprintf("success: %s", statusIcon(!isFailed))
fmt.Fprintf(w, fmt.Sprintf("│ %s │\n",
util.RightPad(
fmt.Sprintf("#%d", rID), colWidth*2)))
if url != "" {
fmt.Fprintf(w, fmt.Sprintf("│ %s │\n",
util.RightPad(
fmt.Sprintf("%s (%s)", url, urlCategoryCode), colWidth*2)))
}
fmt.Fprintf(w, fmt.Sprintf("│ %s %s│\n",
util.RightPad(testName, colWidth),
util.RightPad(anomalyStr, colWidth)))
fmt.Fprintf(w, fmt.Sprintf("│ %s │\n",
util.RightPad(testKeys, colWidth*2)))
fmt.Fprintf(w, fmt.Sprintf("│ %s %s│\n",
util.RightPad(failureStr, colWidth),
util.RightPad(uploadStr, colWidth)))
if isLast {
fmt.Fprintf(w, "└┬────────────────────────────────────────────────┬┘\n")
}
return nil
}
func logMeasurementSummary(w io.Writer, f log.Fields) error {
colWidth := 12
totalCount := f.Get("total_count").(int64)
anomalyCount := f.Get("anomaly_count").(int64)
totalRuntime := f.Get("total_runtime").(float64)
dataUp := f.Get("data_usage_up").(float64)
dataDown := f.Get("data_usage_down").(float64)
startTime := f.Get("start_time").(time.Time)
asn := f.Get("asn").(uint)
countryCode := f.Get("network_country_code").(string)
networkName := f.Get("network_name").(string)
fmt.Fprintf(w, " │ %s │\n",
util.RightPad(startTime.Format(time.RFC822), (colWidth+3)*3),
)
fmt.Fprintf(w, " │ %s │\n",
util.RightPad(fmt.Sprintf("AS%d, %s (%s)", asn, networkName, countryCode), (colWidth+3)*3),
)
fmt.Fprintf(w, " │ %s %s %s │\n",
util.RightPad(fmt.Sprintf("%.2fs", totalRuntime), colWidth),
util.RightPad(fmt.Sprintf("%d/%d anmls", anomalyCount, totalCount), colWidth),
util.RightPad(fmt.Sprintf("⬆ %s ⬇ %s", formatSize(dataUp), formatSize(dataDown)), colWidth+4))
fmt.Fprintf(w, " └────────────────────────────────────────────────┘\n")
return nil
}

View File

@ -126,7 +126,7 @@ func Onboarding(config *config.Config) error {
config.Lock() config.Lock()
config.InformedConsent = true config.InformedConsent = true
config.Advanced.IncludeCountry = settings.IncludeCountry config.Sharing.IncludeCountry = settings.IncludeCountry
config.Advanced.SendCrashReports = settings.SendCrashReports config.Advanced.SendCrashReports = settings.SendCrashReports
config.Sharing.IncludeIP = settings.IncludeIP config.Sharing.IncludeIP = settings.IncludeIP
config.Sharing.IncludeASN = settings.IncludeNetwork config.Sharing.IncludeASN = settings.IncludeNetwork

View File

@ -26,6 +26,10 @@ type MeasurementSummaryData struct {
AnomalyCount int64 AnomalyCount int64
DataUsageUp float64 DataUsageUp float64
DataUsageDown float64 DataUsageDown float64
ASN uint
NetworkName string
NetworkCountryCode string
StartTime time.Time
} }
func MeasurementSummary(msmt MeasurementSummaryData) { func MeasurementSummary(msmt MeasurementSummaryData) {
@ -36,13 +40,20 @@ func MeasurementSummary(msmt MeasurementSummaryData) {
"anomaly_count": msmt.AnomalyCount, "anomaly_count": msmt.AnomalyCount,
"data_usage_down": msmt.DataUsageDown, "data_usage_down": msmt.DataUsageDown,
"data_usage_up": msmt.DataUsageUp, "data_usage_up": msmt.DataUsageUp,
"asn": msmt.ASN,
"network_country_code": msmt.NetworkCountryCode,
"network_name": msmt.NetworkName,
"start_time": msmt.StartTime,
}).Info("measurement summary") }).Info("measurement summary")
} }
// MeasurementItem logs a progress type event // MeasurementItem logs a progress type event
func MeasurementItem(msmt database.MeasurementURLNetwork) { func MeasurementItem(msmt database.MeasurementURLNetwork, isFirst bool, isLast bool) {
log.WithFields(log.Fields{ log.WithFields(log.Fields{
"type": "measurement_item", "type": "measurement_item",
"is_first": isFirst,
"is_last": isLast,
"id": msmt.MsmtTblID, "id": msmt.MsmtTblID,
"test_name": msmt.TestName, "test_name": msmt.TestName,
"test_group_name": msmt.Result.TestGroupName, "test_group_name": msmt.Result.TestGroupName,

View File

@ -120,7 +120,7 @@ func (c *Controller) Init(nt *mk.Nettest) error {
nt.Options = mk.NettestOptions{ nt.Options = mk.NettestOptions{
IncludeIP: c.Ctx.Config.Sharing.IncludeIP, IncludeIP: c.Ctx.Config.Sharing.IncludeIP,
IncludeASN: c.Ctx.Config.Sharing.IncludeASN, IncludeASN: c.Ctx.Config.Sharing.IncludeASN,
IncludeCountry: c.Ctx.Config.Advanced.IncludeCountry, IncludeCountry: c.Ctx.Config.Sharing.IncludeCountry,
LogLevel: "DEBUG", LogLevel: "DEBUG",
ProbeCC: c.Ctx.Location.CountryCode, ProbeCC: c.Ctx.Location.CountryCode,