73cc7e67d3
* feat(webhooks): multi region webhook resolver * feat(webhooks): multi region webhook cleanup * fix(webhooks): DI fixes * feat(activitystream): region aware save activity * feat(accessrequests): multi region * feat(cli): allow multi region project and commit download * feat(postgres): make docker postgres 0 day multi region ready * feat(cli): allow multi region project and commit download properly * fix(cross-server-sync): di fix * feat(activitystream): non region aware activities, they are not project data * fix(webhooks): triggers need to be included * feat(stream/projectCreate): activity save is not needed any more, its all event based * feat(multiRegion): get all registered db clients * fix(regions): test equal in any order * fix(projectDownload): need to await
352 lines
11 KiB
JavaScript
352 lines
11 KiB
JavaScript
/* istanbul ignore file */
|
|
const { mockRequireModule } = require('@/test/mockHelper')
|
|
const envHelperMock = mockRequireModule(
|
|
[
|
|
'@/modules/shared/helpers/envHelper',
|
|
require.resolve('../../shared/helpers/envHelper')
|
|
],
|
|
['@/modules/shared/index']
|
|
)
|
|
const expect = require('chai').expect
|
|
|
|
const { beforeEachContext } = require('@/test/hooks')
|
|
|
|
const { validateScopes, authorizeResolver } = require('@/modules/shared')
|
|
const { buildContext } = require('@/modules/shared/middleware')
|
|
const { Roles, Scopes } = require('@speckle/shared')
|
|
const { throwForNotHavingServerRole } = require('@/modules/shared/authz')
|
|
const { ForbiddenError } = require('@/modules/shared/errors')
|
|
const {
|
|
getStreamFactory,
|
|
createStreamFactory
|
|
} = require('@/modules/core/repositories/streams')
|
|
const { db } = require('@/db/knex')
|
|
const {
|
|
legacyCreateStreamFactory,
|
|
createStreamReturnRecordFactory
|
|
} = require('@/modules/core/services/streams/management')
|
|
const {
|
|
inviteUsersToProjectFactory
|
|
} = require('@/modules/serverinvites/services/projectInviteManagement')
|
|
const {
|
|
createAndSendInviteFactory
|
|
} = require('@/modules/serverinvites/services/creation')
|
|
const {
|
|
findUserByTargetFactory,
|
|
insertInviteAndDeleteOldFactory,
|
|
deleteServerOnlyInvitesFactory,
|
|
updateAllInviteTargetsFactory
|
|
} = require('@/modules/serverinvites/repositories/serverInvites')
|
|
const {
|
|
collectAndValidateCoreTargetsFactory
|
|
} = require('@/modules/serverinvites/services/coreResourceCollection')
|
|
const {
|
|
buildCoreInviteEmailContentsFactory
|
|
} = require('@/modules/serverinvites/services/coreEmailContents')
|
|
const { getEventBus } = require('@/modules/shared/services/eventBus')
|
|
const { createBranchFactory } = require('@/modules/core/repositories/branches')
|
|
const { ProjectsEmitter } = require('@/modules/core/events/projectsEmitter')
|
|
const {
|
|
getUsersFactory,
|
|
getUserFactory,
|
|
storeUserFactory,
|
|
countAdminUsersFactory,
|
|
storeUserAclFactory
|
|
} = require('@/modules/core/repositories/users')
|
|
const {
|
|
findEmailFactory,
|
|
createUserEmailFactory,
|
|
ensureNoPrimaryEmailForUserFactory
|
|
} = require('@/modules/core/repositories/userEmails')
|
|
const {
|
|
requestNewEmailVerificationFactory
|
|
} = require('@/modules/emails/services/verification/request')
|
|
const {
|
|
deleteOldAndInsertNewVerificationFactory
|
|
} = require('@/modules/emails/repositories')
|
|
const { renderEmail } = require('@/modules/emails/services/emailRendering')
|
|
const { sendEmail } = require('@/modules/emails/services/sending')
|
|
const { createUserFactory } = require('@/modules/core/services/users/management')
|
|
const {
|
|
validateAndCreateUserEmailFactory
|
|
} = require('@/modules/core/services/userEmails')
|
|
const {
|
|
finalizeInvitedServerRegistrationFactory
|
|
} = require('@/modules/serverinvites/services/processing')
|
|
const { UsersEmitter } = require('@/modules/core/events/usersEmitter')
|
|
const { getServerInfoFactory } = require('@/modules/core/repositories/server')
|
|
|
|
const getServerInfo = getServerInfoFactory({ db })
|
|
const getUser = getUserFactory({ db })
|
|
const getUsers = getUsersFactory({ db })
|
|
const getStream = getStreamFactory({ db })
|
|
const createStream = legacyCreateStreamFactory({
|
|
createStreamReturnRecord: createStreamReturnRecordFactory({
|
|
inviteUsersToProject: inviteUsersToProjectFactory({
|
|
createAndSendInvite: createAndSendInviteFactory({
|
|
findUserByTarget: findUserByTargetFactory({ db }),
|
|
insertInviteAndDeleteOld: insertInviteAndDeleteOldFactory({ db }),
|
|
collectAndValidateResourceTargets: collectAndValidateCoreTargetsFactory({
|
|
getStream
|
|
}),
|
|
buildInviteEmailContents: buildCoreInviteEmailContentsFactory({
|
|
getStream
|
|
}),
|
|
emitEvent: ({ eventName, payload }) =>
|
|
getEventBus().emit({
|
|
eventName,
|
|
payload
|
|
}),
|
|
getUser,
|
|
getServerInfo
|
|
}),
|
|
getUsers
|
|
}),
|
|
createStream: createStreamFactory({ db }),
|
|
createBranch: createBranchFactory({ db }),
|
|
projectsEventsEmitter: ProjectsEmitter.emit
|
|
})
|
|
})
|
|
|
|
const findEmail = findEmailFactory({ db })
|
|
const requestNewEmailVerification = requestNewEmailVerificationFactory({
|
|
findEmail,
|
|
getUser: getUserFactory({ db }),
|
|
getServerInfo,
|
|
deleteOldAndInsertNewVerification: deleteOldAndInsertNewVerificationFactory({ db }),
|
|
renderEmail,
|
|
sendEmail
|
|
})
|
|
const createUser = createUserFactory({
|
|
getServerInfo,
|
|
findEmail,
|
|
storeUser: storeUserFactory({ db }),
|
|
countAdminUsers: countAdminUsersFactory({ db }),
|
|
storeUserAcl: storeUserAclFactory({ db }),
|
|
validateAndCreateUserEmail: validateAndCreateUserEmailFactory({
|
|
createUserEmail: createUserEmailFactory({ db }),
|
|
ensureNoPrimaryEmailForUser: ensureNoPrimaryEmailForUserFactory({ db }),
|
|
findEmail,
|
|
updateEmailInvites: finalizeInvitedServerRegistrationFactory({
|
|
deleteServerOnlyInvites: deleteServerOnlyInvitesFactory({ db }),
|
|
updateAllInviteTargets: updateAllInviteTargetsFactory({ db })
|
|
}),
|
|
requestNewEmailVerification
|
|
}),
|
|
usersEventsEmitter: UsersEmitter.emit
|
|
})
|
|
|
|
describe('Generic AuthN & AuthZ controller tests', () => {
|
|
before(async () => {
|
|
await beforeEachContext()
|
|
})
|
|
|
|
it('Validate scopes', async () => {
|
|
await validateScopes()
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) =>
|
|
expect('Your auth token does not have the required scope.').to.equal(
|
|
err.message
|
|
)
|
|
)
|
|
|
|
await validateScopes(['a'], 'b')
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) =>
|
|
expect('Your auth token does not have the required scope: b.').to.equal(
|
|
err.message
|
|
)
|
|
)
|
|
|
|
await validateScopes(['a', 'b'], 'b') // should pass
|
|
})
|
|
;[
|
|
['BS header', { req: { headers: { authorization: 'Bearer BS' } } }],
|
|
['Null header', { req: { headers: { authorization: null } } }],
|
|
['Undefined header', { req: { headers: { authorization: undefined } } }],
|
|
['BS token', { token: 'Bearer BS' }],
|
|
['Null token', { token: null }],
|
|
['Undefined token', { token: undefined }]
|
|
].map(([caseName, contextInput]) =>
|
|
it(`Should create proper context ${caseName}`, async () => {
|
|
const res = await buildContext(contextInput)
|
|
expect(res.auth).to.equal(false)
|
|
})
|
|
)
|
|
|
|
it('Should validate server role', async () => {
|
|
await throwForNotHavingServerRole(
|
|
{ auth: true, role: Roles.Server.User },
|
|
Roles.Server.Admin
|
|
)
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) =>
|
|
expect('You do not have the required server role').to.equal(err.message)
|
|
)
|
|
|
|
await throwForNotHavingServerRole({ auth: true, role: 'HACZOR' }, '133TCR3w')
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) =>
|
|
expect('Invalid role requirement specified').to.equal(err.message)
|
|
)
|
|
|
|
await throwForNotHavingServerRole(
|
|
{ auth: true, role: Roles.Server.Admin },
|
|
'133TCR3w'
|
|
)
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) =>
|
|
expect('Invalid role requirement specified').to.equal(err.message)
|
|
)
|
|
|
|
const test = await throwForNotHavingServerRole(
|
|
{ auth: true, role: Roles.Server.Admin },
|
|
Roles.Server.User
|
|
)
|
|
expect(test).to.equal(true)
|
|
})
|
|
|
|
it('Resolver Authorization Should fail nicely when roles & resources are wanky', async () => {
|
|
await authorizeResolver(null, 'foo', 'bar')
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) => expect('Unknown role: bar').to.equal(err.message))
|
|
|
|
// this caught me out, but streams:read is not a valid role for now
|
|
await authorizeResolver('foo', 'bar', Scopes.Streams.Read)
|
|
.then(() => {
|
|
throw new Error('This should have been rejected')
|
|
})
|
|
.catch((err) => expect('Unknown role: streams:read').to.equal(err.message))
|
|
})
|
|
|
|
describe('Authorize resolver ', () => {
|
|
const myStream = {
|
|
name: 'My Stream 2',
|
|
isPublic: true
|
|
}
|
|
const notMyStream = {
|
|
name: 'Not My Stream 1',
|
|
isPublic: false
|
|
}
|
|
const serverOwner = {
|
|
name: 'Itsa Me',
|
|
email: 'me@gmail.com',
|
|
password: 'sn3aky-1337-b1m'
|
|
}
|
|
const otherGuy = {
|
|
name: 'Some Other DUde',
|
|
email: 'otherguy@gmail.com',
|
|
password: 'sn3aky-1337-b1m'
|
|
}
|
|
|
|
before(async function () {
|
|
// Seeding
|
|
serverOwner.id = await createUser(serverOwner)
|
|
otherGuy.id = await createUser(otherGuy)
|
|
|
|
await Promise.all([
|
|
createStream({ ...myStream, ownerId: serverOwner.id }).then(
|
|
(id) => (myStream.id = id)
|
|
),
|
|
createStream({ ...notMyStream, ownerId: otherGuy.id }).then(
|
|
(id) => (notMyStream.id = id)
|
|
)
|
|
])
|
|
})
|
|
|
|
afterEach(() => {
|
|
envHelperMock.disable()
|
|
})
|
|
after(() => {
|
|
envHelperMock.destroy()
|
|
envHelperMock.resetMockedFunctions()
|
|
})
|
|
it('should allow stream:owners to be stream:owners', async () => {
|
|
await authorizeResolver(
|
|
serverOwner.id,
|
|
myStream.id,
|
|
Roles.Stream.Contributor,
|
|
null
|
|
)
|
|
})
|
|
|
|
it('should get the passed in role for server:admins if override enabled', async () => {
|
|
envHelperMock.enable()
|
|
envHelperMock.mockFunction('adminOverrideEnabled', () => true)
|
|
await authorizeResolver(
|
|
serverOwner.id,
|
|
myStream.id,
|
|
Roles.Stream.Contributor,
|
|
null
|
|
)
|
|
})
|
|
it('should not allow server:admins to be anything if adminOverride is disabled', async () => {
|
|
try {
|
|
await authorizeResolver(
|
|
serverOwner.id,
|
|
notMyStream.id,
|
|
Roles.Stream.Contributor,
|
|
null
|
|
)
|
|
throw 'This should have thrown'
|
|
} catch (e) {
|
|
expect(e instanceof ForbiddenError)
|
|
}
|
|
})
|
|
|
|
it('should allow server:admins to be anything if adminOverride is enabled', async () => {
|
|
envHelperMock.enable()
|
|
envHelperMock.mockFunction('adminOverrideEnabled', () => true)
|
|
|
|
await authorizeResolver(
|
|
serverOwner.id,
|
|
notMyStream.id,
|
|
Roles.Stream.Contributor,
|
|
null
|
|
)
|
|
})
|
|
|
|
it('should not allow server:users to be anything if adminOverride is disabled', async () => {
|
|
try {
|
|
await authorizeResolver(
|
|
otherGuy.id,
|
|
myStream.id,
|
|
Roles.Stream.Contributor,
|
|
null
|
|
)
|
|
throw 'This should have thrown'
|
|
} catch (e) {
|
|
expect(e instanceof ForbiddenError)
|
|
}
|
|
})
|
|
|
|
it('should not allow server:users to be anything if adminOverride is enabled', async () => {
|
|
envHelperMock.enable()
|
|
envHelperMock.mockFunction('adminOverrideEnabled', () => true)
|
|
try {
|
|
await authorizeResolver(
|
|
otherGuy.id,
|
|
myStream.id,
|
|
Roles.Stream.Contributor,
|
|
null
|
|
)
|
|
throw 'This should have thrown'
|
|
} catch (e) {
|
|
expect(e instanceof ForbiddenError)
|
|
}
|
|
})
|
|
})
|
|
})
|