mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
56aad8a83e
- Bug 1149987 - Part 2: Make ErrorResult unassignable; r=bzbarsky (32661559b) - Bug 1149987 - Part 3: Give ErrorResult a move constructor and a move assignment operator; (27f4c6125) - Bug 1149987 - Part 4: Do not attempt to delete ErrorResult::mMessage when deserializing the object from IPDL; r=bzbarsky (0f9dcc603) - Bug 1110485 P0 Add an ErrorResult constructor that takes nsresult. (72a779666) - Bug 1110485 P1 Refactor Cache IPC requests to use a separate actor. (a7e4c1959) - Bug 1127914 - Part 1 - Duplicate keyed histograms for double submission. (78673277f) - Bug 1127914 - Part 2 - Duplicate normal histograms for double submission. (55c302057) - Bug 1127914 - Part 3 - Submit duplicate histogram data for 'non-classic' telemetry sessions. r=vladan (bb3e49c43) - Bug 1120362 - Part 1 - Enable snapshotting and clearing subsession histograms. (14378a6e5) - Bug 1120362 - Part 2 - Enable snapshotting and clearing keyed subsession histograms. r=vladan (c0e0bfb3e) - partial apply of Bug 1119281 - Fix missing telemetry client id (ae0dc0194) - Bug 1122047 - Part 1 - Sketch out Telemetry environment module. (0419391b0) - Bug 1122047 - Part 2 - Make TelemetryPing shutdown properly on delayed initialization (0102cef09) - Bug 1122061 - Give TelemetryPing a common API for sending pings. (999cb825d) - Bug 1122061 - Move TelemetrySession tests out of test_telemetryPing.js. (2d5b61de1) - Bug 1120362 - Part 3 - Reset subsession histograms on telemetry payload collections. r=vladan (0d3f04df1) - Bug 1120362 - Part 4 - Start new telemetry subsessions on local midnight. r=vladan (93eb9ca21) - Bug 1120363 - Break up Telemetry sessions on environment changes. (a7c8d70c7) - Bug 1122052 - Remove duplicated data from TelemetrySession. (bb905d602) - Bug 1122050 - Remove persona and experiment data from TelemetrySession. (40ca59a9e) - Bug 1134268 - Part 1 - Fix and order Telemetry shutdown for TelemetryPing and TelemetrySession. r=yoric (30d0f0656) - Bug 1134268 - Part 2 - Fixup TelemetryEnvironment shutdown if the module wasnt initialized. r=vladan (ec2875fea) - Bug 1135076 - Missing histograms in childPayloads. r=vladan (9f317cf9d) - Bug 1134279 - Make TelemetryPing and TelemetrySession code use the "FHR enabled" & "Telemetry enabled" prefs properly. r=vladan (4050d7f24) - Bug 1128768: Part 1 - Modify IPC to allow retrieval of topmost routing id on the stack; (cd2e8a2f0) - Bug 1129249 - Add a "restyle" feature to profiler and split the style label in Cleopatra based on the restyleSource, r=dholbert,mstange (b37df94d1) - Bug 1150684: Remove XPCOM.h from IOInterposer.h (5b7e1cef3) - Bug 1093934 - Create a XPCOM library that can be used to support standalone WebRTC. (9ec8a819f) - Merge branch 'master' of https://github.com/rmottola/Arctic-Fox (d0f05eea4) - Bug 1128768: Part 2,3,4 - Refactor hang annotation code; (f5086aba9) (with xpcom/threads/ fixes for my tele-removed tree) - Bug 1128768: Part 5 - Update plugin code to retrieve SWF file for hang annotations; (774a47aec) - Bug 1110485 P2 Remove 'P' prefix from non-protocol IPC types in Cache API. r=baku (ea29a10cf) - Bug 1110485 P3 Move Fetch IPC PHeaderEntry type to Cache. Rename HeadesEntry. (9eba0aca0) - Bug 1110485 P4 Keep Cache Actors alive during async operations. (eb75f2316) - Bug 1110485 P5 Replace useless DBSchema class type with namespace. (159b902db) - Bug 1110485 P6 Remove useless cache::FileUtils type (1bdf00fc3) - Bug 1110485 P7 Rename DeleteCache() to DeleteCacheId() better distinguish it from CacheDelete(). (5199f9d6f) - Bug 1110485 P8 Correctly set the Feature on the stream control child actor. (c8673cb13) - Bug 1150691 Fix Cache API race with storage invalidation. (2723dff50) - Bug 1151892 Refactor Cache Manager Context usage to be more sane and fix shutdown assert. r=ehsan (ea96381cf) - Bug 1136331 - OdinMonkey: allow stdlib calls in heap expressions (2fc5e2bfd) - Bug 1141439 - Exit with an error code instead of falling through the REMOTE_NOT_FOUND code path when the X-remote returns an explicit command line handler error. (afcf9b1aa) - Bug 1135825: Add missing MOZ_OVERRIDE annotation in RTCIdentityProviderRegistrar.h (e8beec4e8) - (Bug 1135138 is not merged due to broken build)
304 lines
8.1 KiB
JavaScript
304 lines
8.1 KiB
JavaScript
/* -*- js-indent-level: 2; indent-tabs-mode: nil -*- */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
"use strict";
|
|
|
|
this.EXPORTED_SYMBOLS = ["TelemetryFile"];
|
|
|
|
const Cc = Components.classes;
|
|
const Ci = Components.interfaces;
|
|
const Cr = Components.results;
|
|
const Cu = Components.utils;
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm", this);
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm", this);
|
|
Cu.import("resource://gre/modules/osfile.jsm", this);
|
|
Cu.import("resource://gre/modules/Task.jsm", this);
|
|
Cu.import("resource://gre/modules/Promise.jsm", this);
|
|
|
|
XPCOMUtils.defineLazyModuleGetter(this, 'Deprecated',
|
|
'resource://gre/modules/Deprecated.jsm');
|
|
|
|
const Telemetry = Services.telemetry;
|
|
|
|
// Files that have been lying around for longer than MAX_PING_FILE_AGE are
|
|
// deleted without being loaded.
|
|
const MAX_PING_FILE_AGE = 14 * 24 * 60 * 60 * 1000; // 2 weeks
|
|
|
|
// Files that are older than OVERDUE_PING_FILE_AGE, but younger than
|
|
// MAX_PING_FILE_AGE indicate that we need to send all of our pings ASAP.
|
|
const OVERDUE_PING_FILE_AGE = 7 * 24 * 60 * 60 * 1000; // 1 week
|
|
|
|
// Maximum number of pings to save.
|
|
const MAX_LRU_PINGS = 17;
|
|
|
|
// The number of outstanding saved pings that we have issued loading
|
|
// requests for.
|
|
let pingsLoaded = 0;
|
|
|
|
// The number of pings that we have destroyed due to being older
|
|
// than MAX_PING_FILE_AGE.
|
|
let pingsDiscarded = 0;
|
|
|
|
// The number of pings that are older than OVERDUE_PING_FILE_AGE
|
|
// but younger than MAX_PING_FILE_AGE.
|
|
let pingsOverdue = 0;
|
|
|
|
// Data that has neither been saved nor sent by ping
|
|
let pendingPings = [];
|
|
|
|
let isPingDirectoryCreated = false;
|
|
|
|
this.TelemetryFile = {
|
|
|
|
get MAX_PING_FILE_AGE() {
|
|
return MAX_PING_FILE_AGE;
|
|
},
|
|
|
|
get OVERDUE_PING_FILE_AGE() {
|
|
return OVERDUE_PING_FILE_AGE;
|
|
},
|
|
|
|
get MAX_LRU_PINGS() {
|
|
return MAX_LRU_PINGS;
|
|
},
|
|
|
|
get pingDirectoryPath() {
|
|
return OS.Path.join(OS.Constants.Path.profileDir, "saved-telemetry-pings");
|
|
},
|
|
|
|
/**
|
|
* Save a single ping to a file.
|
|
*
|
|
* @param {object} ping The content of the ping to save.
|
|
* @param {string} file The destination file.
|
|
* @param {bool} overwrite If |true|, the file will be overwritten if it exists,
|
|
* if |false| the file will not be overwritten and no error will be reported if
|
|
* the file exists.
|
|
* @returns {promise}
|
|
*/
|
|
savePingToFile: function(ping, file, overwrite) {
|
|
return Task.spawn(function*() {
|
|
try {
|
|
let pingString = JSON.stringify(ping);
|
|
yield OS.File.writeAtomic(file, pingString, {tmpPath: file + ".tmp",
|
|
noOverwrite: !overwrite});
|
|
} catch(e if e.becauseExists) {
|
|
}
|
|
})
|
|
},
|
|
|
|
/**
|
|
* Save a ping to its file.
|
|
*
|
|
* @param {object} ping The content of the ping to save.
|
|
* @param {bool} overwrite If |true|, the file will be overwritten
|
|
* if it exists.
|
|
* @returns {promise}
|
|
*/
|
|
savePing: function(ping, overwrite) {
|
|
return Task.spawn(function*() {
|
|
yield getPingDirectory();
|
|
let file = pingFilePath(ping);
|
|
yield this.savePingToFile(ping, file, overwrite);
|
|
}.bind(this));
|
|
},
|
|
|
|
/**
|
|
* Save all pending pings.
|
|
*
|
|
* @param {object} sessionPing The additional session ping.
|
|
* @returns {promise}
|
|
*/
|
|
savePendingPings: function(sessionPing) {
|
|
let p = pendingPings.reduce((p, ping) => {
|
|
// Restore the files with the previous pings if for some reason they have
|
|
// been deleted, don't overwrite them otherwise.
|
|
p.push(this.savePing(ping, false));
|
|
return p;}, [this.savePing(sessionPing, true)]);
|
|
|
|
pendingPings = [];
|
|
return Promise.all(p);
|
|
},
|
|
|
|
/**
|
|
* Remove the file for a ping
|
|
*
|
|
* @param {object} ping The ping.
|
|
* @returns {promise}
|
|
*/
|
|
cleanupPingFile: function(ping) {
|
|
return OS.File.remove(pingFilePath(ping));
|
|
},
|
|
|
|
/**
|
|
* Load all saved pings.
|
|
*
|
|
* Once loaded, the saved pings can be accessed (destructively only)
|
|
* through |popPendingPings|.
|
|
*
|
|
* @returns {promise}
|
|
*/
|
|
loadSavedPings: function() {
|
|
return Task.spawn(function*() {
|
|
let directory = TelemetryFile.pingDirectoryPath;
|
|
let iter = new OS.File.DirectoryIterator(directory);
|
|
let exists = yield iter.exists();
|
|
|
|
if (exists) {
|
|
let entries = yield iter.nextBatch();
|
|
let sortedEntries = [];
|
|
|
|
for (let entry of entries) {
|
|
if (entry.isDir) {
|
|
continue;
|
|
}
|
|
|
|
let info = yield OS.File.stat(entry.path);
|
|
sortedEntries.push({entry:entry, lastModificationDate: info.lastModificationDate});
|
|
}
|
|
|
|
sortedEntries.sort(function compare(a, b) {
|
|
return b.lastModificationDate - a.lastModificationDate;
|
|
});
|
|
|
|
let count = 0;
|
|
let result = [];
|
|
|
|
// Keep only the last MAX_LRU_PINGS entries to avoid that the backlog overgrows.
|
|
for (let i = 0; i < MAX_LRU_PINGS && i < sortedEntries.length; i++) {
|
|
let entry = sortedEntries[i].entry;
|
|
result.push(this.loadHistograms(entry.path))
|
|
}
|
|
|
|
for (let i = MAX_LRU_PINGS; i < sortedEntries.length; i++) {
|
|
let entry = sortedEntries[i].entry;
|
|
OS.File.remove(entry.path);
|
|
}
|
|
|
|
yield Promise.all(result);
|
|
|
|
Services.telemetry.getHistogramById('TELEMETRY_FILES_EVICTED').
|
|
add(sortedEntries.length - MAX_LRU_PINGS);
|
|
}
|
|
|
|
yield iter.close();
|
|
}.bind(this));
|
|
},
|
|
|
|
/**
|
|
* Load the histograms from a file.
|
|
*
|
|
* Once loaded, the saved pings can be accessed (destructively only)
|
|
* through |popPendingPings|.
|
|
*
|
|
* @param {string} file The file to load.
|
|
* @returns {promise}
|
|
*/
|
|
loadHistograms: function loadHistograms(file) {
|
|
return OS.File.stat(file).then(function(info){
|
|
let now = Date.now();
|
|
if (now - info.lastModificationDate > MAX_PING_FILE_AGE) {
|
|
// We haven't had much luck in sending this file; delete it.
|
|
pingsDiscarded++;
|
|
return OS.File.remove(file);
|
|
}
|
|
|
|
// This file is a bit stale, and overdue for sending.
|
|
if (now - info.lastModificationDate > OVERDUE_PING_FILE_AGE) {
|
|
pingsOverdue++;
|
|
}
|
|
|
|
pingsLoaded++;
|
|
return addToPendingPings(file);
|
|
});
|
|
},
|
|
|
|
/**
|
|
* The number of pings loaded since the beginning of time.
|
|
*/
|
|
get pingsLoaded() {
|
|
return pingsLoaded;
|
|
},
|
|
|
|
/**
|
|
* The number of pings loaded that are older than OVERDUE_PING_FILE_AGE
|
|
* but younger than MAX_PING_FILE_AGE.
|
|
*/
|
|
get pingsOverdue() {
|
|
return pingsOverdue;
|
|
},
|
|
|
|
/**
|
|
* The number of pings that we just tossed out for being older than
|
|
* MAX_PING_FILE_AGE.
|
|
*/
|
|
get pingsDiscarded() {
|
|
return pingsDiscarded;
|
|
},
|
|
|
|
/**
|
|
* Iterate destructively through the pending pings.
|
|
*
|
|
* @return {iterator}
|
|
*/
|
|
popPendingPings: function*() {
|
|
while (pendingPings.length > 0) {
|
|
let data = pendingPings.pop();
|
|
yield data;
|
|
}
|
|
},
|
|
|
|
testLoadHistograms: function(file) {
|
|
pingsLoaded = 0;
|
|
return this.loadHistograms(file.path);
|
|
}
|
|
};
|
|
|
|
///// Utility functions
|
|
function pingFilePath(ping) {
|
|
return OS.Path.join(TelemetryFile.pingDirectoryPath, ping.id);
|
|
}
|
|
|
|
function getPingDirectory() {
|
|
return Task.spawn(function*() {
|
|
let directory = TelemetryFile.pingDirectoryPath;
|
|
|
|
if (!isPingDirectoryCreated) {
|
|
yield OS.File.makeDir(directory, { unixMode: OS.Constants.S_IRWXU });
|
|
isPingDirectoryCreated = true;
|
|
}
|
|
|
|
return directory;
|
|
});
|
|
}
|
|
|
|
function addToPendingPings(file) {
|
|
function onLoad(success) {
|
|
let success_histogram = Telemetry.getHistogramById("READ_SAVED_PING_SUCCESS");
|
|
success_histogram.add(success);
|
|
}
|
|
|
|
return Task.spawn(function*() {
|
|
try {
|
|
let array = yield OS.File.read(file);
|
|
let decoder = new TextDecoder();
|
|
let string = decoder.decode(array);
|
|
|
|
let ping = JSON.parse(string);
|
|
// The ping's payload used to be stringified JSON. Deal with that.
|
|
if (typeof(ping.payload) == "string") {
|
|
ping.payload = JSON.parse(ping.payload);
|
|
}
|
|
|
|
pendingPings.push(ping);
|
|
onLoad(true);
|
|
} catch (e) {
|
|
onLoad(false);
|
|
yield OS.File.remove(file);
|
|
}
|
|
});
|
|
}
|