mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-06-10 06:39:01 +02:00
Add audit log entries for project-role-changed and group-role-changed
GitOrigin-RevId: 4c326dd922bede6f218a6d89e4f18c312a9abf98
This commit is contained in:
@@ -112,10 +112,17 @@ async function setCollaboratorInfo(req, res, next) {
|
||||
)
|
||||
}
|
||||
|
||||
const auditInfo = {
|
||||
ipAddress: req.ip,
|
||||
initiatorId: SessionManager.getLoggedInUserId(req.session),
|
||||
}
|
||||
|
||||
await CollaboratorsHandler.promises.setCollaboratorPrivilegeLevel(
|
||||
projectId,
|
||||
userId,
|
||||
privilegeLevel
|
||||
privilegeLevel,
|
||||
{},
|
||||
auditInfo
|
||||
)
|
||||
EditorRealTimeController.emitToRoom(
|
||||
projectId,
|
||||
|
||||
@@ -10,6 +10,7 @@ import CollaboratorsGetter from './CollaboratorsGetter.mjs'
|
||||
import Errors from '../Errors/Errors.js'
|
||||
import TpdsUpdateSender from '../ThirdPartyDataStore/TpdsUpdateSender.mjs'
|
||||
import EditorRealTimeController from '../Editor/EditorRealTimeController.mjs'
|
||||
import ProjectAuditLogHandler from '../Project/ProjectAuditLogHandler.mjs'
|
||||
|
||||
export default {
|
||||
userIsTokenMember: callbackify(userIsTokenMember),
|
||||
@@ -268,7 +269,8 @@ async function setCollaboratorPrivilegeLevel(
|
||||
projectId,
|
||||
userId,
|
||||
privilegeLevel,
|
||||
{ pendingEditor, pendingReviewer } = {}
|
||||
{ pendingEditor, pendingReviewer } = {},
|
||||
auditInfo = {}
|
||||
) {
|
||||
// Make sure we're only updating the project if the user is already a
|
||||
// collaborator
|
||||
@@ -352,6 +354,17 @@ async function setCollaboratorPrivilegeLevel(
|
||||
throw new Errors.NotFoundError('project or collaborator not found')
|
||||
}
|
||||
|
||||
ProjectAuditLogHandler.addEntryInBackground(
|
||||
projectId,
|
||||
'project-role-changed',
|
||||
auditInfo.initiatorId,
|
||||
auditInfo.ipAddress,
|
||||
{
|
||||
userId,
|
||||
role: _privilegeLevelToRole(privilegeLevel),
|
||||
}
|
||||
)
|
||||
|
||||
if (update.$set?.track_changes) {
|
||||
EditorRealTimeController.emitToRoom(
|
||||
projectId,
|
||||
@@ -361,6 +374,19 @@ async function setCollaboratorPrivilegeLevel(
|
||||
}
|
||||
}
|
||||
|
||||
function _privilegeLevelToRole(privilegeLevel) {
|
||||
switch (privilegeLevel) {
|
||||
case 'readOnly':
|
||||
return 'Viewer'
|
||||
case 'readAndWrite':
|
||||
return 'Editor'
|
||||
case 'review':
|
||||
return 'Reviewer'
|
||||
default:
|
||||
return privilegeLevel
|
||||
}
|
||||
}
|
||||
|
||||
async function userIsTokenMember(userId, projectId) {
|
||||
if (!userId) {
|
||||
return false
|
||||
|
||||
@@ -15,6 +15,11 @@ const JSON_ESCAPE = {
|
||||
'\u2029': '\\u2029',
|
||||
}
|
||||
|
||||
export function capitalise(str) {
|
||||
if (!str || str.length === 0 || typeof str !== 'string') return str
|
||||
return str.substring(0, 1).toUpperCase() + str.slice(1)
|
||||
}
|
||||
|
||||
/**
|
||||
* Converts a snake_case string into a user friendly string with each word capitalized.
|
||||
* @param {string} snakecaseStr
|
||||
|
||||
@@ -15,6 +15,7 @@ const MANAGED_GROUP_PROJECT_EVENTS = [
|
||||
'project-untrashed',
|
||||
'project-restored',
|
||||
'project-cloned',
|
||||
'project-role-changed',
|
||||
'project-history-version-restored',
|
||||
'project-history-version-downloaded',
|
||||
'transfer-ownership',
|
||||
|
||||
@@ -542,13 +542,17 @@ async function moveReadWriteToCollaborators(req, res, next) {
|
||||
userId,
|
||||
projectId
|
||||
)
|
||||
|
||||
const auditInfo = { ipAddress: req.ip, initiatorId: userId }
|
||||
|
||||
await CollaboratorsHandler.promises.setCollaboratorPrivilegeLevel(
|
||||
projectId,
|
||||
userId,
|
||||
pendingEditor
|
||||
? PrivilegeLevels.READ_ONLY
|
||||
: PrivilegeLevels.READ_AND_WRITE,
|
||||
{ pendingEditor }
|
||||
{ pendingEditor },
|
||||
auditInfo
|
||||
)
|
||||
} else {
|
||||
// Normal case, not invited, joining via link sharing
|
||||
|
||||
@@ -202,10 +202,15 @@ async function add(req, res) {
|
||||
}
|
||||
let user
|
||||
try {
|
||||
const auditInfo = {
|
||||
ipAddress: req.ip,
|
||||
initiatorId: SessionManager.getLoggedInUserId(req.session),
|
||||
}
|
||||
user = await UserMembershipHandler.promises.addUser(
|
||||
entity,
|
||||
entityConfig,
|
||||
email
|
||||
email,
|
||||
auditInfo
|
||||
)
|
||||
} catch (err) {
|
||||
if (err instanceof UserMembershipErrors.UserAlreadyAddedError) {
|
||||
@@ -245,10 +250,15 @@ async function remove(req, res) {
|
||||
})
|
||||
}
|
||||
try {
|
||||
const auditInfo = {
|
||||
ipAddress: req.ip,
|
||||
initiatorId: loggedInUserId,
|
||||
}
|
||||
await UserMembershipHandler.promises.removeUser(
|
||||
entity,
|
||||
entityConfig,
|
||||
userId
|
||||
userId,
|
||||
auditInfo
|
||||
)
|
||||
} catch (err) {
|
||||
if (err instanceof UserMembershipErrors.UserIsManagerError) {
|
||||
|
||||
@@ -6,6 +6,8 @@ import { Publisher } from '../../models/Publisher.mjs'
|
||||
import UserMembershipViewModel from './UserMembershipViewModel.mjs'
|
||||
import UserGetter from '../User/UserGetter.mjs'
|
||||
import UserMembershipErrors from './UserMembershipErrors.mjs'
|
||||
import Modules from '../../infrastructure/Modules.mjs'
|
||||
import mongoose from '../../infrastructure/Mongoose.mjs'
|
||||
|
||||
const { ObjectId } = mongodb
|
||||
|
||||
@@ -27,7 +29,7 @@ const UserMembershipHandler = {
|
||||
return await getPopulatedListOfMembers(entity, attributes)
|
||||
},
|
||||
|
||||
async addUser(entity, entityConfig, email) {
|
||||
async addUser(entity, entityConfig, email, auditInfo) {
|
||||
const attribute = entityConfig.fields.write
|
||||
const user = await UserGetter.promises.getUserByAnyEmail(email)
|
||||
|
||||
@@ -39,15 +41,63 @@ const UserMembershipHandler = {
|
||||
throw new UserMembershipErrors.UserAlreadyAddedError()
|
||||
}
|
||||
|
||||
// if the entity is a Subscription with managed users enabled, then audit log the event
|
||||
if (
|
||||
entityConfig.modelName === 'Subscription' &&
|
||||
entity.managedUsersEnabled
|
||||
) {
|
||||
const session = await mongoose.startSession()
|
||||
try {
|
||||
await session.withTransaction(async () => {
|
||||
const auditLog = {
|
||||
groupId: entity._id,
|
||||
operation: 'group-role-changed',
|
||||
initiatorId: auditInfo.initiatorId,
|
||||
ipAddress: auditInfo.ipAddress,
|
||||
info: {
|
||||
userId: user._id,
|
||||
role: 'manager',
|
||||
},
|
||||
}
|
||||
await Modules.promises.hooks.fire(
|
||||
'addGroupAuditLogEntry',
|
||||
auditLog,
|
||||
session
|
||||
)
|
||||
})
|
||||
} finally {
|
||||
await session.endSession()
|
||||
}
|
||||
}
|
||||
|
||||
await addUserToEntity(entity, attribute, user)
|
||||
return UserMembershipViewModel.build(user)
|
||||
},
|
||||
|
||||
async removeUser(entity, entityConfig, userId) {
|
||||
async removeUser(entity, entityConfig, userId, auditInfo) {
|
||||
const attribute = entityConfig.fields.write
|
||||
if (entity.admin_id ? entity.admin_id.equals(userId) : undefined) {
|
||||
throw new UserMembershipErrors.UserIsManagerError()
|
||||
}
|
||||
|
||||
// if the entity is a Subscription with managed users enabled, then audit log the event
|
||||
if (
|
||||
entityConfig.modelName === 'Subscription' &&
|
||||
entity.managedUsersEnabled
|
||||
) {
|
||||
const auditLog = {
|
||||
groupId: entity._id,
|
||||
operation: 'group-role-changed',
|
||||
initiatorId: auditInfo.initiatorId,
|
||||
ipAddress: auditInfo.ipAddress,
|
||||
info: {
|
||||
userId,
|
||||
role: 'member',
|
||||
},
|
||||
}
|
||||
await Modules.promises.hooks.fire('addGroupAuditLogEntry', auditLog)
|
||||
}
|
||||
|
||||
return await removeUserFromEntity(entity, attribute, userId)
|
||||
},
|
||||
}
|
||||
|
||||
@@ -104,6 +104,17 @@ describe('CollaboratorsHandler', function () {
|
||||
})
|
||||
)
|
||||
|
||||
ctx.ProjectAuditLogHandler = {
|
||||
addEntryInBackground: sinon.stub(),
|
||||
}
|
||||
|
||||
vi.doMock(
|
||||
'../../../../app/src/Features/Project/ProjectAuditLogHandler',
|
||||
() => ({
|
||||
default: ctx.ProjectAuditLogHandler,
|
||||
})
|
||||
)
|
||||
|
||||
ctx.CollaboratorsHandler = (await import(MODULE_PATH)).default
|
||||
})
|
||||
|
||||
@@ -812,5 +823,29 @@ describe('CollaboratorsHandler', function () {
|
||||
)
|
||||
).to.be.rejectedWith(Errors.NotFoundError)
|
||||
})
|
||||
|
||||
it('should write a project audit log', async function (ctx) {
|
||||
ctx.ProjectMock.expects('updateOne')
|
||||
.chain('exec')
|
||||
.resolves({ matchedCount: 1 })
|
||||
const auditInfo = {
|
||||
initiatorId: new ObjectId(),
|
||||
ipAddress: '192.168.1.1',
|
||||
}
|
||||
await ctx.CollaboratorsHandler.promises.setCollaboratorPrivilegeLevel(
|
||||
ctx.project._id,
|
||||
ctx.userId,
|
||||
'readOnly',
|
||||
{},
|
||||
auditInfo
|
||||
)
|
||||
ctx.ProjectAuditLogHandler.addEntryInBackground.should.have.been.calledWith(
|
||||
ctx.project._id,
|
||||
'project-role-changed',
|
||||
auditInfo.initiatorId,
|
||||
auditInfo.ipAddress,
|
||||
{ userId: ctx.userId, role: 'Viewer' }
|
||||
)
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
@@ -27,6 +27,7 @@ describe('UserMembershipController', () => {
|
||||
recurlySubscriptionId = 'mock-recurly-subscription-id'
|
||||
ctx.req = new MockRequest(vi)
|
||||
ctx.req.params.id = 'mock-entity-id'
|
||||
ctx.req.ip = '1.2.3.4'
|
||||
ctx.user = { _id: 'mock-user-id' }
|
||||
ctx.newUser = { _id: 'mock-new-user-id', email: 'new-user-email@foo.bar' }
|
||||
ctx.subscription = {
|
||||
@@ -495,7 +496,8 @@ describe('UserMembershipController', () => {
|
||||
write: 'manager_ids',
|
||||
},
|
||||
},
|
||||
newUser.email
|
||||
newUser.email,
|
||||
{ initiatorId: 'mock-user-id', ipAddress: '1.2.3.4' }
|
||||
)
|
||||
},
|
||||
})
|
||||
@@ -613,7 +615,8 @@ describe('UserMembershipController', () => {
|
||||
write: 'manager_ids',
|
||||
},
|
||||
},
|
||||
newUser._id
|
||||
newUser._id,
|
||||
{ initiatorId: 'mock-user-id', ipAddress: '1.2.3.4' }
|
||||
)
|
||||
},
|
||||
})
|
||||
|
||||
@@ -35,6 +35,9 @@ describe('UserMembershipHandler', function () {
|
||||
update: vi.fn().mockReturnValue({
|
||||
exec: vi.fn().mockResolvedValue(),
|
||||
}),
|
||||
updateOne: vi.fn().mockReturnValue({
|
||||
exec: vi.fn().mockResolvedValue(),
|
||||
}),
|
||||
}
|
||||
ctx.institution = {
|
||||
_id: 'mock-institution-id',
|
||||
@@ -83,6 +86,21 @@ describe('UserMembershipHandler', function () {
|
||||
}),
|
||||
}
|
||||
|
||||
ctx.Modules = {
|
||||
promises: {
|
||||
hooks: {
|
||||
fire: vi.fn().mockResolvedValue(),
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
ctx.mongoose = {
|
||||
startSession: vi.fn().mockResolvedValue({
|
||||
withTransaction: vi.fn(async callback => await callback()),
|
||||
endSession: vi.fn().mockResolvedValue(),
|
||||
}),
|
||||
}
|
||||
|
||||
vi.doMock('mongodb-legacy', () => ({
|
||||
default: { ObjectId },
|
||||
}))
|
||||
@@ -110,6 +128,14 @@ describe('UserMembershipHandler', function () {
|
||||
Publisher: ctx.Publisher,
|
||||
}))
|
||||
|
||||
vi.doMock('../../../../app/src/infrastructure/Modules', () => ({
|
||||
default: ctx.Modules,
|
||||
}))
|
||||
|
||||
vi.doMock('../../../../app/src/infrastructure/Mongoose', () => ({
|
||||
default: ctx.mongoose,
|
||||
}))
|
||||
|
||||
ctx.UserMembershipHandler = (await import(modulePath)).default
|
||||
})
|
||||
|
||||
@@ -254,6 +280,66 @@ describe('UserMembershipHandler', function () {
|
||||
expect(user).to.equal(ctx.newUser)
|
||||
})
|
||||
})
|
||||
|
||||
describe('group managers', function () {
|
||||
it('add user to group managers', async function (ctx) {
|
||||
await ctx.UserMembershipHandler.promises.addUser(
|
||||
ctx.subscription,
|
||||
EntityConfigs.groupManagers,
|
||||
ctx.email
|
||||
)
|
||||
expect(ctx.subscription.updateOne).toHaveBeenCalledWith({
|
||||
$addToSet: { manager_ids: ctx.newUser._id },
|
||||
})
|
||||
})
|
||||
|
||||
it('should write a group audit log when subscription has managed users enabled', async function (ctx) {
|
||||
ctx.subscription.managedUsersEnabled = true
|
||||
const auditInfo = {
|
||||
initiatorId: new ObjectId(),
|
||||
ipAddress: '192.168.1.1',
|
||||
}
|
||||
|
||||
await ctx.UserMembershipHandler.promises.addUser(
|
||||
ctx.subscription,
|
||||
EntityConfigs.groupManagers,
|
||||
ctx.email,
|
||||
auditInfo
|
||||
)
|
||||
|
||||
expect(ctx.Modules.promises.hooks.fire).toHaveBeenCalledWith(
|
||||
'addGroupAuditLogEntry',
|
||||
{
|
||||
groupId: ctx.subscription._id,
|
||||
operation: 'group-role-changed',
|
||||
initiatorId: auditInfo.initiatorId,
|
||||
ipAddress: auditInfo.ipAddress,
|
||||
info: {
|
||||
userId: ctx.newUser._id,
|
||||
role: 'manager',
|
||||
},
|
||||
},
|
||||
expect.anything() // session object
|
||||
)
|
||||
})
|
||||
|
||||
it('should not write a group audit log when subscription does not have managed users enabled', async function (ctx) {
|
||||
ctx.subscription.managedUsersEnabled = false
|
||||
const auditInfo = {
|
||||
initiatorId: new ObjectId(),
|
||||
ipAddress: '192.168.1.1',
|
||||
}
|
||||
|
||||
await ctx.UserMembershipHandler.promises.addUser(
|
||||
ctx.subscription,
|
||||
EntityConfigs.groupManagers,
|
||||
ctx.email,
|
||||
auditInfo
|
||||
)
|
||||
|
||||
expect(ctx.Modules.promises.hooks.fire).not.toHaveBeenCalled()
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
describe('removeUser', function () {
|
||||
@@ -283,5 +369,64 @@ describe('UserMembershipHandler', function () {
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
describe('group managers', function () {
|
||||
it('remove user from group managers', async function (ctx) {
|
||||
await ctx.UserMembershipHandler.promises.removeUser(
|
||||
ctx.subscription,
|
||||
EntityConfigs.groupManagers,
|
||||
ctx.newUser._id
|
||||
)
|
||||
expect(ctx.subscription.updateOne).toHaveBeenCalledWith({
|
||||
$pull: { manager_ids: ctx.newUser._id },
|
||||
})
|
||||
})
|
||||
|
||||
it('should write a group audit log when subscription has managed users enabled', async function (ctx) {
|
||||
ctx.subscription.managedUsersEnabled = true
|
||||
const auditInfo = {
|
||||
initiatorId: new ObjectId(),
|
||||
ipAddress: '192.168.1.1',
|
||||
}
|
||||
|
||||
await ctx.UserMembershipHandler.promises.removeUser(
|
||||
ctx.subscription,
|
||||
EntityConfigs.groupManagers,
|
||||
ctx.newUser._id,
|
||||
auditInfo
|
||||
)
|
||||
|
||||
expect(ctx.Modules.promises.hooks.fire).toHaveBeenCalledWith(
|
||||
'addGroupAuditLogEntry',
|
||||
{
|
||||
groupId: ctx.subscription._id,
|
||||
operation: 'group-role-changed',
|
||||
initiatorId: auditInfo.initiatorId,
|
||||
ipAddress: auditInfo.ipAddress,
|
||||
info: {
|
||||
userId: ctx.newUser._id,
|
||||
role: 'member',
|
||||
},
|
||||
}
|
||||
)
|
||||
})
|
||||
|
||||
it('should not write a group audit log when subscription does not have managed users enabled', async function (ctx) {
|
||||
ctx.subscription.managedUsersEnabled = false
|
||||
const auditInfo = {
|
||||
initiatorId: new ObjectId(),
|
||||
ipAddress: '192.168.1.1',
|
||||
}
|
||||
|
||||
await ctx.UserMembershipHandler.promises.removeUser(
|
||||
ctx.subscription,
|
||||
EntityConfigs.groupManagers,
|
||||
ctx.newUser._id,
|
||||
auditInfo
|
||||
)
|
||||
|
||||
expect(ctx.Modules.promises.hooks.fire).not.toHaveBeenCalled()
|
||||
})
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
Reference in New Issue
Block a user