toolkit/components/telemetry/tests/unit/test_nsITelemetry.js

Wed, 31 Dec 2014 06:09:35 +0100

author
Michael Schloh von Bennewitz <michael@schloh.com>
date
Wed, 31 Dec 2014 06:09:35 +0100
changeset 0
6474c204b198
permissions
-rw-r--r--

Cloned upstream origin tor-browser at tor-browser-31.3.0esr-4.5-1-build1
revision ID fc1c9ff7c1b2defdbc039f12214767608f46423f for hacking purpose.

michael@0 1 /* Any copyright is dedicated to the Public Domain.
michael@0 2 http://creativecommons.org/publicdomain/zero/1.0/ */
michael@0 3
michael@0 4 const Cc = Components.classes;
michael@0 5 const Ci = Components.interfaces;
michael@0 6 const Cu = Components.utils;
michael@0 7 const INT_MAX = 0x7FFFFFFF;
michael@0 8
michael@0 9 const Telemetry = Cc["@mozilla.org/base/telemetry;1"].getService(Ci.nsITelemetry);
michael@0 10 Cu.import("resource://gre/modules/Services.jsm", this);
michael@0 11
michael@0 12 function test_expired_histogram() {
michael@0 13 var histogram_id = "FOOBAR";
michael@0 14 var test_expired_id = "TELEMETRY_TEST_EXPIRED";
michael@0 15 var clone_id = "ExpiredClone";
michael@0 16 var dummy = Telemetry.newHistogram(histogram_id, "28.0a1", 1, 2, 3, Telemetry.HISTOGRAM_EXPONENTIAL);
michael@0 17 var dummy_clone = Telemetry.histogramFrom(clone_id, test_expired_id);
michael@0 18 var rh = Telemetry.registeredHistograms([]);
michael@0 19
michael@0 20 dummy.add(1);
michael@0 21 dummy_clone.add(1);
michael@0 22
michael@0 23 do_check_eq(Telemetry.histogramSnapshots["__expired__"], undefined);
michael@0 24 do_check_eq(Telemetry.histogramSnapshots[histogram_id], undefined);
michael@0 25 do_check_eq(Telemetry.histogramSnapshots[test_expired_id], undefined);
michael@0 26 do_check_eq(Telemetry.histogramSnapshots[clone_id], undefined);
michael@0 27 do_check_eq(rh[test_expired_id], undefined);
michael@0 28 }
michael@0 29
michael@0 30 function test_histogram(histogram_type, name, min, max, bucket_count) {
michael@0 31 var h = Telemetry.newHistogram(name, "never", min, max, bucket_count, histogram_type);
michael@0 32 var r = h.snapshot().ranges;
michael@0 33 var sum = 0;
michael@0 34 var log_sum = 0;
michael@0 35 var log_sum_squares = 0;
michael@0 36 for(var i=0;i<r.length;i++) {
michael@0 37 var v = r[i];
michael@0 38 sum += v;
michael@0 39 if (histogram_type == Telemetry.HISTOGRAM_EXPONENTIAL) {
michael@0 40 var log_v = Math.log(1+v);
michael@0 41 log_sum += log_v;
michael@0 42 log_sum_squares += log_v*log_v;
michael@0 43 }
michael@0 44 h.add(v);
michael@0 45 }
michael@0 46 var s = h.snapshot();
michael@0 47 // verify properties
michael@0 48 do_check_eq(sum, s.sum);
michael@0 49 if (histogram_type == Telemetry.HISTOGRAM_EXPONENTIAL) {
michael@0 50 // We do the log with float precision in C++ and double precision in
michael@0 51 // JS, so there's bound to be tiny discrepancies. Just check the
michael@0 52 // integer part.
michael@0 53 do_check_eq(Math.floor(log_sum), Math.floor(s.log_sum));
michael@0 54 do_check_eq(Math.floor(log_sum_squares), Math.floor(s.log_sum_squares));
michael@0 55 do_check_false("sum_squares_lo" in s);
michael@0 56 do_check_false("sum_squares_hi" in s);
michael@0 57 } else {
michael@0 58 // Doing the math to verify sum_squares was reflected correctly is
michael@0 59 // tedious in JavaScript. Just make sure we have something.
michael@0 60 do_check_neq(s.sum_squares_lo + s.sum_squares_hi, 0);
michael@0 61 do_check_false("log_sum" in s);
michael@0 62 do_check_false("log_sum_squares" in s);
michael@0 63 }
michael@0 64
michael@0 65 // there should be exactly one element per bucket
michael@0 66 for each(var i in s.counts) {
michael@0 67 do_check_eq(i, 1);
michael@0 68 }
michael@0 69 var hgrams = Telemetry.histogramSnapshots
michael@0 70 gh = hgrams[name]
michael@0 71 do_check_eq(gh.histogram_type, histogram_type);
michael@0 72
michael@0 73 do_check_eq(gh.min, min)
michael@0 74 do_check_eq(gh.max, max)
michael@0 75
michael@0 76 // Check that booleans work with nonboolean histograms
michael@0 77 h.add(false);
michael@0 78 h.add(true);
michael@0 79 var s = h.snapshot().counts;
michael@0 80 do_check_eq(s[0], 2)
michael@0 81 do_check_eq(s[1], 2)
michael@0 82
michael@0 83 // Check that clearing works.
michael@0 84 h.clear();
michael@0 85 var s = h.snapshot();
michael@0 86 for each(var i in s.counts) {
michael@0 87 do_check_eq(i, 0);
michael@0 88 }
michael@0 89 do_check_eq(s.sum, 0);
michael@0 90 if (histogram_type == Telemetry.HISTOGRAM_EXPONENTIAL) {
michael@0 91 do_check_eq(s.log_sum, 0);
michael@0 92 do_check_eq(s.log_sum_squares, 0);
michael@0 93 } else {
michael@0 94 do_check_eq(s.sum_squares_lo, 0);
michael@0 95 do_check_eq(s.sum_squares_hi, 0);
michael@0 96 }
michael@0 97
michael@0 98 h.add(0);
michael@0 99 h.add(1);
michael@0 100 var c = h.snapshot().counts;
michael@0 101 do_check_eq(c[0], 1);
michael@0 102 do_check_eq(c[1], 1);
michael@0 103 }
michael@0 104
michael@0 105 function expect_fail(f) {
michael@0 106 let failed = false;
michael@0 107 try {
michael@0 108 f();
michael@0 109 failed = false;
michael@0 110 } catch (e) {
michael@0 111 failed = true;
michael@0 112 }
michael@0 113 do_check_true(failed);
michael@0 114 }
michael@0 115
michael@0 116 function expect_success(f) {
michael@0 117 let succeeded = false;
michael@0 118 try {
michael@0 119 f();
michael@0 120 succeeded = true;
michael@0 121 } catch (e) {
michael@0 122 succeeded = false;
michael@0 123 }
michael@0 124 do_check_true(succeeded);
michael@0 125 }
michael@0 126
michael@0 127 function test_boolean_histogram()
michael@0 128 {
michael@0 129 var h = Telemetry.newHistogram("test::boolean histogram", "never", 99,1,4, Telemetry.HISTOGRAM_BOOLEAN);
michael@0 130 var r = h.snapshot().ranges;
michael@0 131 // boolean histograms ignore numeric parameters
michael@0 132 do_check_eq(uneval(r), uneval([0, 1, 2]))
michael@0 133 var sum = 0
michael@0 134 for(var i=0;i<r.length;i++) {
michael@0 135 var v = r[i];
michael@0 136 sum += v;
michael@0 137 h.add(v);
michael@0 138 }
michael@0 139 h.add(true);
michael@0 140 h.add(false);
michael@0 141 var s = h.snapshot();
michael@0 142 do_check_eq(s.histogram_type, Telemetry.HISTOGRAM_BOOLEAN);
michael@0 143 // last bucket should always be 0 since .add parameters are normalized to either 0 or 1
michael@0 144 do_check_eq(s.counts[2], 0);
michael@0 145 do_check_eq(s.sum, 3);
michael@0 146 do_check_eq(s.counts[0], 2);
michael@0 147 }
michael@0 148
michael@0 149 function test_flag_histogram()
michael@0 150 {
michael@0 151 var h = Telemetry.newHistogram("test::flag histogram", "never", 130, 4, 5, Telemetry.HISTOGRAM_FLAG);
michael@0 152 var r = h.snapshot().ranges;
michael@0 153 // Flag histograms ignore numeric parameters.
michael@0 154 do_check_eq(uneval(r), uneval([0, 1, 2]))
michael@0 155 // Should already have a 0 counted.
michael@0 156 var c = h.snapshot().counts;
michael@0 157 var s = h.snapshot().sum;
michael@0 158 do_check_eq(uneval(c), uneval([1, 0, 0]));
michael@0 159 do_check_eq(s, 1);
michael@0 160 // Should switch counts.
michael@0 161 h.add(2);
michael@0 162 var c2 = h.snapshot().counts;
michael@0 163 var s2 = h.snapshot().sum;
michael@0 164 do_check_eq(uneval(c2), uneval([0, 1, 0]));
michael@0 165 do_check_eq(s, 1);
michael@0 166 // Should only switch counts once.
michael@0 167 h.add(3);
michael@0 168 var c3 = h.snapshot().counts;
michael@0 169 var s3 = h.snapshot().sum;
michael@0 170 do_check_eq(uneval(c3), uneval([0, 1, 0]));
michael@0 171 do_check_eq(s3, 1);
michael@0 172 do_check_eq(h.snapshot().histogram_type, Telemetry.FLAG_HISTOGRAM);
michael@0 173 }
michael@0 174
michael@0 175 function test_getHistogramById() {
michael@0 176 try {
michael@0 177 Telemetry.getHistogramById("nonexistent");
michael@0 178 do_throw("This can't happen");
michael@0 179 } catch (e) {
michael@0 180
michael@0 181 }
michael@0 182 var h = Telemetry.getHistogramById("CYCLE_COLLECTOR");
michael@0 183 var s = h.snapshot();
michael@0 184 do_check_eq(s.histogram_type, Telemetry.HISTOGRAM_EXPONENTIAL);
michael@0 185 do_check_eq(s.min, 1);
michael@0 186 do_check_eq(s.max, 10000);
michael@0 187 }
michael@0 188
michael@0 189 function compareHistograms(h1, h2) {
michael@0 190 let s1 = h1.snapshot();
michael@0 191 let s2 = h2.snapshot();
michael@0 192
michael@0 193 do_check_eq(s1.histogram_type, s2.histogram_type);
michael@0 194 do_check_eq(s1.min, s2.min);
michael@0 195 do_check_eq(s1.max, s2.max);
michael@0 196 do_check_eq(s1.sum, s2.sum);
michael@0 197 if (s1.histogram_type == Telemetry.HISTOGRAM_EXPONENTIAL) {
michael@0 198 do_check_eq(s1.log_sum, s2.log_sum);
michael@0 199 do_check_eq(s1.log_sum_squares, s2.log_sum_squares);
michael@0 200 } else {
michael@0 201 do_check_eq(s1.sum_squares_lo, s2.sum_squares_lo);
michael@0 202 do_check_eq(s1.sum_squares_hi, s2.sum_squares_hi);
michael@0 203 }
michael@0 204
michael@0 205 do_check_eq(s1.counts.length, s2.counts.length);
michael@0 206 for (let i = 0; i < s1.counts.length; i++)
michael@0 207 do_check_eq(s1.counts[i], s2.counts[i]);
michael@0 208
michael@0 209 do_check_eq(s1.ranges.length, s2.ranges.length);
michael@0 210 for (let i = 0; i < s1.ranges.length; i++)
michael@0 211 do_check_eq(s1.ranges[i], s2.ranges[i]);
michael@0 212 }
michael@0 213
michael@0 214 function test_histogramFrom() {
michael@0 215 // Test one histogram of each type.
michael@0 216 let names = [
michael@0 217 "CYCLE_COLLECTOR", // EXPONENTIAL
michael@0 218 "GC_REASON_2", // LINEAR
michael@0 219 "GC_RESET", // BOOLEAN
michael@0 220 "TELEMETRY_TEST_FLAG" // FLAG
michael@0 221 ];
michael@0 222
michael@0 223 for each (let name in names) {
michael@0 224 let [min, max, bucket_count] = [1, INT_MAX - 1, 10]
michael@0 225 let original = Telemetry.getHistogramById(name);
michael@0 226 let clone = Telemetry.histogramFrom("clone" + name, name);
michael@0 227 compareHistograms(original, clone);
michael@0 228 }
michael@0 229
michael@0 230 // Additionally, set the flag on TELEMETRY_TEST_FLAG, and check it gets set on the clone.
michael@0 231 let testFlag = Telemetry.getHistogramById("TELEMETRY_TEST_FLAG");
michael@0 232 testFlag.add(1);
michael@0 233 let clone = Telemetry.histogramFrom("FlagClone", "TELEMETRY_TEST_FLAG");
michael@0 234 compareHistograms(testFlag, clone);
michael@0 235 }
michael@0 236
michael@0 237 function test_getSlowSQL() {
michael@0 238 var slow = Telemetry.slowSQL;
michael@0 239 do_check_true(("mainThread" in slow) && ("otherThreads" in slow));
michael@0 240 }
michael@0 241
michael@0 242 function test_addons() {
michael@0 243 var addon_id = "testing-addon";
michael@0 244 var fake_addon_id = "fake-addon";
michael@0 245 var name1 = "testing-histogram1";
michael@0 246 var register = Telemetry.registerAddonHistogram;
michael@0 247 expect_success(function ()
michael@0 248 register(addon_id, name1, 1, 5, 6, Telemetry.HISTOGRAM_LINEAR));
michael@0 249 // Can't register the same histogram multiple times.
michael@0 250 expect_fail(function ()
michael@0 251 register(addon_id, name1, 1, 5, 6, Telemetry.HISTOGRAM_LINEAR));
michael@0 252 // Make sure we can't get at it with another name.
michael@0 253 expect_fail(function () Telemetry.getAddonHistogram(fake_addon_id, name1));
michael@0 254
michael@0 255 // Check for reflection capabilities.
michael@0 256 var h1 = Telemetry.getAddonHistogram(addon_id, name1);
michael@0 257 // Verify that although we've created storage for it, we don't reflect it into JS.
michael@0 258 var snapshots = Telemetry.addonHistogramSnapshots;
michael@0 259 do_check_false(name1 in snapshots[addon_id]);
michael@0 260 h1.add(1);
michael@0 261 h1.add(3);
michael@0 262 var s1 = h1.snapshot();
michael@0 263 do_check_eq(s1.histogram_type, Telemetry.HISTOGRAM_LINEAR);
michael@0 264 do_check_eq(s1.min, 1);
michael@0 265 do_check_eq(s1.max, 5);
michael@0 266 do_check_eq(s1.counts[1], 1);
michael@0 267 do_check_eq(s1.counts[3], 1);
michael@0 268
michael@0 269 var name2 = "testing-histogram2";
michael@0 270 expect_success(function ()
michael@0 271 register(addon_id, name2, 2, 4, 4, Telemetry.HISTOGRAM_LINEAR));
michael@0 272
michael@0 273 var h2 = Telemetry.getAddonHistogram(addon_id, name2);
michael@0 274 h2.add(2);
michael@0 275 h2.add(3);
michael@0 276 var s2 = h2.snapshot();
michael@0 277 do_check_eq(s2.histogram_type, Telemetry.HISTOGRAM_LINEAR);
michael@0 278 do_check_eq(s2.min, 2);
michael@0 279 do_check_eq(s2.max, 4);
michael@0 280 do_check_eq(s2.counts[1], 1);
michael@0 281 do_check_eq(s2.counts[2], 1);
michael@0 282
michael@0 283 // Check that we can register histograms for a different addon with
michael@0 284 // identical names.
michael@0 285 var extra_addon = "testing-extra-addon";
michael@0 286 expect_success(function ()
michael@0 287 register(extra_addon, name1, 0, 1, 2, Telemetry.HISTOGRAM_BOOLEAN));
michael@0 288
michael@0 289 // Check that we can register flag histograms.
michael@0 290 var flag_addon = "testing-flag-addon";
michael@0 291 var flag_histogram = "flag-histogram";
michael@0 292 expect_success(function()
michael@0 293 register(flag_addon, flag_histogram, 0, 1, 2, Telemetry.HISTOGRAM_FLAG))
michael@0 294 expect_success(function()
michael@0 295 register(flag_addon, name2, 2, 4, 4, Telemetry.HISTOGRAM_LINEAR));
michael@0 296
michael@0 297 // Check that we reflect registered addons and histograms.
michael@0 298 snapshots = Telemetry.addonHistogramSnapshots;
michael@0 299 do_check_true(addon_id in snapshots)
michael@0 300 do_check_true(extra_addon in snapshots);
michael@0 301 do_check_true(flag_addon in snapshots);
michael@0 302
michael@0 303 // Check that we have data for our created histograms.
michael@0 304 do_check_true(name1 in snapshots[addon_id]);
michael@0 305 do_check_true(name2 in snapshots[addon_id]);
michael@0 306 var s1_alt = snapshots[addon_id][name1];
michael@0 307 var s2_alt = snapshots[addon_id][name2];
michael@0 308 do_check_eq(s1_alt.min, s1.min);
michael@0 309 do_check_eq(s1_alt.max, s1.max);
michael@0 310 do_check_eq(s1_alt.histogram_type, s1.histogram_type);
michael@0 311 do_check_eq(s2_alt.min, s2.min);
michael@0 312 do_check_eq(s2_alt.max, s2.max);
michael@0 313 do_check_eq(s2_alt.histogram_type, s2.histogram_type);
michael@0 314
michael@0 315 // Even though we've registered it, it shouldn't show up until data is added to it.
michael@0 316 do_check_false(name1 in snapshots[extra_addon]);
michael@0 317
michael@0 318 // Flag histograms should show up automagically.
michael@0 319 do_check_true(flag_histogram in snapshots[flag_addon]);
michael@0 320 do_check_false(name2 in snapshots[flag_addon]);
michael@0 321
michael@0 322 // Check that we can remove addon histograms.
michael@0 323 Telemetry.unregisterAddonHistograms(addon_id);
michael@0 324 snapshots = Telemetry.addonHistogramSnapshots;
michael@0 325 do_check_false(addon_id in snapshots);
michael@0 326 // Make sure other addons are unaffected.
michael@0 327 do_check_true(extra_addon in snapshots);
michael@0 328 }
michael@0 329
michael@0 330 // Check that telemetry doesn't record in private mode
michael@0 331 function test_privateMode() {
michael@0 332 var h = Telemetry.newHistogram("test::private_mode_boolean", "never", 1,2,3, Telemetry.HISTOGRAM_BOOLEAN);
michael@0 333 var orig = h.snapshot();
michael@0 334 Telemetry.canRecord = false;
michael@0 335 h.add(1);
michael@0 336 do_check_eq(uneval(orig), uneval(h.snapshot()));
michael@0 337 Telemetry.canRecord = true;
michael@0 338 h.add(1);
michael@0 339 do_check_neq(uneval(orig), uneval(h.snapshot()));
michael@0 340 }
michael@0 341
michael@0 342 // Check that histograms that aren't flagged as needing extended stats
michael@0 343 // don't record extended stats.
michael@0 344 function test_extended_stats() {
michael@0 345 var h = Telemetry.getHistogramById("GRADIENT_DURATION");
michael@0 346 var s = h.snapshot();
michael@0 347 do_check_eq(s.sum, 0);
michael@0 348 do_check_eq(s.log_sum, 0);
michael@0 349 do_check_eq(s.log_sum_squares, 0);
michael@0 350 h.add(1);
michael@0 351 s = h.snapshot();
michael@0 352 do_check_eq(s.sum, 1);
michael@0 353 do_check_eq(s.log_sum, 0);
michael@0 354 do_check_eq(s.log_sum_squares, 0);
michael@0 355 }
michael@0 356
michael@0 357 function generateUUID() {
michael@0 358 let str = Cc["@mozilla.org/uuid-generator;1"].getService(Ci.nsIUUIDGenerator).generateUUID().toString();
michael@0 359 // strip {}
michael@0 360 return str.substring(1, str.length - 1);
michael@0 361 }
michael@0 362
michael@0 363 function run_test()
michael@0 364 {
michael@0 365 let kinds = [Telemetry.HISTOGRAM_EXPONENTIAL, Telemetry.HISTOGRAM_LINEAR]
michael@0 366 for each (let histogram_type in kinds) {
michael@0 367 let [min, max, bucket_count] = [1, INT_MAX - 1, 10]
michael@0 368 test_histogram(histogram_type, "test::"+histogram_type, min, max, bucket_count);
michael@0 369
michael@0 370 const nh = Telemetry.newHistogram;
michael@0 371 expect_fail(function () nh("test::min", "never", 0, max, bucket_count, histogram_type));
michael@0 372 expect_fail(function () nh("test::bucket_count", "never", min, max, 1, histogram_type));
michael@0 373 }
michael@0 374
michael@0 375 // Instantiate the storage for this histogram and make sure it doesn't
michael@0 376 // get reflected into JS, as it has no interesting data in it.
michael@0 377 let h = Telemetry.getHistogramById("NEWTAB_PAGE_PINNED_SITES_COUNT");
michael@0 378 do_check_false("NEWTAB_PAGE_PINNED_SITES_COUNT" in Telemetry.histogramSnapshots);
michael@0 379
michael@0 380 test_boolean_histogram();
michael@0 381 test_getHistogramById();
michael@0 382 test_histogramFrom();
michael@0 383 test_getSlowSQL();
michael@0 384 test_privateMode();
michael@0 385 test_addons();
michael@0 386 test_extended_stats();
michael@0 387 test_expired_histogram();
michael@0 388 }

mercurial