Files
gerrit/polygerrit-ui/app/elements/shared/gr-rest-api-interface/gr-rest-api-interface_test.html
Urs Wolfer 97926a41f3 Fix server error handling in 'gr-rest-api-interface'
When a response is not 'ok', an event 'server-error' is fired
with the response as argument. If an error handler reads
the text content of this response, it cannot be read again
in following call to 'getResponseObject'. If we try to read it
again, it will lead in an error 'Already read' and the event
'network-error' will be fired because of this.

This issue can be reproduced with executing an invalid
change query. The meaningful error message will be hidden
directly with a generic error message 'Server unavailable'.

Change-Id: Iffa85c2762a965f5f55fcb7041ea5fad4ed4f475
2016-06-04 20:00:26 +00:00

273 lines
8.8 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE html>
<!--
Copyright (C) 2016 The Android Open Source Project
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
-->
<meta name="viewport" content="width=device-width, minimum-scale=1.0, initial-scale=1.0, user-scalable=yes">
<title>gr-rest-api-interface</title>
<script src="../../../bower_components/webcomponentsjs/webcomponents.min.js"></script>
<script src="../../../bower_components/web-component-tester/browser.js"></script>
<link rel="import" href="../../../bower_components/iron-test-helpers/iron-test-helpers.html">
<link rel="import" href="gr-rest-api-interface.html">
<test-fixture id="basic">
<template>
<gr-rest-api-interface></gr-rest-api-interface>
</template>
</test-fixture>
<script>
suite('gr-rest-api-interface tests', function() {
var element;
setup(function() {
element = fixture('basic');
});
test('JSON prefix is properly removed', function(done) {
var testJSON = ')]}\'\n{"hello": "bonjour"}';
var fetchStub = sinon.stub(window, 'fetch', function() {
return Promise.resolve({ok: true, text: function() {
return Promise.resolve(testJSON);
}});
});
element.fetchJSON('/dummy/url').then(function(obj) {
assert.deepEqual(obj, {hello: 'bonjour'});
fetchStub.restore();
done();
});
});
test('cached results', function(done) {
var n = 0;
var fetchJSONStub = sinon.stub(element, 'fetchJSON', function() {
return Promise.resolve(++n);
});
var promises = [];
promises.push(element._fetchSharedCacheURL('/foo'));
promises.push(element._fetchSharedCacheURL('/foo'));
promises.push(element._fetchSharedCacheURL('/foo'));
Promise.all(promises).then(function(results) {
assert.deepEqual(results, [1, 1, 1]);
element._fetchSharedCacheURL('/foo').then(function(foo) {
assert.equal(foo, 1);
fetchJSONStub.restore();
done();
});
});
});
test('cached promise', function(done) {
var promise = Promise.reject('foo');
element._cache['/foo'] = promise;
element._fetchSharedCacheURL('/foo').catch(function(p) {
assert.equal(p, 'foo');
done();
});
});
test('params are properly encoded', function() {
var url = element._urlWithParams('/path/', {
sp: 'hola',
gr: 'guten tag',
noval: null,
});
assert.equal(url, '/path/?sp=hola&gr=guten%20tag&noval');
url = element._urlWithParams('/path/', {
sp: 'hola',
en: ['hey', 'hi'],
});
assert.equal(url, '/path/?sp=hola&en=hey&en=hi');
// Order must be maintained with array params.
url = element._urlWithParams('/path/', {
l: ['c', 'b', 'a'],
});
assert.equal(url, '/path/?l=c&l=b&l=a');
});
test('request callbacks can be canceled', function(done) {
var cancelCalled = false;
var fetchStub = sinon.stub(window, 'fetch', function() {
return Promise.resolve({body: {
cancel: function() { cancelCalled = true; }
}});
});
element.fetchJSON('/dummy/url', null, function() { return true; }).then(
function(obj) {
assert.isUndefined(obj);
assert.isTrue(cancelCalled);
fetchStub.restore();
done();
});
});
test('parent diff comments are properly grouped', function(done) {
var fetchJSONStub = sinon.stub(element, 'fetchJSON', function() {
return Promise.resolve({
'/COMMIT_MSG': [],
'sieve.go': [
{
message: 'this isnt quite right',
},
{
side: 'PARENT',
message: 'how did this work in the first place?',
},
],
});
});
element._getDiffComments('42', '', 'PARENT', 1, 'sieve.go').then(
function(obj) {
assert.equal(obj.baseComments.length, 1);
assert.deepEqual(obj.baseComments[0], {
side: 'PARENT',
message: 'how did this work in the first place?',
path: 'sieve.go',
});
assert.equal(obj.comments.length, 1);
assert.deepEqual(obj.comments[0], {
message: 'this isnt quite right',
path: 'sieve.go',
});
fetchJSONStub.restore();
done();
});
});
test('differing patch diff comments are properly grouped', function(done) {
var fetchJSONStub = sinon.stub(element, 'fetchJSON', function(url) {
if (url == '/changes/42/revisions/1') {
return Promise.resolve({
'/COMMIT_MSG': [],
'sieve.go': [
{
message: 'this isnt quite right',
},
{
side: 'PARENT',
message: 'how did this work in the first place?',
},
],
});
} else if (url == '/changes/42/revisions/2') {
return Promise.resolve({
'/COMMIT_MSG': [],
'sieve.go': [
{
message: 'What on earth are you thinking, here?',
},
{
side: 'PARENT',
message: 'Yeah not sure how this worked either?',
},
{
message: '¯\\_(ツ)_/¯',
},
],
});
}
});
element._getDiffComments('42', '', 1, 2, 'sieve.go').then(
function(obj) {
assert.equal(obj.baseComments.length, 1);
assert.deepEqual(obj.baseComments[0], {
message: 'this isnt quite right',
path: 'sieve.go',
});
assert.equal(obj.comments.length, 2);
assert.deepEqual(obj.comments[0], {
message: 'What on earth are you thinking, here?',
path: 'sieve.go',
});
assert.deepEqual(obj.comments[1], {
message: '¯\\_(ツ)_/¯',
path: 'sieve.go',
});
fetchJSONStub.restore();
done();
});
});
test('special file path sorting', function() {
assert.deepEqual(
['.b', '/COMMIT_MSG', '.a', 'file'].sort(
element._specialFilePathCompare),
['/COMMIT_MSG', '.a', '.b', 'file']);
assert.deepEqual(
['.b', '/COMMIT_MSG', 'foo/bar/baz.cc', 'foo/bar/baz.h'].sort(
element._specialFilePathCompare),
['/COMMIT_MSG', '.b', 'foo/bar/baz.h', 'foo/bar/baz.cc']);
assert.deepEqual(
['.b', '/COMMIT_MSG', 'foo/bar/baz.cc', 'foo/bar/baz.hpp'].sort(
element._specialFilePathCompare),
['/COMMIT_MSG', '.b', 'foo/bar/baz.hpp', 'foo/bar/baz.cc']);
assert.deepEqual(
['.b', '/COMMIT_MSG', 'foo/bar/baz.cc', 'foo/bar/baz.hxx'].sort(
element._specialFilePathCompare),
['/COMMIT_MSG', '.b', 'foo/bar/baz.hxx', 'foo/bar/baz.cc']);
assert.deepEqual(
['foo/bar.h', 'foo/bar.hxx', 'foo/bar.hpp'].sort(
element._specialFilePathCompare),
['foo/bar.h', 'foo/bar.hpp', 'foo/bar.hxx']);
});
test('rebase always enabled', function(done) {
var resolveFetchJSON;
var fetchJSONStub = sinon.stub(element, 'fetchJSON').returns(
new Promise(function(resolve) {
resolveFetchJSON = resolve;
}));
element.getChangeRevisionActions('42', '1337').then(
function(response) {
assert.isTrue(response.rebase.enabled);
fetchJSONStub.restore();
done();
});
resolveFetchJSON({rebase: {}});
});
test('server error', function(done) {
var getResponseObjectStub = sinon.stub(element, 'getResponseObject');
var fetchStub = sinon.stub(window, 'fetch', function() {
return Promise.resolve({ok: false});
});
var serverErrorEventPromise = new Promise(function(resolve) {
element.addEventListener('server-error', function() { resolve(); });
});
element.fetchJSON().then(
function(response) {
assert.isUndefined(response);
assert.isTrue(getResponseObjectStub.notCalled);
getResponseObjectStub.restore();
fetchStub.restore();
serverErrorEventPromise.then(function () {
done();
});
});
});
});
</script>