Skip to content

Commit

Permalink
Merge pull request #1876 from mozilla/sendbeacon_fallback
Browse files Browse the repository at this point in the history
Bug 1878140 - Make sendBeacon fallback the default uploader.
  • Loading branch information
rosahbruno authored Feb 15, 2024
2 parents 9b87d37 + aaf3ddc commit 702c810
Show file tree
Hide file tree
Showing 6 changed files with 51 additions and 30 deletions.
3 changes: 2 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,8 @@

[Full changelog](https://github.com/mozilla/glean.js/compare/v4.0.0...main)

* [#1866](https://github.com/mozilla/glean.js/pull/1866): Added a new uploader that falls back to `fetch` if `navigator.sendBeacon` fails. This is not used by default and needs to be enabled manually.
* [#1866](https://github.com/mozilla/glean.js/pull/1866): Added a new uploader that falls back to `fetch` if `navigator.sendBeacon` fails.
* [#1876](https://github.com/mozilla/glean.js/pull/1876): **BREAKING CHANGE**: `navigator.sendBeacon` with fallback to `fetch` (see #1866) is now the default uploader. This can be changed manually.

# v4.0.0 (2024-01-24)

Expand Down
15 changes: 1 addition & 14 deletions glean/src/core/upload/worker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -70,22 +70,9 @@ class PingUploadWorker {
try {
const finalPing = this.buildPingRequest(ping);

let safeUploader = this.uploader;
if (!this.uploader.supportsCustomHeaders()) {
// Some options require us to submit custom headers. Unfortunately not all the
// uploaders support them (e.g. `sendBeacon`). In case headers are required, switch
// back to the default uploader that, for now, supports headers.
const needsHeaders = !(
(Context.config.sourceTags === undefined) && (Context.config.debugViewTag === undefined)
);
if (needsHeaders) {
safeUploader = Context.platform.uploader;
}
}

// The POST call has to be asynchronous. Once the API call is triggered,
// we rely on the browser's "keepalive" header.
return safeUploader.post(
return this.uploader.post(
`${this.serverEndpoint}${ping.path}`,
finalPing
);
Expand Down
23 changes: 17 additions & 6 deletions glean/src/platform/browser/sendbeacon_fallback_uploader.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import BrowserFetchUploader from "./fetch_uploader.js";
import BrowserSendBeaconUploader from "./sendbeacon_uploader.js";
import { UploadResultStatus } from "../../core/upload/uploader.js";
import type { UploadResult } from "../../core/upload/uploader.js";
import { Context } from "../../core/context.js";

const LOG_TAG = "platform.browser.SendBeaconFallbackUploader";

Expand All @@ -23,13 +24,23 @@ class BrowserSendBeaconFallbackUploader extends Uploader {
pingRequest: PingRequest<string | Uint8Array>
): Promise<UploadResult> {

// Try `sendBeacon` first,
// fall back to `fetch` if `sendBeacon` reports an error.
const beaconStatus = await this.sendBeaconUploader.post(url, pingRequest, false);
if (beaconStatus.result == UploadResultStatus.Success) {
return beaconStatus;
// Some options require us to submit custom headers. Unfortunately not all the
// uploaders support them (e.g. `sendBeacon`). In case headers are required, switch
// back to the `fetch` uploader that supports headers.
// Then try `sendBeacon` first,
// fall back to `fetch` if `sendBeacon` reports an error or `sendBeacon` is
// not defined.
const hasNoCustomHeaders = !Context.config?.sourceTags && !Context.config?.debugViewTag;

if (hasNoCustomHeaders && !!navigator && !!navigator.sendBeacon) {
const beaconStatus = await this.sendBeaconUploader.post(url, pingRequest, false);
if (beaconStatus.result == UploadResultStatus.Success) {
return beaconStatus;
}
log(LOG_TAG, "The `sendBeacon` call was not serviced by the browser. Falling back to the `fetch` uploader.", LoggingLevel.Warn);
} else {
log(LOG_TAG, "`sendBeacon` is not available. Falling back to the `fetch` uploader.", LoggingLevel.Warn);
}
log(LOG_TAG, "The `sendBeacon` call was not serviced by the browser. Falling back to the fetch uploader.", LoggingLevel.Warn);

return this.fetchUploader.post(url, pingRequest);
}
Expand Down
2 changes: 1 addition & 1 deletion glean/src/platform/browser/web/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

import type Platform from "../../index.js";

import uploader from "../fetch_uploader.js";
import uploader from "../sendbeacon_fallback_uploader.js";
import info from "./platform_info.js";
import Storage from "./storage.js";

Expand Down
7 changes: 6 additions & 1 deletion glean/src/platform/browser/web/platform_info.ts
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,12 @@ const BrowserPlatformInfo: PlatformInfo = {
},

locale(): string {
return navigator.language || "und";
if (!!navigator && !!navigator.language) {
return navigator.language;
}
else {
return "und";
}
}
};

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,17 +33,17 @@ function setGlobalSendBeacon() {
// @ts-ignore
global.fetch = fetch;

describe("Uploader/BrowserSendBeaconFallback", function () {
describe("Uploader/BrowserSendBeaconFallback", function() {
beforeEach(function() {
setGlobalSendBeacon();
});

afterEach(function () {
afterEach(function() {
sandbox.restore();
});

it("returns the correct status for successful requests", async function () {
const TEST_PING_CONTENT = {"my-test-value": 40721};
it("returns the correct status for successful requests", async function() {
const TEST_PING_CONTENT = { "my-test-value": 40721 };
for (const status of [200, 400, 500]) {
nock(MOCK_ENDPOINT).post(/./i, body => {
return JSON.stringify(body) == JSON.stringify(TEST_PING_CONTENT);
Expand All @@ -60,8 +60,8 @@ describe("Uploader/BrowserSendBeaconFallback", function () {
}
});

it("returns the correct status after fallback", async function () {
const TEST_PING_CONTENT = {"my-test-value": 40721};
it("returns the correct status after fallback", async function() {
const TEST_PING_CONTENT = { "my-test-value": 40721 };
nock(MOCK_ENDPOINT).post(/./i).reply(200);

// Reset `fetch` to a known state.
Expand All @@ -80,7 +80,7 @@ describe("Uploader/BrowserSendBeaconFallback", function () {
);
});

it("returns the correct status when both uploads fail", async function () {
it("returns the correct status when both uploads fail", async function() {
nock(MOCK_ENDPOINT).post(/./i).replyWithError({
message: "something awful happened",
code: "AWFUL_ERROR",
Expand All @@ -101,4 +101,21 @@ describe("Uploader/BrowserSendBeaconFallback", function () {
expectedResponse
);
});

it("still sends even when navigator has been unset", async function() {
const TEST_PING_CONTENT = { "my-test-value": 40721 };
nock(MOCK_ENDPOINT).post(/./i).reply(200);

// Unset Navigator, to emulate add-on behavior.
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
global.navigator.sendBeacon = undefined;

const response = BrowserSendBeaconFallbackUploader.post(MOCK_ENDPOINT, new PingRequest("abc", {}, JSON.stringify(TEST_PING_CONTENT), 1024));
const expectedResponse = new UploadResult(UploadResultStatus.Success, 200);
assert.deepStrictEqual(
await response,
expectedResponse
);
});
});

0 comments on commit 702c810

Please sign in to comment.