Ghost/ghost/admin/tests/integration/components/gh-image-uploader-test.js

578 lines
22 KiB
JavaScript
Raw Normal View History

/* jshint expr:true */
import sinon from 'sinon';
import {expect} from 'chai';
2016-11-24 01:50:57 +03:00
import {describe, it} from 'mocha';
import {setupComponentTest} from 'ember-mocha';
import hbs from 'htmlbars-inline-precompile';
import Pretender from 'pretender';
import wait from 'ember-test-helpers/wait';
import {createFile, fileUpload} from '../../helpers/file-upload';
import $ from 'jquery';
import run from 'ember-runloop';
import Service from 'ember-service';
import {UnsupportedMediaTypeError} from 'ghost-admin/services/ajax';
const keyCodes = {
enter: 13
};
const configStub = Service.extend({
fileStorage: true
});
const notificationsStub = Service.extend({
showAPIError(/* error, options */) {
// noop - to be stubbed
}
});
const sessionStub = Service.extend({
isAuthenticated: false,
authorize(authorizer, block) {
if (this.get('isAuthenticated')) {
block('Authorization', 'Bearer token');
}
}
});
const stubSuccessfulUpload = function (server, delay = 0) {
server.post('/ghost/api/v0.1/uploads/', function () {
return [200, {'Content-Type': 'application/json'}, '"/content/images/test.png"'];
}, delay);
};
const stubFailedUpload = function (server, code, error, delay = 0) {
server.post('/ghost/api/v0.1/uploads/', function () {
return [code, {'Content-Type': 'application/json'}, JSON.stringify({
errors: [{
errorType: error,
message: `Error: ${error}`
}]
})];
}, delay);
};
2016-11-24 01:50:57 +03:00
describe('Integration: Component: gh-image-uploader', function() {
setupComponentTest('gh-image-upload', {
integration: true
2016-11-24 01:50:57 +03:00
});
let server;
beforeEach(function () {
this.register('service:config', configStub);
this.register('service:session', sessionStub);
this.register('service:notifications', notificationsStub);
this.inject.service('config', {as: 'configService'});
this.inject.service('session', {as: 'sessionService'});
this.inject.service('notifications', {as: 'notifications'});
this.set('update', function () {});
server = new Pretender();
});
afterEach(function () {
server.shutdown();
});
it('renders', function() {
this.set('image', 'http://example.com/test.png');
this.render(hbs`{{gh-image-uploader image=image}}`);
expect(this.$()).to.have.length(1);
});
it('defaults to upload form', function () {
this.render(hbs`{{gh-image-uploader image=image}}`);
expect(this.$('input[type="file"]').length).to.equal(1);
});
it('defaults to url form with no filestorage config', function () {
this.set('configService.fileStorage', false);
this.render(hbs`{{gh-image-uploader image=image}}`);
expect(this.$('input[type="file"]').length).to.equal(0);
expect(this.$('input[type="text"].url').length).to.equal(1);
});
it('can switch between form types', function () {
this.render(hbs`{{gh-image-uploader image=image}}`);
expect(this.$('input[type="file"]').length).to.equal(1);
expect(this.$('input[type="text"].url').length).to.equal(0);
this.$('a.image-url').click();
expect(this.$('input[type="file"]').length, 'upload form is visible after switch to url form')
.to.equal(0);
expect(this.$('input[type="text"].url').length, 'url form is visible after switch to url form')
.to.equal(1);
this.$('a.image-upload').click();
expect(this.$('input[type="file"]').length, 'upload form is visible after switch to upload form')
.to.equal(1);
expect(this.$('input[type="text"].url').length, 'url form is visible after switch to upload form')
.to.equal(0);
});
it('triggers formChanged action when switching between forms', function () {
let formChanged = sinon.spy();
this.set('formChanged', formChanged);
this.render(hbs`{{gh-image-uploader image=image formChanged=(action formChanged)}}`);
this.$('a.image-url').click();
this.$('a.image-upload').click();
expect(formChanged.calledTwice).to.be.true;
expect(formChanged.firstCall.args[0]).to.equal('url-input');
expect(formChanged.secondCall.args[0]).to.equal('upload');
});
describe('file upload form', function () {
it('renders form with supplied alt text', function () {
this.render(hbs`{{gh-image-uploader image=image altText="text test"}}`);
expect(this.$('.description').text().trim()).to.equal('Upload image of "text test"');
});
2016-11-24 01:50:57 +03:00
it('renders form with supplied text', function () {
this.render(hbs`{{gh-image-uploader image=image text="text test"}}`);
expect(this.$('.description').text().trim()).to.equal('text test');
});
2016-11-24 01:50:57 +03:00
it('generates request to correct endpoint', function (done) {
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(server.handledRequests.length).to.equal(1);
expect(server.handledRequests[0].url).to.equal('/ghost/api/v0.1/uploads/');
expect(server.handledRequests[0].requestHeaders.Authorization).to.be.undefined;
done();
});
});
2016-11-24 01:50:57 +03:00
it('adds authentication headers to request', function (done) {
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.get('sessionService').set('isAuthenticated', true);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
let [request] = server.handledRequests;
expect(request.requestHeaders.Authorization).to.equal('Bearer token');
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('fires update action on successful upload', function (done) {
let update = sinon.spy();
this.set('update', update);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(update.calledOnce).to.be.true;
expect(update.firstCall.args[0]).to.equal('/content/images/test.png');
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('doesn\'t fire update action on failed upload', function (done) {
let update = sinon.spy();
this.set('update', update);
2016-11-24 01:50:57 +03:00
stubFailedUpload(server, 500);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(update.calledOnce).to.be.false;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('fires fileSelected action on file selection', function (done) {
let fileSelected = sinon.spy();
this.set('fileSelected', fileSelected);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
2017-01-31 23:02:10 +03:00
this.render(hbs`{{gh-image-uploader image=image fileSelected=(action fileSelected) update=(action update)}}`);
2016-11-24 01:50:57 +03:00
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(fileSelected.calledOnce).to.be.true;
expect(fileSelected.args[0]).to.not.be.blank;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('fires uploadStarted action on upload start', function (done) {
let uploadStarted = sinon.spy();
this.set('uploadStarted', uploadStarted);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image uploadStarted=(action uploadStarted) update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(uploadStarted.calledOnce).to.be.true;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('fires uploadFinished action on successful upload', function (done) {
let uploadFinished = sinon.spy();
this.set('uploadFinished', uploadFinished);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image uploadFinished=(action uploadFinished) update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(uploadFinished.calledOnce).to.be.true;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('fires uploadFinished action on failed upload', function (done) {
let uploadFinished = sinon.spy();
this.set('uploadFinished', uploadFinished);
2016-11-24 01:50:57 +03:00
stubFailedUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image uploadFinished=(action uploadFinished) update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(uploadFinished.calledOnce).to.be.true;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('displays invalid file type error', function (done) {
stubFailedUpload(server, 415, 'UnsupportedMediaTypeError');
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
wait().then(() => {
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/The image type you uploaded is not supported/);
expect(this.$('.gh-btn-green').length, 'reset button is displayed').to.equal(1);
expect(this.$('.gh-btn-green').text()).to.equal('Try Again');
2016-11-24 01:50:57 +03:00
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('displays file too large for server error', function (done) {
stubFailedUpload(server, 413, 'RequestEntityTooLargeError');
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/The image you uploaded was larger/);
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('handles file too large error directly from the web server', function (done) {
server.post('/ghost/api/v0.1/uploads/', function () {
return [413, {}, ''];
});
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/The image you uploaded was larger/);
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('displays other server-side error with message', function (done) {
stubFailedUpload(server, 400, 'UnknownError');
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/Error: UnknownError/);
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('handles unknown failure', function (done) {
server.post('/ghost/api/v0.1/uploads/', function () {
return [500, {'Content-Type': 'application/json'}, ''];
});
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/Something went wrong/);
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('triggers notifications.showAPIError for VersionMismatchError', function (done) {
let showAPIError = sinon.spy();
this.set('notifications.showAPIError', showAPIError);
2016-11-24 01:50:57 +03:00
stubFailedUpload(server, 400, 'VersionMismatchError');
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(showAPIError.calledOnce).to.be.true;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('doesn\'t trigger notifications.showAPIError for other errors', function (done) {
let showAPIError = sinon.spy();
this.set('notifications.showAPIError', showAPIError);
2016-11-24 01:50:57 +03:00
stubFailedUpload(server, 400, 'UnknownError');
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(showAPIError.called).to.be.false;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('can be reset after a failed upload', function (done) {
stubFailedUpload(server, 400, 'UnknownError');
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {type: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
run(() => {
this.$('.gh-btn-green').click();
});
});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(this.$('input[type="file"]').length).to.equal(1);
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('displays upload progress', function (done) {
this.set('done', done);
2016-11-24 01:50:57 +03:00
// pretender fires a progress event every 50ms
stubSuccessfulUpload(server, 150);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image uploadFinished=(action done) update=(action update)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
// after 75ms we should have had one progress event
run.later(this, function () {
expect(this.$('.progress .bar').length).to.equal(1);
let [, percentageWidth] = this.$('.progress .bar').attr('style').match(/width: (\d+)%?/);
expect(percentageWidth).to.be.above(0);
expect(percentageWidth).to.be.below(100);
}, 75);
});
2016-11-24 01:50:57 +03:00
it('handles drag over/leave', function () {
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
2016-11-24 01:50:57 +03:00
run(() => {
// eslint-disable-next-line new-cap
2016-11-24 01:50:57 +03:00
let dragover = $.Event('dragover', {
dataTransfer: {
2016-11-24 01:50:57 +03:00
files: []
}
});
2016-11-24 01:50:57 +03:00
this.$('.gh-image-uploader').trigger(dragover);
});
2016-11-24 01:50:57 +03:00
expect(this.$('.gh-image-uploader').hasClass('-drag-over'), 'has drag-over class').to.be.true;
2016-11-24 01:50:57 +03:00
run(() => {
this.$('.gh-image-uploader').trigger('dragleave');
});
2016-11-24 01:50:57 +03:00
expect(this.$('.gh-image-uploader').hasClass('-drag-over'), 'has drag-over class').to.be.false;
});
2016-11-24 01:50:57 +03:00
it('triggers file upload on file drop', function (done) {
let uploadSuccess = sinon.spy();
// eslint-disable-next-line new-cap
let drop = $.Event('drop', {
dataTransfer: {
files: [createFile(['test'], {name: 'test.png'})]
}
});
2016-11-24 01:50:57 +03:00
this.set('uploadSuccess', uploadSuccess);
stubSuccessfulUpload(server);
this.render(hbs`{{gh-image-uploader uploadSuccess=(action uploadSuccess)}}`);
2016-11-24 01:50:57 +03:00
run(() => {
this.$('.gh-image-uploader').trigger(drop);
});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(uploadSuccess.calledOnce).to.be.true;
expect(uploadSuccess.firstCall.args[0]).to.equal('/content/images/test.png');
done();
});
});
2016-11-24 01:50:57 +03:00
it('validates extension by default', function (done) {
let uploadSuccess = sinon.spy();
let uploadFailed = sinon.spy();
2016-11-24 01:50:57 +03:00
this.set('uploadSuccess', uploadSuccess);
this.set('uploadFailed', uploadFailed);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
this.render(hbs`{{gh-image-uploader
uploadSuccess=(action uploadSuccess)
uploadFailed=(action uploadFailed)}}`);
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.json'});
wait().then(() => {
expect(uploadSuccess.called).to.be.false;
expect(uploadFailed.calledOnce).to.be.true;
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/The image type you uploaded is not supported/);
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('uploads if validate action supplied and returns true', function (done) {
let validate = sinon.stub().returns(true);
let uploadSuccess = sinon.spy();
2016-11-24 01:50:57 +03:00
this.set('validate', validate);
this.set('uploadSuccess', uploadSuccess);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader
uploadSuccess=(action uploadSuccess)
validate=(action validate)}}`);
2016-11-24 01:50:57 +03:00
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.txt'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(validate.calledOnce).to.be.true;
expect(uploadSuccess.calledOnce).to.be.true;
done();
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
it('skips upload and displays error if validate action supplied and doesn\'t return true', function (done) {
let validate = sinon.stub().returns(new UnsupportedMediaTypeError());
let uploadSuccess = sinon.spy();
let uploadFailed = sinon.spy();
2016-11-24 01:50:57 +03:00
this.set('validate', validate);
this.set('uploadSuccess', uploadSuccess);
this.set('uploadFailed', uploadFailed);
2016-11-24 01:50:57 +03:00
stubSuccessfulUpload(server);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader
uploadSuccess=(action uploadSuccess)
uploadFailed=(action uploadFailed)
validate=(action validate)}}`);
2016-11-24 01:50:57 +03:00
fileUpload(this.$('input[type="file"]'), ['test'], {name: 'test.png'});
2016-11-24 01:50:57 +03:00
wait().then(() => {
expect(validate.calledOnce).to.be.true;
expect(uploadSuccess.called).to.be.false;
expect(uploadFailed.calledOnce).to.be.true;
expect(this.$('.failed').length, 'error message is displayed').to.equal(1);
expect(this.$('.failed').text()).to.match(/The image type you uploaded is not supported/);
done();
});
});
2016-11-24 01:50:57 +03:00
});
2016-11-24 01:50:57 +03:00
describe('URL input form', function () {
beforeEach(function () {
this.set('configService.fileStorage', false);
});
2016-11-24 01:50:57 +03:00
it('displays save button by default', function () {
this.set('image', 'http://example.com/test.png');
this.render(hbs`{{gh-image-uploader image=image text="text test"}}`);
expect(this.$('button').length).to.equal(1);
expect(this.$('input[type="text"]').val()).to.equal('http://example.com/test.png');
});
2016-11-24 01:50:57 +03:00
it('can render without a save button', function () {
this.render(hbs`{{gh-image-uploader image=image saveButton=false text="text test"}}`);
expect(this.$('button').length).to.equal(0);
expect(this.$('.description').text().trim()).to.equal('text test');
});
2016-11-24 01:50:57 +03:00
it('fires update action when save button clicked', function () {
let update = sinon.spy();
this.set('update', update);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
2016-11-24 01:50:57 +03:00
this.$('input[type="text"]').val('saved url');
this.$('input[type="text"]').change();
this.$('button.gh-btn-blue').click();
2016-11-24 01:50:57 +03:00
expect(update.calledOnce).to.be.true;
expect(update.firstCall.args[0]).to.equal('saved url');
});
2016-11-24 01:50:57 +03:00
it('fires onInput action when typing URL', function () {
let onInput = sinon.spy();
this.set('onInput', onInput);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image onInput=(action onInput)}}`);
2016-11-24 01:50:57 +03:00
this.$('input[type="text"]').val('input url');
this.$('input[type="text"]').change();
2016-11-24 01:50:57 +03:00
expect(onInput.calledOnce).to.be.true;
expect(onInput.firstCall.args[0]).to.equal('input url');
});
2016-11-24 01:50:57 +03:00
it('saves on enter key', function () {
let update = sinon.spy();
this.set('update', update);
2016-11-24 01:50:57 +03:00
this.render(hbs`{{gh-image-uploader image=image update=(action update)}}`);
2016-11-24 01:50:57 +03:00
this.$('input[type="text"]').val('saved url');
this.$('input[type="text"]').change();
this.$('input[type="text"]').trigger(
// eslint-disable-next-line new-cap
$.Event('keyup', {keyCode: keyCodes.enter, which: keyCodes.enter})
);
2016-11-24 01:50:57 +03:00
expect(update.calledOnce).to.be.true;
expect(update.firstCall.args[0]).to.equal('saved url');
});
2016-11-24 01:50:57 +03:00
});
});