Skip to content
This repository has been archived by the owner on Jan 17, 2023. It is now read-only.

Commit

Permalink
Upload jpeg (#3513)
Browse files Browse the repository at this point in the history
* Start #220, allow JPEG uploads, and respect content-type for JPEG or PNG

* Fix #220, use JPEG for large shots

- Allows JPEGs on the server, both to pass content checks, and to make use of stored content-types (instead of assuming image/png).
- Puts an clip.image.type into shot objects
- Uses .jpg for filenames when appropriate
- Adds a new buildSetting for controlling the cutoff when we use JPEG
- If a PNG image is too large, tries to make a JPEG and substitutes if the JPEG is actually smaller
- Refactor some data:-URL and blob converstion functions into their own module
  • Loading branch information
ianb authored and jaredhirsch committed Sep 19, 2017
1 parent 197b691 commit 82139ed
Show file tree
Hide file tree
Showing 13 changed files with 161 additions and 72 deletions.
6 changes: 2 additions & 4 deletions addon/webextension/background/main.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
/* globals selectorLoader, analytics, communication, catcher, log, makeUuid, auth, senderror, startBackground */
/* globals selectorLoader, analytics, communication, catcher, log, makeUuid, auth, senderror, startBackground, blobConverters */

"use strict";

Expand Down Expand Up @@ -223,9 +223,7 @@ this.main = (function() {
communication.register("downloadShot", (sender, info) => {
// 'data:' urls don't work directly, let's use a Blob
// see http://stackoverflow.com/questions/40269862/save-data-uri-as-file-using-downloads-download-api
const binary = atob(info.url.split(',')[1]); // just the base64 data
const data = Uint8Array.from(binary, char => char.charCodeAt(0))
const blob = new Blob([data], {type: "image/png"})
const blob = blobConverters.dataUrlToBlob(info.url);
let url = URL.createObjectURL(blob);
let downloadId;
let onChangedCallback = catcher.watchFunction(function(change) {
Expand Down
1 change: 1 addition & 0 deletions addon/webextension/background/selectorLoader.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ this.selectorLoader = (function() {
"catcher.js",
"assertIsTrusted.js",
"assertIsBlankDocument.js",
"blobConverters.js",
"background/selectorLoader.js",
"selector/callBackground.js",
"selector/util.js"
Expand Down
1 change: 1 addition & 0 deletions addon/webextension/background/startBackground.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ this.startBackground = (function() {
"log.js",
"makeUuid.js",
"catcher.js",
"blobConverters.js",
"background/selectorLoader.js",
"background/communication.js",
"background/auth.js",
Expand Down
36 changes: 4 additions & 32 deletions addon/webextension/background/takeshot.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
/* globals communication, shot, main, auth, catcher, analytics, buildSettings */
/* globals communication, shot, main, auth, catcher, analytics, buildSettings, blobConverters */

"use strict";

Expand Down Expand Up @@ -33,10 +33,10 @@ this.takeshot = (function() {
}
let convertBlobPromise = Promise.resolve();
if (buildSettings.uploadBinary && !imageBlob) {
imageBlob = base64ToBinary(shot.getClip(shot.clipNames()[0]).image.url);
imageBlob = blobConverters.dataUrlToBlob(shot.getClip(shot.clipNames()[0]).image.url);
shot.getClip(shot.clipNames()[0]).image.url = "";
} else if (!buildSettings.uploadBinary && imageBlob) {
convertBlobPromise = blobToDataUrl(imageBlob).then((dataUrl) => {
convertBlobPromise = blobConverters.blobToDataUrl(imageBlob).then((dataUrl) => {
shot.getClip(shot.clipNames()[0]).image.url = dataUrl;
});
imageBlob = null;
Expand Down Expand Up @@ -120,13 +120,6 @@ this.takeshot = (function() {
}));
}

function base64ToBinary(url) {
const binary = atob(url.split(',')[1]);
const data = Uint8Array.from(binary, char => char.charCodeAt(0));
const blob = new Blob([data], {type: "image/png"});
return blob;
}

/** Combines two buffers or Uint8Array's */
function concatBuffers(buffer1, buffer2) {
var tmp = new Uint8Array(buffer1.byteLength + buffer2.byteLength);
Expand All @@ -135,35 +128,14 @@ this.takeshot = (function() {
return tmp.buffer;
}

/** Returns a promise that converts a Blob to a TypedArray */
function blobToArray(blob) {
return new Promise((resolve, reject) => {
let reader = new FileReader();
reader.addEventListener("loadend", function() {
resolve(reader.result);
});
reader.readAsArrayBuffer(blob);
});
}

function blobToDataUrl(blob) {
return new Promise((resolve, reject) => {
let reader = new FileReader();
reader.addEventListener("loadend", function() {
resolve(reader.result);
});
reader.readAsDataURL(blob);
});
}

/** Creates a multipart TypedArray, given {name: value} fields
and {name: blob} files
Returns {body, "content-type"}
*/
function createMultipart(fields, fileField, fileFilename, blob) {
let boundary = "---------------------------ScreenshotBoundary" + Date.now();
return blobToArray(blob).then((blobAsBuffer) => {
return blobConverters.blobToArray(blob).then((blobAsBuffer) => {
let body = [];
for (let name in fields) {
body.push("--" + boundary);
Expand Down
44 changes: 44 additions & 0 deletions addon/webextension/blobConverters.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
this.blobConverters = (function () {
let exports = {};

exports.dataUrlToBlob = function(url) {
const binary = atob(url.split(',')[1]);
let contentType = exports.getTypeFromDataUrl(url);
if (contentType != "image/png" && contentType != "image/jpeg") {
contentType = "image/png";
}
const data = Uint8Array.from(binary, char => char.charCodeAt(0));
const blob = new Blob([data], {type: contentType});
return blob;
};

exports.getTypeFromDataUrl = function(url) {
let contentType = url.split(',')[0];
contentType = contentType.split(';')[0];
contentType = contentType.split(':')[1];
return contentType;
};

exports.blobToArray = function(blob) {
return new Promise((resolve, reject) => {
let reader = new FileReader();
reader.addEventListener("loadend", function() {
resolve(reader.result);
});
reader.readAsArrayBuffer(blob);
});
};

exports.blobToDataUrl = function(blob) {
return new Promise((resolve, reject) => {
let reader = new FileReader();
reader.addEventListener("loadend", function() {
resolve(reader.result);
});
reader.readAsDataURL(blob);
});
};

return exports;
})();
null;
3 changes: 2 additions & 1 deletion addon/webextension/buildSettings.js.template
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ window.buildSettings = {
defaultSentryDsn: process.env.SCREENSHOTS_SENTRY,
logLevel: process.env.SCREENSHOTS_LOG_LEVEL || "warn",
captureText: (process.env.SCREENSHOTS_CAPTURE_TEXT === "true"),
uploadBinary: (process.env.SCREENSHOTS_UPLOAD_BINARY === "true")
uploadBinary: (process.env.SCREENSHOTS_UPLOAD_BINARY === "true"),
pngToJpegCutoff: parseInt(process.env.SCREENSHOTS_PNG_TO_JPEG_CUTOFF || 2500000, 10)
};
null;
25 changes: 15 additions & 10 deletions addon/webextension/selector/shooter.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/* globals global, documentMetadata, util, uicontrol, ui, catcher */
/* globals buildSettings, domainFromUrl, randomString, shot */
/* globals buildSettings, domainFromUrl, randomString, shot, blobConverters */

"use strict";

Expand Down Expand Up @@ -29,13 +29,6 @@ this.shooter = (function() { // eslint-disable-line no-unused-vars
return result;
}

function base64ToBinary(url) {
const binary = atob(url.split(',')[1]);
const data = Uint8Array.from(binary, char => char.charCodeAt(0));
const blob = new Blob([data], {type: "image/png"});
return blob;
}

catcher.registerHandler((errorObj) => {
callBackground("reportError", sanitizeError(errorObj));
});
Expand Down Expand Up @@ -70,7 +63,16 @@ this.shooter = (function() { // eslint-disable-line no-unused-vars
} finally {
ui.iframe.unhide();
}
return canvas.toDataURL();
let limit = buildSettings.pngToJpegCutoff;
let dataUrl = canvas.toDataURL();
if (limit && dataUrl.length > limit) {
let jpegDataUrl = canvas.toDataURL("image/jpeg");
if (jpegDataUrl.length < dataUrl.length) {
// Only use the JPEG if it is actually smaller
dataUrl = jpegDataUrl;
}
}
return dataUrl;
};

let isSaving = null;
Expand Down Expand Up @@ -106,13 +108,16 @@ this.shooter = (function() { // eslint-disable-line no-unused-vars
captureText = util.captureEnclosedText(selectedPos);
}
let dataUrl = url || screenshotPage(selectedPos, captureType);
let type = blobConverters.getTypeFromDataUrl(dataUrl);
type = type ? type.split("/")[1] : null;
if (dataUrl) {
imageBlob = base64ToBinary(dataUrl);
imageBlob = blobConverters.dataUrlToBlob(dataUrl);
shotObject.delAllClips();
shotObject.addClip({
createdDate: Date.now(),
image: {
url: "data:",
type,
captureType,
text: captureText,
location: selectedPos,
Expand Down
6 changes: 6 additions & 0 deletions bin/load_test_exercise_images.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,12 @@
width=400,
height=267,
),
dict(
source="https://upload.wikimedia.org/wikipedia/en/thumb/a/a9/Example.jpg/111px-Example.jpg",
url="",
width=111,
height=120,
),
dict(
source="http://www.1x1px.me/",
url="",
Expand Down
14 changes: 12 additions & 2 deletions server/src/server.js
Original file line number Diff line number Diff line change
Expand Up @@ -600,7 +600,13 @@ app.put("/data/:id/:domain", upload.single('blob'), function(req, res) {
bodyObj = JSON.parse(req.body.shot);
let clipId = Object.getOwnPropertyNames(bodyObj.clips)[0];
let b64 = req.file.buffer.toString("base64");
b64 = "data:image/png;base64," + b64;
let contentType = req.file.mimetype;
if (contentType != "image/png" && contentType != "image/jpeg") {
// Force PNG as a fallback
mozlog.warn("invalid-upload-content-type", {contentType});
contentType = "image/png";
}
b64 = `data:${contentType};base64,${b64}`;
bodyObj.clips[clipId].image.url = b64;
} else if (req.body) {
bodyObj = req.body;
Expand Down Expand Up @@ -829,7 +835,11 @@ app.get("/images/:imageid", function(req, res) {
el
}).send();
}
res.header("Content-Type", "image/png");
let contentType = obj.contentType;
if (contentType != "image/png" && contentType != "image/jpeg") {
contentType = "image/png";
}
res.header("Content-Type", contentType);
if (download) {
if (dbschema.getKeygrip().verify(new Buffer(download, 'utf8'), sig)) {
res.header("Content-Disposition", contentDisposition(download));
Expand Down
Loading

0 comments on commit 82139ed

Please sign in to comment.