blob: f658b22a7eee2cc2114d67303ef2e1f0a802216f [file] [log] [blame]
Bertrand SIMONNET52e5b992015-08-10 15:18:00 -07001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
Darin Petkov65b01462010-04-14 13:32:20 -070016
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070017#include <cstdio>
Darin Petkov65b01462010-04-14 13:32:20 -070018#include <cstdlib>
Darin Petkov65b01462010-04-14 13:32:20 -070019
James Hawkins5f646002015-09-08 15:18:17 -070020#include <base/memory/scoped_vector.h>
21
22#include "constants.h"
Bertrand SIMONNETe6cfd642014-07-09 16:35:23 -070023#include "metrics/metrics_library.h"
James Hawkins5f646002015-09-08 15:18:17 -070024#include "serialization/metric_sample.h"
25#include "serialization/serialization_utils.h"
Darin Petkov65b01462010-04-14 13:32:20 -070026
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -070027enum Mode {
James Hawkins5f646002015-09-08 15:18:17 -070028 kModeDumpLogs,
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -070029 kModeSendSample,
30 kModeSendEnumSample,
31 kModeSendSparseSample,
32 kModeSendUserAction,
33 kModeSendCrosEvent,
34 kModeHasConsent,
35 kModeIsGuestMode,
36};
37
Ken Mixtereafbbdf2010-10-01 15:38:42 -070038void ShowUsage() {
39 fprintf(stderr,
Bertrand SIMONNET475dfa62015-08-04 14:10:10 -070040 "Usage: metrics_client [-t] name sample min max nbuckets\n"
41 " metrics_client -e name sample max\n"
42 " metrics_client -s name sample\n"
43 " metrics_client -v event\n"
Darin Petkoved824852011-01-06 10:51:47 -080044 " metrics_client -u action\n"
James Hawkins5f646002015-09-08 15:18:17 -070045 " metrics_client [-cdg]\n"
Ken Mixtereafbbdf2010-10-01 15:38:42 -070046 "\n"
Bertrand SIMONNET475dfa62015-08-04 14:10:10 -070047 " default: send metric with integer values \n"
Ken Mixtereafbbdf2010-10-01 15:38:42 -070048 " |min| > 0, |min| <= sample < |max|\n"
Thiemo Nagel0b8cc1c2014-08-21 15:00:50 +020049 " -c: return exit status 0 if user consents to stats, 1 otherwise,\n"
50 " in guest mode always return 1\n"
James Hawkins5f646002015-09-08 15:18:17 -070051 " -d: dump cached logs to the console\n"
Ken Mixtereafbbdf2010-10-01 15:38:42 -070052 " -e: send linear/enumeration histogram data\n"
53 " -g: return exit status 0 if machine in guest mode, 1 otherwise\n"
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -070054 " -s: send a sparse histogram sample\n"
Darin Petkoved824852011-01-06 10:51:47 -080055 " -t: convert sample from double seconds to int milliseconds\n"
Luigi Semenzato32684222013-03-13 10:53:55 -070056 " -u: send a user action to Chrome\n"
Bertrand SIMONNETe6cfd642014-07-09 16:35:23 -070057 " -v: send a Platform.CrOSEvent enum histogram sample\n");
Ken Mixtereafbbdf2010-10-01 15:38:42 -070058 exit(1);
59}
Darin Petkov65b01462010-04-14 13:32:20 -070060
Luigi Semenzatod8abf552014-03-27 14:19:06 -070061static int ParseInt(const char *arg) {
62 char *endptr;
63 int value = strtol(arg, &endptr, 0);
64 if (*endptr != '\0') {
65 fprintf(stderr, "metrics client: bad integer \"%s\"\n", arg);
66 ShowUsage();
67 }
68 return value;
69}
70
71static double ParseDouble(const char *arg) {
72 char *endptr;
73 double value = strtod(arg, &endptr);
74 if (*endptr != '\0') {
75 fprintf(stderr, "metrics client: bad double \"%s\"\n", arg);
76 ShowUsage();
77 }
78 return value;
79}
80
Ken Mixtereafbbdf2010-10-01 15:38:42 -070081static int SendStats(char* argv[],
82 int name_index,
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -070083 enum Mode mode,
Bertrand SIMONNET475dfa62015-08-04 14:10:10 -070084 bool secs_to_msecs) {
Darin Petkovc2526a12010-04-21 14:24:04 -070085 const char* name = argv[name_index];
86 int sample;
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070087 if (secs_to_msecs) {
Luigi Semenzatod8abf552014-03-27 14:19:06 -070088 sample = static_cast<int>(ParseDouble(argv[name_index + 1]) * 1000.0);
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070089 } else {
Luigi Semenzatod8abf552014-03-27 14:19:06 -070090 sample = ParseInt(argv[name_index + 1]);
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070091 }
92
Bertrand SIMONNET475dfa62015-08-04 14:10:10 -070093 MetricsLibrary metrics_lib;
94 metrics_lib.Init();
95 if (mode == kModeSendSparseSample) {
96 metrics_lib.SendSparseToUMA(name, sample);
97 } else if (mode == kModeSendEnumSample) {
98 int max = ParseInt(argv[name_index + 2]);
99 metrics_lib.SendEnumToUMA(name, sample, max);
100 } else {
101 int min = ParseInt(argv[name_index + 2]);
102 int max = ParseInt(argv[name_index + 3]);
103 int nbuckets = ParseInt(argv[name_index + 4]);
104 metrics_lib.SendToUMA(name, sample, min, max, nbuckets);
Darin Petkov65b01462010-04-14 13:32:20 -0700105 }
106 return 0;
107}
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700108
Darin Petkoved824852011-01-06 10:51:47 -0800109static int SendUserAction(char* argv[], int action_index) {
110 const char* action = argv[action_index];
111 MetricsLibrary metrics_lib;
112 metrics_lib.Init();
113 metrics_lib.SendUserActionToUMA(action);
114 return 0;
115}
116
Luigi Semenzato32684222013-03-13 10:53:55 -0700117static int SendCrosEvent(char* argv[], int action_index) {
118 const char* event = argv[action_index];
119 bool result;
120 MetricsLibrary metrics_lib;
121 metrics_lib.Init();
122 result = metrics_lib.SendCrosEventToUMA(event);
123 if (!result) {
124 fprintf(stderr, "metrics_client: could not send event %s\n", event);
125 return 1;
126 }
127 return 0;
128}
129
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700130static int HasConsent() {
131 MetricsLibrary metrics_lib;
132 metrics_lib.Init();
133 return metrics_lib.AreMetricsEnabled() ? 0 : 1;
134}
135
136static int IsGuestMode() {
137 MetricsLibrary metrics_lib;
138 metrics_lib.Init();
139 return metrics_lib.IsGuestMode() ? 0 : 1;
140}
141
James Hawkins5f646002015-09-08 15:18:17 -0700142static int DumpLogs() {
143 printf("Metrics from %s\n\n", metrics::kMetricsEventsFilePath);
144
145 ScopedVector<metrics::MetricSample> metrics;
146 metrics::SerializationUtils::ReadMetricsFromFile(
147 metrics::kMetricsEventsFilePath, &metrics);
148
149 for (ScopedVector<metrics::MetricSample>::const_iterator i = metrics.begin();
150 i != metrics.end(); ++i) {
151 const metrics::MetricSample* sample = *i;
152 printf("name: %s\t", sample->name().c_str());
153 printf("type: ");
154
155 switch (sample->type()) {
156 case metrics::MetricSample::CRASH:
157 printf("CRASH");
158 break;
159 case metrics::MetricSample::HISTOGRAM:
160 printf("HISTOGRAM");
161 break;
162 case metrics::MetricSample::LINEAR_HISTOGRAM:
163 printf("LINEAR_HISTOGRAM");
164 break;
165 case metrics::MetricSample::SPARSE_HISTOGRAM:
166 printf("SPARSE_HISTOGRAM");
167 break;
168 case metrics::MetricSample::USER_ACTION:
169 printf("USER_ACTION");
170 break;
171 }
172
173 printf("\n");
174 }
175
176 return 0;
177}
178
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700179int main(int argc, char** argv) {
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700180 enum Mode mode = kModeSendSample;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700181 bool secs_to_msecs = false;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700182
183 // Parse arguments
184 int flag;
James Hawkins5f646002015-09-08 15:18:17 -0700185 while ((flag = getopt(argc, argv, "abcdegstuv")) != -1) {
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700186 switch (flag) {
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700187 case 'c':
188 mode = kModeHasConsent;
189 break;
James Hawkins5f646002015-09-08 15:18:17 -0700190 case 'd':
191 mode = kModeDumpLogs;
192 break;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700193 case 'e':
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700194 mode = kModeSendEnumSample;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700195 break;
196 case 'g':
197 mode = kModeIsGuestMode;
198 break;
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700199 case 's':
200 mode = kModeSendSparseSample;
201 break;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700202 case 't':
203 secs_to_msecs = true;
204 break;
Darin Petkoved824852011-01-06 10:51:47 -0800205 case 'u':
206 mode = kModeSendUserAction;
207 break;
Luigi Semenzato32684222013-03-13 10:53:55 -0700208 case 'v':
209 mode = kModeSendCrosEvent;
210 break;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700211 default:
Darin Petkov8032dd02011-05-09 16:33:19 -0700212 ShowUsage();
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700213 break;
214 }
215 }
Darin Petkoved824852011-01-06 10:51:47 -0800216 int arg_index = optind;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700217
218 int expected_args = 0;
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700219 if (mode == kModeSendSample)
220 expected_args = 5;
221 else if (mode == kModeSendEnumSample)
222 expected_args = 3;
223 else if (mode == kModeSendSparseSample)
224 expected_args = 2;
Darin Petkoved824852011-01-06 10:51:47 -0800225 else if (mode == kModeSendUserAction)
226 expected_args = 1;
Luigi Semenzato32684222013-03-13 10:53:55 -0700227 else if (mode == kModeSendCrosEvent)
228 expected_args = 1;
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700229
Darin Petkoved824852011-01-06 10:51:47 -0800230 if ((arg_index + expected_args) != argc) {
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700231 ShowUsage();
232 }
233
Bertrand SIMONNETe6cfd642014-07-09 16:35:23 -0700234 switch (mode) {
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700235 case kModeSendSample:
236 case kModeSendEnumSample:
237 case kModeSendSparseSample:
238 if ((mode != kModeSendSample) && secs_to_msecs) {
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700239 ShowUsage();
240 }
241 return SendStats(argv,
Darin Petkoved824852011-01-06 10:51:47 -0800242 arg_index,
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700243 mode,
Bertrand SIMONNET475dfa62015-08-04 14:10:10 -0700244 secs_to_msecs);
Darin Petkoved824852011-01-06 10:51:47 -0800245 case kModeSendUserAction:
246 return SendUserAction(argv, arg_index);
Luigi Semenzato32684222013-03-13 10:53:55 -0700247 case kModeSendCrosEvent:
248 return SendCrosEvent(argv, arg_index);
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700249 case kModeHasConsent:
250 return HasConsent();
251 case kModeIsGuestMode:
252 return IsGuestMode();
James Hawkins5f646002015-09-08 15:18:17 -0700253 case kModeDumpLogs:
254 return DumpLogs();
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700255 default:
256 ShowUsage();
257 return 0;
258 }
259}