2021-02-05 14:01:37 +00:00
|
|
|
import { expect } from 'chai'
|
|
|
|
import fetchMock from 'fetch-mock'
|
2021-08-24 08:54:30 +00:00
|
|
|
import { Response } from 'node-fetch'
|
2021-02-05 14:01:37 +00:00
|
|
|
import {
|
|
|
|
deleteJSON,
|
2021-02-10 09:36:54 +00:00
|
|
|
FetchError,
|
2022-05-18 13:46:18 +00:00
|
|
|
getUserFacingMessage,
|
2021-02-05 14:01:37 +00:00
|
|
|
getJSON,
|
|
|
|
postJSON,
|
2021-04-27 07:52:58 +00:00
|
|
|
putJSON,
|
2021-02-05 14:01:37 +00:00
|
|
|
} from '../../../frontend/js/infrastructure/fetch-json'
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
describe('fetchJSON', function () {
|
|
|
|
before(function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
fetchMock.restore()
|
|
|
|
})
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
afterEach(function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
fetchMock.restore()
|
|
|
|
})
|
|
|
|
|
|
|
|
const headers = {
|
|
|
|
Accept: 'application/json',
|
2021-04-27 07:52:58 +00:00
|
|
|
'Content-Type': 'application/json',
|
2021-02-05 14:01:37 +00:00
|
|
|
}
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
it('handles GET requests', function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
fetchMock.once(
|
|
|
|
{ method: 'GET', url: '/test', headers },
|
|
|
|
{ status: 200, body: { result: 'success' } }
|
|
|
|
)
|
|
|
|
|
|
|
|
return expect(getJSON('/test')).to.eventually.deep.equal({
|
2021-04-27 07:52:58 +00:00
|
|
|
result: 'success',
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
it('handles 4xx responses', function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
fetchMock.get('/test', {
|
|
|
|
status: 400,
|
2021-04-27 07:52:58 +00:00
|
|
|
body: { message: 'The request was invalid' },
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return expect(getJSON('/test'))
|
|
|
|
.to.eventually.be.rejectedWith('Bad Request')
|
2021-02-10 09:36:54 +00:00
|
|
|
.and.be.an.instanceOf(FetchError)
|
2021-02-05 14:01:37 +00:00
|
|
|
.to.nested.include({
|
2021-02-10 09:36:54 +00:00
|
|
|
message: 'Bad Request',
|
|
|
|
'data.message': 'The request was invalid',
|
|
|
|
'response.status': 400,
|
2021-04-27 07:52:58 +00:00
|
|
|
'info.statusCode': 400,
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
it('handles 5xx responses', async function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
fetchMock.get('/test', { status: 500 })
|
|
|
|
|
|
|
|
return expect(getJSON('/test'))
|
|
|
|
.to.eventually.be.rejectedWith('Internal Server Error')
|
2021-02-10 09:36:54 +00:00
|
|
|
.and.be.an.instanceOf(FetchError)
|
2021-02-05 14:01:37 +00:00
|
|
|
.to.nested.include({
|
2021-02-10 09:36:54 +00:00
|
|
|
'response.status': 500,
|
2021-04-27 07:52:58 +00:00
|
|
|
'info.statusCode': 500,
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-06-24 08:28:23 +00:00
|
|
|
it('handles JSON error responses', async function () {
|
|
|
|
fetchMock.get('/test', {
|
|
|
|
status: 500,
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
},
|
|
|
|
body: { message: 'lorem ipsum' },
|
|
|
|
})
|
|
|
|
|
|
|
|
return expect(getJSON('/test'))
|
|
|
|
.to.eventually.be.rejectedWith('Internal Server Error')
|
|
|
|
.and.be.an.instanceOf(FetchError)
|
|
|
|
.to.nested.include({
|
|
|
|
'data.message': 'lorem ipsum',
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('handles text error responses', async function () {
|
|
|
|
fetchMock.get('/test', {
|
|
|
|
status: 500,
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
},
|
|
|
|
body: 'lorem ipsum',
|
|
|
|
})
|
|
|
|
|
|
|
|
return expect(getJSON('/test'))
|
|
|
|
.to.eventually.be.rejectedWith('Internal Server Error')
|
|
|
|
.and.be.an.instanceOf(FetchError)
|
|
|
|
.to.nested.include({
|
|
|
|
'data.message': 'lorem ipsum',
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('handles text error responses sent as HTML', async function () {
|
|
|
|
fetchMock.get('/test', {
|
|
|
|
status: 500,
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'text/html',
|
|
|
|
},
|
|
|
|
body: 'lorem ipsum',
|
|
|
|
})
|
|
|
|
|
|
|
|
return expect(getJSON('/test'))
|
|
|
|
.to.eventually.be.rejectedWith('Internal Server Error')
|
|
|
|
.and.be.an.instanceOf(FetchError)
|
|
|
|
.to.nested.include({
|
|
|
|
'data.message': 'lorem ipsum',
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('handles (ignores) HTML error responses sent as HTML', async function () {
|
|
|
|
fetchMock.get('/test', {
|
|
|
|
status: 500,
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'text/html',
|
|
|
|
},
|
2022-01-10 10:23:05 +00:00
|
|
|
body: '<!doctype html><html lang="en"><body><p>lorem ipsum</p></body></html>',
|
2021-06-24 08:28:23 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
const promise = getJSON('/test')
|
|
|
|
|
|
|
|
expect(promise)
|
|
|
|
.to.eventually.be.rejectedWith('Internal Server Error')
|
|
|
|
.and.be.an.instanceOf(FetchError)
|
|
|
|
|
|
|
|
try {
|
|
|
|
await promise
|
|
|
|
} catch (error) {
|
|
|
|
expect(error.data).to.eql({})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2021-08-24 08:54:30 +00:00
|
|
|
it('handles 5xx responses without a status message', async function () {
|
|
|
|
// It's hard to make a Response object with statusText=null,
|
|
|
|
// so we need to do some monkey-work to make it happen
|
|
|
|
const response = new Response('weird scary error', {
|
|
|
|
ok: false,
|
|
|
|
status: 599,
|
|
|
|
})
|
|
|
|
Object.defineProperty(response, 'statusText', {
|
|
|
|
get: () => null,
|
|
|
|
set: () => {},
|
|
|
|
})
|
|
|
|
fetchMock.get('/test', response)
|
|
|
|
|
|
|
|
return expect(getJSON('/test'))
|
|
|
|
.to.eventually.be.rejectedWith('Unexpected Error: 599')
|
|
|
|
.and.be.an.instanceOf(FetchError)
|
|
|
|
.to.nested.include({
|
|
|
|
'response.status': 599,
|
|
|
|
'info.statusCode': 599,
|
|
|
|
message: 'Unexpected Error: 599',
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
it('handles POST requests', function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
const body = { example: true }
|
|
|
|
|
|
|
|
fetchMock.once(
|
|
|
|
{ method: 'POST', url: '/test', headers, body },
|
|
|
|
{ status: 200, body: { result: 'success' } }
|
|
|
|
)
|
|
|
|
|
|
|
|
return expect(postJSON('/test', { body })).to.eventually.deep.equal({
|
2021-04-27 07:52:58 +00:00
|
|
|
result: 'success',
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
it('handles PUT requests', function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
const body = { example: true }
|
|
|
|
|
|
|
|
fetchMock.once(
|
|
|
|
{ method: 'PUT', url: '/test', headers, body },
|
|
|
|
{ status: 200, body: { result: 'success' } }
|
|
|
|
)
|
|
|
|
|
|
|
|
return expect(putJSON('/test', { body })).to.eventually.deep.equal({
|
2021-04-27 07:52:58 +00:00
|
|
|
result: 'success',
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2021-04-14 13:17:21 +00:00
|
|
|
it('handles DELETE requests', function () {
|
2021-02-05 14:01:37 +00:00
|
|
|
fetchMock.once({ method: 'DELETE', url: '/test', headers }, { status: 204 })
|
|
|
|
|
|
|
|
return expect(deleteJSON('/test')).to.eventually.deep.equal({})
|
|
|
|
})
|
2022-05-18 13:46:18 +00:00
|
|
|
|
|
|
|
describe('getUserFacingMessage()', function () {
|
|
|
|
it('returns the error facing message for FetchError instances', function () {
|
|
|
|
const error = new FetchError(
|
|
|
|
'403 error',
|
|
|
|
'http:/example.com',
|
|
|
|
{},
|
|
|
|
{ status: 403 }
|
|
|
|
)
|
|
|
|
expect(getUserFacingMessage(error)).to.equal(
|
|
|
|
'Session error. Please check you have cookies enabled. If the problem persists, try clearing your cache and cookies.'
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('returns `message` for Error instances different than FetchError', function () {
|
|
|
|
const error = new Error('403 error')
|
|
|
|
expect(getUserFacingMessage(error)).to.equal('403 error')
|
|
|
|
})
|
|
|
|
|
|
|
|
it('returns `undefined` for non-Error instances', function () {
|
|
|
|
expect(getUserFacingMessage(undefined)).to.be.undefined
|
|
|
|
expect(getUserFacingMessage(null)).to.be.undefined
|
|
|
|
expect(getUserFacingMessage('error')).to.be.undefined
|
|
|
|
})
|
|
|
|
})
|
2021-02-05 14:01:37 +00:00
|
|
|
})
|