|  | // Package parser parses incoming JSON files from Android Testing and converts them | 
|  | // into a format acceptable to Skia Perf. | 
|  | package parser | 
|  |  | 
|  | import ( | 
|  | "bytes" | 
|  | "encoding/json" | 
|  | "errors" | 
|  | "fmt" | 
|  | "io" | 
|  | "strconv" | 
|  | "strings" | 
|  |  | 
|  | "go.skia.org/infra/go/metrics2" | 
|  | "go.skia.org/infra/go/skerr" | 
|  | "go.skia.org/infra/go/sklog" | 
|  | "go.skia.org/infra/perf/go/ingest/format" | 
|  | ) | 
|  |  | 
|  | const rawLogLocationKey = "raw_log_location" | 
|  |  | 
|  | var ( | 
|  | // ErrIgnorable is returned from Convert if the file can safely be ignored. | 
|  | ErrIgnorable = errors.New("File should be ignored.") | 
|  | ) | 
|  |  | 
|  | // Incoming is the JSON structure of the data sent to us from the Android | 
|  | // testing infrastructure. | 
|  | type Incoming struct { | 
|  | BuildId        string `json:"build_id"` | 
|  | BuildFlavor    string `json:"build_flavor"` | 
|  | Branch         string `json:"branch"` | 
|  | DeviceName     string `json:"device_name"` | 
|  | SDKReleaseName string `json:"sdk_release_name"` | 
|  | JIT            string `json:"jit"` | 
|  |  | 
|  | // Metrics is a map[test name]map[metric]value, where value | 
|  | // is a string encoded float, thus the use of json.Number. | 
|  | Metrics map[string]map[string]json.Number `json:"metrics"` | 
|  | } | 
|  |  | 
|  | // Parse the 'incoming' stream into an *Incoming struct. | 
|  | func Parse(incoming io.Reader) (*Incoming, error) { | 
|  | ret := &Incoming{} | 
|  | if err := json.NewDecoder(incoming).Decode(ret); err != nil { | 
|  | return nil, fmt.Errorf("Failed to decode incoming JSON: %s", err) | 
|  | } | 
|  | return ret, nil | 
|  | } | 
|  |  | 
|  | // Lookup is an interface for looking up a git hashes from a buildid. | 
|  | // | 
|  | // The *lookup.Cache satisfies this interface. | 
|  | type Lookup interface { | 
|  | Lookup(buildid int64) (string, error) | 
|  | } | 
|  |  | 
|  | // Converter converts a serialized *Incoming into an *format.BenchData. | 
|  | type Converter struct { | 
|  | lookup Lookup | 
|  | } | 
|  |  | 
|  | // New creates a new *Converter. | 
|  | func New(lookup Lookup) *Converter { | 
|  | return &Converter{ | 
|  | lookup: lookup, | 
|  | } | 
|  | } | 
|  |  | 
|  | // Convert the serialize *Incoming JSON into a JSON serialized format that Perf | 
|  | // supports. Also return the global keys and the buildID from the parsed file. | 
|  | func (c *Converter) Convert(incoming io.Reader, txLogName string) (map[string]string, string, []byte, error) { | 
|  | b, err := io.ReadAll(incoming) | 
|  | if err != nil { | 
|  | return nil, "", nil, skerr.Wrapf(err, "Failed to read during convert %q", txLogName) | 
|  | } | 
|  |  | 
|  | key, gitHash, encodedAsJSON, err := c.convertFromV1Format(b, txLogName) | 
|  | if err != nil { | 
|  | return c.convertFromAndroidSpecificFormat(b, txLogName) | 
|  | } | 
|  | return key, gitHash, encodedAsJSON, nil | 
|  | } | 
|  |  | 
|  | func (c *Converter) convertFromV1Format(b []byte, txLogName string) (map[string]string, string, []byte, error) { | 
|  | in, err := format.Parse(bytes.NewReader(b)) | 
|  | if err != nil { | 
|  | return nil, "", nil, fmt.Errorf("Failed to parse during convert %q: %s", txLogName, err) | 
|  | } | 
|  | // The Build ID is supplied via the GitHash. | 
|  | buildID := in.GitHash | 
|  |  | 
|  | metrics2.GetCounter("androidingest_upload_received_v1_format").Inc(1) | 
|  | sklog.Infof("POST V1 Format for filename %q buildid: %s num metrics: %d", txLogName, buildID, len(in.Results)) | 
|  |  | 
|  | // Files where the BuildId is prefixed with "P" are presubmits and don't | 
|  | // produce any data and can be ignored. | 
|  | if strings.HasPrefix(buildID, "P") { | 
|  | return nil, "", nil, ErrIgnorable | 
|  | } | 
|  | buildid, err := strconv.ParseInt(buildID, 10, 64) | 
|  | if err != nil { | 
|  | return nil, "", nil, fmt.Errorf("parse buildid %q: %q %s", buildID, txLogName, err) | 
|  | } | 
|  | hash, err := c.lookup.Lookup(buildid) | 
|  | if err != nil { | 
|  | return nil, "", nil, fmt.Errorf("find matching hash for buildid %d: %q %s", buildid, txLogName, err) | 
|  | } | 
|  |  | 
|  | in.GitHash = hash | 
|  | if in.Links == nil { | 
|  | in.Links = map[string]string{} | 
|  | } | 
|  | in.Links[rawLogLocationKey] = txLogName | 
|  |  | 
|  | encodedAsJSON, err := json.MarshalIndent(in, "", "  ") | 
|  | if err != nil { | 
|  | return nil, "", nil, skerr.Wrapf(err, "encoding benchData") | 
|  | } | 
|  |  | 
|  | if len(in.Results) == 0 { | 
|  | sklog.Warningf("Failed to extract any data from incoming file: %q", txLogName) | 
|  | return nil, "", nil, ErrIgnorable | 
|  | } | 
|  |  | 
|  | return in.Key, in.GitHash, encodedAsJSON, nil | 
|  | } | 
|  |  | 
|  | // TODO(jcgregorio) This should be changed to emit the V1 Format. | 
|  | func (c *Converter) convertFromAndroidSpecificFormat(b []byte, txLogName string) (map[string]string, string, []byte, error) { | 
|  | in, err := Parse(bytes.NewReader(b)) | 
|  | if err != nil { | 
|  | return nil, "", nil, fmt.Errorf("parse during convert %q: %s", txLogName, err) | 
|  | } | 
|  | metrics2.GetCounter("androidingest_upload_received", map[string]string{"branch": in.Branch}).Inc(1) | 
|  | sklog.Infof("POST for filename %q buildid: %s branch: %s flavor: %s num metrics: %d", txLogName, in.BuildId, in.Branch, in.BuildFlavor, len(in.Metrics)) | 
|  |  | 
|  | // Files where the BuildId is prefixed with "P" are presubmits and don't | 
|  | // produce any data and can be ignored. | 
|  | if strings.HasPrefix(in.BuildId, "P") { | 
|  | return nil, "", nil, ErrIgnorable | 
|  | } | 
|  | buildid, err := strconv.ParseInt(in.BuildId, 10, 64) | 
|  | if err != nil { | 
|  | return nil, "", nil, fmt.Errorf("parse buildid %q: %q %q %s", in.BuildId, txLogName, in.Branch, err) | 
|  | } | 
|  | hash, err := c.lookup.Lookup(buildid) | 
|  | if err != nil { | 
|  | return nil, "", nil, fmt.Errorf("find matching hash for buildid %d: %q %q %s", buildid, txLogName, in.Branch, err) | 
|  | } | 
|  |  | 
|  | // Convert Incoming into format.BenchData, i.e. convert the following: | 
|  | // | 
|  | //		{ | 
|  | //			"build_id": "3567162", | 
|  | //			"build_flavor": "marlin-userdebug", | 
|  | //			"metrics": { | 
|  | //				"android.platform.systemui.tests.jank.LauncherJankTests#testAppSwitchGMailtoHome": { | 
|  | //				"frame-fps": "9.328892269753897", | 
|  | //				"frame-avg-jank": "8.4", | 
|  | //				"frame-max-frame-duration": "7.834711093388444", | 
|  | //				"frame-max-jank": "10" | 
|  | //			}, | 
|  | //	    ... | 
|  | //    } | 
|  | //  } | 
|  | // | 
|  | //  into | 
|  | // | 
|  | //  { | 
|  | //    "gitHash" : "8dcc84f7dc8523dd90501a4feb1f632808337c34", | 
|  | //    "key" : { | 
|  | //      "build_flavor" : "marlin-userdebug" | 
|  | //    }, | 
|  | //    "results" : { | 
|  | //      "android.platform.systemui.tests.jank.LauncherJankTests#testAppSwitchGMailtoHome" : { | 
|  | //        "default" : { | 
|  | //          "frame-fps": 9.328892269753897, | 
|  | //          "frame-avg-jank": 8.4, | 
|  | //          "frame-max-frame-duration": 7.834711093388444, | 
|  | //          "frame-max-jank": 10 | 
|  | //          "options" : { | 
|  | //            "name" : "android.platform.systemui.tests.jank.LauncherJankTests", | 
|  | //            "subtest" : "testAppSwitchGMailtoHome", | 
|  | //          }, | 
|  | //        }, | 
|  | //      } | 
|  | //    } | 
|  | //  } | 
|  | // | 
|  | // Note that the incoming data doesn't have a concept similar to "config" so we just | 
|  | // use a value of "default" for config for now. | 
|  | benchData := &format.BenchData{ | 
|  | Hash:   hash, | 
|  | Source: txLogName, | 
|  | Key: map[string]string{ | 
|  | "build_flavor": in.BuildFlavor, | 
|  | }, | 
|  | Results: map[string]format.BenchResults{}, | 
|  | } | 
|  | if in.DeviceName != "" { | 
|  | benchData.Key["device_name"] = in.DeviceName | 
|  | } | 
|  | if in.SDKReleaseName != "" { | 
|  | benchData.Key["sdk_release_name"] = in.SDKReleaseName | 
|  | } | 
|  | if in.JIT != "" { | 
|  | benchData.Key["jit"] = in.JIT | 
|  | } | 
|  |  | 
|  | // Record the branch name. | 
|  | benchData.Key["branch"] = in.Branch | 
|  |  | 
|  | for test, metrics := range in.Metrics { | 
|  | benchData.Results[test] = format.BenchResults{} | 
|  | benchData.Results[test]["default"] = format.BenchResult{} | 
|  | for key, value := range metrics { | 
|  | f, err := value.Float64() | 
|  | if err != nil { | 
|  | sklog.Warningf("Couldn't parse %q as a float64: %s", value.String(), err) | 
|  | continue | 
|  | } | 
|  | benchData.Results[test]["default"][key] = f | 
|  | } | 
|  | } | 
|  | sklog.Infof("Found %d metrics of %d incoming metrics in branch %q buildid %q in file %q", len(benchData.Results), len(in.Metrics), in.Branch, in.BuildId, txLogName) | 
|  | metrics2.GetCounter("androidingest_upload_success", map[string]string{"branch": in.Branch}).Inc(1) | 
|  |  | 
|  | encodedAsJSON, err := json.MarshalIndent(benchData, "", "  ") | 
|  | if err != nil { | 
|  | return nil, "", nil, skerr.Wrapf(err, "encoding benchData") | 
|  | } | 
|  |  | 
|  | if len(benchData.Results) == 0 { | 
|  | sklog.Warningf("Failed to extract any data from incoming file: %q", txLogName) | 
|  | return nil, "", nil, ErrIgnorable | 
|  | } | 
|  |  | 
|  | return benchData.Key, benchData.Hash, encodedAsJSON, nil | 
|  | } |