mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
An unknown error should return a 500
This commit is contained in:
parent
8162028fff
commit
103b995c32
@ -19,73 +19,70 @@ class ScimUserController {
|
|||||||
) {}
|
) {}
|
||||||
|
|
||||||
public async getSingleUser(resource: any, context: RequestContext) {
|
public async getSingleUser(resource: any, context: RequestContext) {
|
||||||
this._checkAccess(context);
|
return this._runAndHandleErrors(context, async () => {
|
||||||
|
const id = ScimUserController._getIdFromResource(resource);
|
||||||
const id = ScimUserController._getIdFromResource(resource);
|
const user = await this._dbManager.getUser(id);
|
||||||
const user = await this._dbManager.getUser(id);
|
if (!user) {
|
||||||
if (!user) {
|
throw new SCIMMY.Types.Error(404, null!, `User with ID ${id} not found`);
|
||||||
throw new SCIMMY.Types.Error(404, null!, `User with ID ${id} not found`);
|
}
|
||||||
}
|
return toSCIMMYUser(user);
|
||||||
return toSCIMMYUser(user);
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getUsers(resource: any, context: RequestContext) {
|
public async getUsers(resource: any, context: RequestContext) {
|
||||||
this._checkAccess(context);
|
return this._runAndHandleErrors(context, async () => {
|
||||||
|
const { filter } = resource;
|
||||||
const { filter } = resource;
|
const scimmyUsers = (await this._dbManager.getUsers()).map(user => toSCIMMYUser(user));
|
||||||
const scimmyUsers = (await this._dbManager.getUsers()).map(user => toSCIMMYUser(user));
|
return filter ? filter.match(scimmyUsers) : scimmyUsers;
|
||||||
return filter ? filter.match(scimmyUsers) : scimmyUsers;
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public async createUser(data: any, context: RequestContext) {
|
public async createUser(data: any, context: RequestContext) {
|
||||||
this._checkAccess(context);
|
return this._runAndHandleErrors(context, async () => {
|
||||||
|
|
||||||
try {
|
|
||||||
await this._checkEmailIsUnique(data.userName);
|
await this._checkEmailIsUnique(data.userName);
|
||||||
const userProfile = toUserProfile(data);
|
const userProfile = toUserProfile(data);
|
||||||
const newUser = await this._dbManager.getUserByLoginWithRetry(userProfile.email, {
|
const newUser = await this._dbManager.getUserByLoginWithRetry(userProfile.email, {
|
||||||
profile: userProfile
|
profile: userProfile
|
||||||
});
|
});
|
||||||
return toSCIMMYUser(newUser);
|
return toSCIMMYUser(newUser);
|
||||||
} catch (ex) {
|
});
|
||||||
return this._toScimError(ex);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async overrideUser(resource: any, data: any, context: RequestContext) {
|
public async overrideUser(resource: any, data: any, context: RequestContext) {
|
||||||
this._checkAccess(context);
|
return this._runAndHandleErrors(context, async () => {
|
||||||
|
|
||||||
try {
|
|
||||||
const id = ScimUserController._getIdFromResource(resource);
|
const id = ScimUserController._getIdFromResource(resource);
|
||||||
await this._checkEmailIsUnique(data.userName, id);
|
await this._checkEmailIsUnique(data.userName, id);
|
||||||
const updatedUser = await this._dbManager.overrideUser(id, toUserProfile(data));
|
const updatedUser = await this._dbManager.overrideUser(id, toUserProfile(data));
|
||||||
return toSCIMMYUser(updatedUser);
|
return toSCIMMYUser(updatedUser);
|
||||||
} catch (ex) {
|
});
|
||||||
return this._toScimError(ex);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async deleteUser(resource: any, context: RequestContext) {
|
public async deleteUser(resource: any, context: RequestContext) {
|
||||||
this._checkAccess(context);
|
return this._runAndHandleErrors(context, async () => {
|
||||||
|
const id = ScimUserController._getIdFromResource(resource);
|
||||||
const id = ScimUserController._getIdFromResource(resource);
|
|
||||||
try {
|
|
||||||
const fakeScope: Scope = { userId: id };
|
const fakeScope: Scope = { userId: id };
|
||||||
// FIXME: deleteUser should probably better not requiring a scope.
|
// FIXME: deleteUser should probably better not requiring a scope.
|
||||||
await this._dbManager.deleteUser(fakeScope, id);
|
await this._dbManager.deleteUser(fakeScope, id);
|
||||||
} catch (ex) {
|
});
|
||||||
return this._toScimError(ex);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _toScimError(ex: Error) {
|
private async _runAndHandleErrors<T>(context: RequestContext, cb: () => Promise<T>): Promise<T> {
|
||||||
if (ex instanceof ApiError) {
|
try {
|
||||||
if (ex.status === 409) {
|
this._checkAccess(context);
|
||||||
throw new SCIMMY.Types.Error(ex.status, 'uniqueness', ex.message);
|
return await cb();
|
||||||
|
} catch (ex) {
|
||||||
|
if (ex instanceof ApiError) {
|
||||||
|
if (ex.status === 409) {
|
||||||
|
throw new SCIMMY.Types.Error(ex.status, 'uniqueness', ex.message);
|
||||||
|
}
|
||||||
|
throw new SCIMMY.Types.Error(ex.status, null!, ex.message);
|
||||||
}
|
}
|
||||||
throw new SCIMMY.Types.Error(ex.status, null!, ex.message);
|
if (ex instanceof SCIMMY.Types.Error) {
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
// By default, return a 500 error
|
||||||
|
throw new SCIMMY.Types.Error(500, null!, ex.message);
|
||||||
}
|
}
|
||||||
throw ex;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _checkEmailIsUnique(email: string, id?: number) {
|
private async _checkEmailIsUnique(email: string, id?: number) {
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import capitalize from 'lodash/capitalize';
|
import capitalize from 'lodash/capitalize';
|
||||||
import { assert } from 'chai';
|
import { assert } from 'chai';
|
||||||
|
import Sinon from 'sinon';
|
||||||
|
|
||||||
import { TestServer } from 'test/gen-server/apiUtils';
|
import { TestServer } from 'test/gen-server/apiUtils';
|
||||||
import { configForUser } from 'test/gen-server/testUtils';
|
import { configForUser } from 'test/gen-server/testUtils';
|
||||||
import * as testUtils from 'test/server/testUtils';
|
import * as testUtils from 'test/server/testUtils';
|
||||||
@ -115,7 +117,7 @@ describe('Scim', () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function checkEndpointNotAccessibleForNonAdminUsers(
|
function checkCommonErrors(
|
||||||
method: 'get' | 'post' | 'put' | 'patch' | 'delete',
|
method: 'get' | 'post' | 'put' | 'patch' | 'delete',
|
||||||
path: string,
|
path: string,
|
||||||
validBody: object = {}
|
validBody: object = {}
|
||||||
@ -127,12 +129,12 @@ describe('Scim', () => {
|
|||||||
return axios[method](scimUrl(path), validBody, USER_CONFIG_BY_NAME[user]);
|
return axios[method](scimUrl(path), validBody, USER_CONFIG_BY_NAME[user]);
|
||||||
}
|
}
|
||||||
it('should return 401 for anonymous', async function () {
|
it('should return 401 for anonymous', async function () {
|
||||||
const res: any = await makeCallWith('anon');
|
const res = await makeCallWith('anon');
|
||||||
assert.equal(res.status, 401);
|
assert.equal(res.status, 401);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return 403 for kiwi', async function () {
|
it('should return 403 for kiwi', async function () {
|
||||||
const res: any = await makeCallWith('kiwi');
|
const res = await makeCallWith('kiwi');
|
||||||
assert.deepEqual(res.data, {
|
assert.deepEqual(res.data, {
|
||||||
schemas: [ 'urn:ietf:params:scim:api:messages:2.0:Error' ],
|
schemas: [ 'urn:ietf:params:scim:api:messages:2.0:Error' ],
|
||||||
status: '403',
|
status: '403',
|
||||||
@ -140,6 +142,30 @@ describe('Scim', () => {
|
|||||||
});
|
});
|
||||||
assert.equal(res.status, 403);
|
assert.equal(res.status, 403);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should return a 500 in case of unknown Error', async function () {
|
||||||
|
const sandbox = Sinon.createSandbox();
|
||||||
|
try {
|
||||||
|
const error = new Error('Some unexpected Error');
|
||||||
|
|
||||||
|
// Stub all the dbManager methods called by the controller
|
||||||
|
sandbox.stub(getDbManager(), 'getUsers').throws(error);
|
||||||
|
sandbox.stub(getDbManager(), 'getUser').throws(error);
|
||||||
|
sandbox.stub(getDbManager(), 'getUserByLoginWithRetry').throws(error);
|
||||||
|
sandbox.stub(getDbManager(), 'overrideUser').throws(error);
|
||||||
|
sandbox.stub(getDbManager(), 'deleteUser').throws(error);
|
||||||
|
|
||||||
|
const res = await makeCallWith('chimpy');
|
||||||
|
assert.deepEqual(res.data, {
|
||||||
|
schemas: [ 'urn:ietf:params:scim:api:messages:2.0:Error' ],
|
||||||
|
status: '500',
|
||||||
|
detail: error.message
|
||||||
|
});
|
||||||
|
assert.equal(res.status, 500);
|
||||||
|
} finally {
|
||||||
|
sandbox.restore();
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
describe('/Me', function () {
|
describe('/Me', function () {
|
||||||
@ -181,10 +207,9 @@ describe('Scim', () => {
|
|||||||
preferredLanguage: 'en',
|
preferredLanguage: 'en',
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('/Users/{id}', function () {
|
describe('GET /Users/{id}', function () {
|
||||||
|
|
||||||
it('should return the user of id=1 for chimpy', async function () {
|
it('should return the user of id=1 for chimpy', async function () {
|
||||||
const res = await axios.get(scimUrl('/Users/1'), chimpy);
|
const res = await axios.get(scimUrl('/Users/1'), chimpy);
|
||||||
@ -208,7 +233,7 @@ describe('Scim', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('get', '/Users/1');
|
checkCommonErrors('get', '/Users/1');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('GET /Users', function () {
|
describe('GET /Users', function () {
|
||||||
@ -220,7 +245,7 @@ describe('Scim', () => {
|
|||||||
assert.deepInclude(res.data.Resources, personaToSCIMMYUserWithId('kiwi'));
|
assert.deepInclude(res.data.Resources, personaToSCIMMYUserWithId('kiwi'));
|
||||||
});
|
});
|
||||||
|
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('get', '/Users');
|
checkCommonErrors('get', '/Users');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('POST /Users/.search', function () {
|
describe('POST /Users/.search', function () {
|
||||||
@ -261,7 +286,7 @@ describe('Scim', () => {
|
|||||||
"should have retrieved only chimpy's username and not other attribute");
|
"should have retrieved only chimpy's username and not other attribute");
|
||||||
});
|
});
|
||||||
|
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('post', '/Users/.search', searchExample);
|
checkCommonErrors('post', '/Users/.search', searchExample);
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('POST /Users', function () { // Create a new users
|
describe('POST /Users', function () { // Create a new users
|
||||||
@ -331,7 +356,7 @@ describe('Scim', () => {
|
|||||||
assert.equal(res.status, 409);
|
assert.equal(res.status, 409);
|
||||||
});
|
});
|
||||||
|
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('post', '/Users', toSCIMUserWithoutId('some-user'));
|
checkCommonErrors('post', '/Users', toSCIMUserWithoutId('some-user'));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('PUT /Users/{id}', function () {
|
describe('PUT /Users/{id}', function () {
|
||||||
@ -443,7 +468,7 @@ describe('Scim', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('put', '/Users/1', toSCIMUserWithoutId('chimpy'));
|
checkCommonErrors('put', '/Users/1', toSCIMUserWithoutId('chimpy'));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('PATCH /Users/{id}', function () {
|
describe('PATCH /Users/{id}', function () {
|
||||||
@ -483,7 +508,7 @@ describe('Scim', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('patch', '/Users/1', validPatchBody('new name2'));
|
checkCommonErrors('patch', '/Users/1', validPatchBody('new name2'));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('DELETE /Users/{id}', function () {
|
describe('DELETE /Users/{id}', function () {
|
||||||
@ -513,7 +538,7 @@ describe('Scim', () => {
|
|||||||
});
|
});
|
||||||
assert.equal(res.status, 404);
|
assert.equal(res.status, 404);
|
||||||
});
|
});
|
||||||
checkEndpointNotAccessibleForNonAdminUsers('delete', '/Users/1');
|
checkCommonErrors('delete', '/Users/1');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('POST /Bulk', function () {
|
describe('POST /Bulk', function () {
|
||||||
|
Loading…
Reference in New Issue
Block a user