diff --git a/tests/test_uploads.py b/tests/test_uploads.py index 9522ebc1e..9b3696fa8 100644 --- a/tests/test_uploads.py +++ b/tests/test_uploads.py @@ -79,15 +79,20 @@ def testGet(self): rv = self.test_client.get('/upload/') self.assertEqual(rv.status_code, 404) - def testBadFileUploads(self): + def testRegularUploads(self): # Loop over some files and the status codes that we are expecting + # Basically it should never be possible to upload a "regular" file. for filename, status_code in ( - ('evil.py', 415), - ('evil', 415), - ('evil.png', 415), - ('green_square.webp', 415)): - - # The reason why we are definisng it in here and not outside + ('evil.py', 501), + ('evil', 501), + ('evil.png', 501), + ('green_square.webp', 501), + ('green_square.png', 501), + ('green_square.jpg', 501), + ('green_square.gif', 501), + ('green_square.bmp', 501)): + + # The reason why we are defining it in here and not outside # this method is that we are setting the filename of the # TestingFileStorage to be the one in the for loop. This way # we can ensure that the filename that we are "uploading" @@ -110,32 +115,6 @@ def files(self): rv = test_client.post('/upload/', data=dict()) self.assertEqual(rv.status_code, status_code) - def testGoodFileUploads(self): - # Loop over some files and the URLs that we are expecting back - for filename, status_code in ( - ('green_square.png', 201), - ('green_square.jpg', 201), - ('green_square.gif', 201), - ('green_square.bmp', 201)): - - class TestingRequest(Request): - """A testing request to use that will return a - TestingFileStorage to test the uploading.""" - @property - def files(self): - d = MultiDict() - f = open(os.path.join('tests', 'fixtures', filename), 'r') - d['image'] = TestingFileStorage(stream=StringIO(f.read()), - filename=filename) - f.close() - return d - - self.app.request_class = TestingRequest - test_client = self.app.test_client() - - rv = test_client.post('/upload/', data=dict()) - self.assertEqual(rv.status_code, status_code) - def testBase64ScreenshotUploads(self): BASE64_PNG = u'' # nopep8 BASE64_PNG_GARBAGE = u'!' @@ -155,7 +134,7 @@ class TestingRequest(Request): @property def form(self): d = MultiDict() - d['screenshot'] = filedata + d['image'] = filedata return d self.app.request_class = TestingRequest diff --git a/webcompat/__init__.py b/webcompat/__init__.py index beb74a5c5..107974e16 100644 --- a/webcompat/__init__.py +++ b/webcompat/__init__.py @@ -15,8 +15,10 @@ app = Flask(__name__, static_url_path='') app.config.from_object('config') -# set limit of 4MB for file uploads -app.config['MAX_CONTENT_LENGTH'] = 4 * 1024 * 1024 +# set limit of 5.5MB for file uploads +# in practice, this is ~4MB (5.5 / 1.37) +# after the data URI is saved to disk +app.config['MAX_CONTENT_LENGTH'] = 5.5 * 1024 * 1024 github = GitHub(app) limiter = Limiter(app) diff --git a/webcompat/api/uploads.py b/webcompat/api/uploads.py index e8d240996..2a28c734a 100644 --- a/webcompat/api/uploads.py +++ b/webcompat/api/uploads.py @@ -17,7 +17,6 @@ from flask import request from io import BytesIO from PIL import Image -from werkzeug.datastructures import FileStorage from werkzeug.exceptions import RequestEntityTooLarge from uuid import uuid4 @@ -44,10 +43,7 @@ def __init__(self, imagedata): def to_image_object(self, imagedata): '''Method to return a Pillow Image object from the raw imagedata.''' try: - # Is this a file uploaded via ? - if isinstance(imagedata, FileStorage): - return Image.open(imagedata) - # Is this base64 encoded screenshot? + # Make sure we're being sent a base64 encoded image if (isinstance(imagedata, unicode) and imagedata.startswith('data:image/')): # Chop off 'data:image/.+;base64,' before decoding @@ -59,7 +55,12 @@ def to_image_object(self, imagedata): abort(415) def get_file_ext(self): - '''Method to return the file extension, as determined by Pillow.''' + '''Method to return the file extension, as determined by Pillow. + + (But, we return jpg for png images, since we convert them always.) + ''' + if self.image_object.format.lower() == 'png': + return 'jpg' return self.image_object.format.lower() def get_filename(self): @@ -105,10 +106,8 @@ def upload(): Returns a JSON string that contains the filename and url. ''' - if 'image' in request.files and request.files['image'].filename: - imagedata = request.files['image'] - elif 'screenshot' in request.form: - imagedata = request.form['screenshot'] + if 'image' in request.form: + imagedata = request.form['image'] else: # We don't know what you're trying to do, but it ain't gonna work. abort(501) diff --git a/webcompat/static/css/development/components/loader.css b/webcompat/static/css/development/components/loader.css index 15a479f80..aeeb85dee 100644 --- a/webcompat/static/css/development/components/loader.css +++ b/webcompat/static/css/development/components/loader.css @@ -15,3 +15,15 @@ display:block; background-color:rgba(255,255,255,.8); } + +.wc-Upload-Loader { + display:none; + position:absolute; + right:4%;top: 27%; + width: 24px; + height: 24px; +} + +.wc-Upload-Loader.is-active { + display: block; +} \ No newline at end of file diff --git a/webcompat/static/img/upload-loader.svg b/webcompat/static/img/upload-loader.svg new file mode 100644 index 000000000..196091fc8 --- /dev/null +++ b/webcompat/static/img/upload-loader.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/webcompat/static/js/lib/bugform.js b/webcompat/static/js/lib/bugform.js index 5e6929835..2e1d1e29c 100644 --- a/webcompat/static/js/lib/bugform.js +++ b/webcompat/static/js/lib/bugform.js @@ -12,6 +12,7 @@ function BugForm() { this.loadingIndicator = $('.js-Loader'); this.reportButton = $('#js-ReportBug'); this.loaderImage = $('.js-Loader'); + this.uploadLoader = $('.js-Upload-Loader'); this.screenshotData = ''; // by default, submission type is anonymous this.submitType = 'github-proxy-report'; @@ -65,13 +66,9 @@ function BugForm() { // Set up listener for message events from screenshot-enabled add-ons window.addEventListener('message', _.bind(function(event) { - var img = document.createElement('img'); - var canvas = document.createElement('canvas'); - var ctx = canvas.getContext('2d'); // Make sure the data is coming from ~*inside the house*~! // (i.e., our add-on sent it) if (location.origin === event.origin) { - this.form.on('submit', _.bind(this.submitFormWithScreenshot, this)); this.screenshotData = event.data; // The final size of Base64-encoded binary data is ~equal to @@ -79,62 +76,44 @@ function BugForm() { // so, bytes = (encoded_string.length - 814) / 1.37) // see https://en.wikipedia.org/wiki/Base64#MIME if ((String(this.screenshotData).length - 814 / 1.37) > this.UPLOAD_LIMIT) { - img.onload = _.bind(function() { - // scale the tmp canvas to 50% - canvas.width = Math.floor(img.width / 2); - canvas.height = Math.floor(img.height / 2); - ctx.scale(0.5, 0.5); - // draw back in the screenshot (at 50% scale) - // and re-serialize to data URI - ctx.drawImage(img, 0, 0); - this.screenshotData = canvas.toDataURL(); - img = null, canvas = null; - - this.addPreviewBackground(this.screenshotData); - }, this); - - img.src = this.screenshotData; + this.downsampleImageAndUpload(this.screenshotData); } else { - this.addPreviewBackground(this.screenshotData); + this.addPreviewBackgroundAndUpload(this.screenshotData); } } }, this), false); }; - // If we've gotten this far, form validation has happened and - // we're reacting to a submit event. - this.submitFormWithScreenshot = function(event) { - // Stop regular form submission - event.preventDefault(); - // construct a FormData object and append the base64 image to it. - var formdata = new FormData(this.form.get(0)); - // we call this 'screenshot', rather than 'image' because it needs to be - // handled differently on the backend. - formdata.append('screenshot', this.screenshotData); - // add in the submit-type, which won't be included in JS form submission by default - formdata.append('submit-type', this.submitType); - - this.loaderImage.show(); - $.ajax({ - contentType: false, - processData: false, - data: formdata, - method: 'POST', - url: '/issues/new', - success: _.bind(function(response) { - // navigate to thanks page. - this.loaderImage.hide(); - window.location.href = '/thanks/' + response.number; - }, this), - error: _.bind(function(response) { - var msg = 'There was an error trying to file the bug, try again?.'; - if (response && response.status === 413) { - msg = 'The image is too big! Please choose something smaller than 4MB.'; - } - wcEvents.trigger('flash:error', {message: msg, timeout: 5000}); - this.loaderImage.hide(); - }, this) - }); + this.downsampleImageAndUpload = function(dataURI) { + var img = document.createElement('img'); + var canvas = document.createElement('canvas'); + var ctx = canvas.getContext('2d'); + + img.onload = _.bind(function() { + // scale the tmp canvas to 50% + canvas.width = Math.floor(img.width / 2); + canvas.height = Math.floor(img.height / 2); + ctx.scale(0.5, 0.5); + // draw back in the screenshot (at 50% scale) + // and re-serialize to data URI + ctx.drawImage(img, 0, 0); + // Note: this will convert GIFs to JPEG, which breaks + // animated GIFs. However, this only will happen if they + // were above the upload limit size. So... sorry? + this.screenshotData = canvas.toDataURL('image/jpeg', 0.8); + + // The limit is 4MB (which is crazy big!), so let the user know if their + // file is unreasonably large at this point (after 1 round of downsampling) + if (this.screenshotData > this.UPLOAD_LIMIT) { + this.makeInvalid('img_too_big'); + return; + } + + this.addPreviewBackgroundAndUpload(this.screenshotData); + img = null, canvas = null; + }, this); + + img.src = dataURI; }; this.checkParams = function() { @@ -298,14 +277,6 @@ function BugForm() { // We can just grab the 0th one, because we only allow uploading // a single image at a time (for now) var img = event.target.files[0]; - // The limit is 4MB (which is crazy big!), so let the user know if their - // file is unreasonably large. - if (img.size > this.UPLOAD_LIMIT) { - this.makeInvalid('img_too_big'); - return; - } else if (img.size < this.UPLOAD_LIMIT) { - this.makeValid('img_too_big'); - } // One last image type validation check. if (!img.type.match('image.*')) { @@ -315,12 +286,17 @@ function BugForm() { var reader = new FileReader(); reader.onload = _.bind(function(event) { - this.addPreviewBackground(event.target.result); + var dataURI = event.target.result; + if ((String(dataURI).length - 814 / 1.37) > this.UPLOAD_LIMIT) { + this.downsampleImageAndUpload(dataURI); + } else { + this.addPreviewBackgroundAndUpload(dataURI); + } }, this); reader.readAsDataURL(img); }; - this.addPreviewBackground = function(dataURI) { + this.addPreviewBackgroundAndUpload = function(dataURI) { if (!_.startsWith(dataURI, 'data:image/')) { return; } @@ -332,6 +308,7 @@ function BugForm() { }); this.showRemoveUpload(label); + this.getUploadURL(dataURI); }; /* Allow users to remove an image from the form upload. @@ -347,16 +324,69 @@ function BugForm() { removeBanner.removeClass('is-hidden'); uploadWrapper.addClass('is-hidden'); removeBanner.on('click', _.bind(function() { - // clear out the input value or screenshot data - this.uploadField.val(this.uploadField.get(0).defaultValue); - this.screenshotData = ''; // remove the preview and hide the banner label.css('background', 'none'); removeBanner.addClass('is-hidden'); uploadWrapper.removeClass('is-hidden'); removeBanner.off('click'); + + // remove the last embedded image URL + // Note: this could fail in weird ways depending on how + // the user has edited the descField. + this.descField.val(function(idx, value) { + return value.replace(/!\[.+\.jpe*g\)$/, ''); + }); }, this)); }; + /* + Upload the image before form submission so we can + put an image link in the bug description. + */ + this.getUploadURL = function(dataURI) { + this.disableSubmits(); + this.uploadLoader.addClass('is-active'); + var formdata = new FormData(); + formdata.append('image', dataURI); + + $.ajax({ + contentType: false, + processData: false, + data: formdata, + method: 'POST', + url: '/upload/', + success: _.bind(function(response) { + this.addImageURL(response.url); + this.uploadLoader.removeClass('is-active'); + this.enableSubmits(); + }, this), + error: _.bind(function(response) { + var msg; + if (response && response.status === 415) { + wcEvents.trigger('flash:error', + {message: this.inputMap.image.helpText, timeout: 5000}); + } + + if (response && response.status === 413) { + msg = 'The image is too big! Please choose something smaller than 4MB.'; + wcEvents.trigger('flash:error', {message: msg, timeout: 5000}); + } + this.loaderImage.hide(); + }, this) + }); + + // clear out the input[type=file], because we don't need it anymore. + this.uploadField.val(this.uploadField.get(0).defaultValue); + }; + /* + copy over the URL of a newly uploaded image asset to the bug + description textarea. + */ + this.addImageURL = function(url) { + var imageURL = ['![Screenshot Description](', url, ')'].join(''); + this.descField.val(function(idx, value) { + return value + '\n\n' + imageURL; + }); + }; /* copy URL from urlField into the first line of the description field. early return if the user has deleted @@ -371,7 +401,6 @@ function BugForm() { } return value.replace(firstLine, prefix + this.urlField.val() + '\n'); }, this)); - }; return this.init(); diff --git a/webcompat/templates/home-page/form.html b/webcompat/templates/home-page/form.html index 5f68948a3..d6d49d6b9 100644 --- a/webcompat/templates/home-page/form.html +++ b/webcompat/templates/home-page/form.html @@ -72,7 +72,11 @@ {% endif %} {{ form.image(class_='wc-ButtonUpload', accept='.jpe,.jpg,.jpeg,.gif,.png,.bmp') }} - + diff --git a/webcompat/views.py b/webcompat/views.py index 6d8e88d89..0e62f8680 100644 --- a/webcompat/views.py +++ b/webcompat/views.py @@ -160,10 +160,6 @@ def create_issue(): # copy the form so we can add the full UA string to it. form = request.form.copy() form['ua_header'] = request.headers.get('User-Agent') - # Do we have an image or screenshot ready to be uploaded? - if ((request.files['image'] and request.files['image'].filename) or - request.form.get('screenshot')): - form['image_upload'] = json.loads(upload()[0]) if form.get('submit-type') == AUTH_REPORT: if g.user: # If you're already authed, submit the bug. response = report_issue(form)