Skip to content

Commit 035065c

Browse files
[FEATURE] Enregistrer la date de dernière connexion aux orgas & cdc seulement sur les memberships actifs (PIX-16990)
#11641
2 parents 2442c9e + 5efbf21 commit 035065c

26 files changed

+229
-85
lines changed

api/src/shared/domain/models/Membership.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -6,12 +6,13 @@ const roles = {
66
};
77

88
class Membership {
9-
constructor({ id, organizationRole = roles.MEMBER, updatedByUserId, organization, user } = {}) {
9+
constructor({ id, organizationRole = roles.MEMBER, updatedByUserId, organization, user, disabledAt } = {}) {
1010
this.id = id;
1111
this.organizationRole = organizationRole;
1212
this.updatedByUserId = updatedByUserId;
1313
this.organization = organization;
1414
this.user = user;
15+
this.disabledAt = disabledAt;
1516
}
1617

1718
get isAdmin() {

api/src/team/application/certification-center-membership/certification-center-membership.controller.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -72,11 +72,11 @@ const updateReferer = async function (request, h) {
7272

7373
const updateLastAccessedAt = async function (request, h, dependencies = { requestResponseUtils }) {
7474
const userId = dependencies.requestResponseUtils.extractUserIdFromRequest(request);
75-
const certificationCenterId = request.params.certificationCenterId;
75+
const certificationCenterMembershipId = request.params.certificationCenterMembershipId;
7676

7777
await usecases.updateCertificationCenterMembershipLastAccessedAt({
7878
userId,
79-
certificationCenterId,
79+
certificationCenterMembershipId,
8080
});
8181

8282
return h.response().code(204);

api/src/team/application/certification-center-membership/certification-center-membership.route.js

+3-9
Original file line numberDiff line numberDiff line change
@@ -93,18 +93,12 @@ export const certificationCenterMembershipRoute = [
9393
},
9494
},
9595
{
96-
method: 'PATCH',
97-
path: '/api/certification-centers/{certificationCenterId}/certification-center-memberships/me',
96+
method: 'POST',
97+
path: '/api/certification-center-memberships/{certificationCenterMembershipId}/access',
9898
config: {
99-
pre: [
100-
{
101-
method: (request, h) => securityPreHandlers.checkUserIsMemberOfCertificationCenter(request, h),
102-
assign: 'isMemberOfCertificationCenter',
103-
},
104-
],
10599
validate: {
106100
params: Joi.object({
107-
certificationCenterId: identifiersType.certificationCenterId,
101+
certificationCenterMembershipId: identifiersType.certificationCenterMembershipId,
108102
}),
109103
},
110104
handler: (request, h) => certificationCenterMembershipController.updateLastAccessedAt(request, h),

api/src/team/application/membership/membership.controller.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -61,11 +61,11 @@ const findPaginatedFilteredMemberships = async function (request) {
6161

6262
async function updateLastAccessedAt(request, h, dependencies = { requestResponseUtils }) {
6363
const userId = dependencies.requestResponseUtils.extractUserIdFromRequest(request);
64-
const organizationId = request.params.organizationId;
64+
const membershipId = request.params.membershipId;
6565

6666
await usecases.updateMembershipLastAccessedAt({
6767
userId,
68-
organizationId,
68+
membershipId,
6969
});
7070

7171
return h.response().code(204);

api/src/team/application/membership/membership.route.js

+3-9
Original file line numberDiff line numberDiff line change
@@ -121,18 +121,12 @@ export const membershipRoutes = [
121121
},
122122
},
123123
{
124-
method: 'PATCH',
125-
path: '/api/organizations/{organizationId}/me',
124+
method: 'POST',
125+
path: '/api/memberships/{membershipId}/access',
126126
config: {
127-
pre: [
128-
{
129-
method: (request, h) => securityPreHandlers.checkUserBelongsToOrganization(request, h),
130-
assign: 'belongsToOrganization',
131-
},
132-
],
133127
validate: {
134128
params: Joi.object({
135-
organizationId: identifiersType.organizationId,
129+
membershipId: identifiersType.membershipId,
136130
}),
137131
},
138132
handler: (request, h) => membershipController.updateLastAccessedAt(request, h),

api/src/team/domain/usecases/update-certification-center-membership-last-accessed-at.usecase.js

+12-3
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,27 @@
1+
import { ForbiddenError } from '../../../shared/application/http-errors.js';
2+
13
/**
24
* @param {Object} params
35
* @param {string} params.userId
4-
* @param {string} params.certificationCenterId
6+
* @param {string} params.certificationCenterMembershipId
57
* @param {CertificationCenterMembershipRepository} params.certificationCenterMembershipRepository
68
*/
79
const updateCertificationCenterMembershipLastAccessedAt = async function ({
810
userId,
9-
certificationCenterId,
11+
certificationCenterMembershipId,
1012
certificationCenterMembershipRepository,
1113
}) {
14+
const certificationCenterMembership = await certificationCenterMembershipRepository.findById(
15+
certificationCenterMembershipId,
16+
);
17+
if (certificationCenterMembership.user.id !== userId || certificationCenterMembership.disabledAt) {
18+
throw new ForbiddenError();
19+
}
20+
1221
return certificationCenterMembershipRepository.updateLastAccessedAt({
1322
lastAccessedAt: new Date(),
1423
userId,
15-
certificationCenterId,
24+
certificationCenterMembershipId,
1625
});
1726
};
1827

api/src/team/domain/usecases/update-membership-last-accessed-at.usecase.js

+10-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,19 @@
1+
import { ForbiddenError } from '../../../shared/application/http-errors.js';
2+
13
/**
24
* @param {Object} params
35
* @param {string} params.userId
4-
* @param {string} params.organizationId
6+
* @param {string} params.membershipId
57
* @param {MembershipRepository} params.membershipRepository
68
*/
7-
const updateMembershipLastAccessedAt = async function ({ userId, organizationId, membershipRepository }) {
9+
const updateMembershipLastAccessedAt = async function ({ userId, membershipId, membershipRepository }) {
10+
const membership = await membershipRepository.get(membershipId);
11+
if (membership.user.id !== userId || membership.disabledAt) {
12+
throw new ForbiddenError();
13+
}
14+
815
return membershipRepository.updateLastAccessedAt({
9-
userId,
10-
organizationId,
16+
membershipId,
1117
lastAccessedAt: new Date(),
1218
});
1319
};

api/src/team/infrastructure/repositories/certification-center-membership.repository.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ function _toDomain(certificationCenterMembershipDTO) {
4545
updatedAt: certificationCenterMembershipDTO.updatedAt,
4646
role: certificationCenterMembershipDTO.role,
4747
lastAccessedAt: certificationCenterMembershipDTO.lastAccessedAt,
48+
disabledAt: certificationCenterMembershipDTO.disabledAt,
4849
});
4950
}
5051

@@ -288,11 +289,11 @@ const update = async function (certificationCenterMembership) {
288289
await knex(CERTIFICATION_CENTER_MEMBERSHIP_TABLE_NAME).update(data).where({ id: certificationCenterMembership.id });
289290
};
290291

291-
const updateLastAccessedAt = async function ({ lastAccessedAt, userId, certificationCenterId }) {
292+
const updateLastAccessedAt = async function ({ lastAccessedAt, certificationCenterMembershipId }) {
292293
const knexConnection = DomainTransaction.getConnection();
293294

294295
await knexConnection(CERTIFICATION_CENTER_MEMBERSHIP_TABLE_NAME)
295-
.where({ userId, certificationCenterId })
296+
.where({ id: certificationCenterMembershipId })
296297
.update({ lastAccessedAt, updatedAt: lastAccessedAt });
297298
};
298299

api/src/team/infrastructure/repositories/membership.repository.js

+2-4
Original file line numberDiff line numberDiff line change
@@ -177,12 +177,10 @@ export const disableMembershipsByUserId = async function ({ userId, updatedByUse
177177
.update({ disabledAt: new Date(), updatedAt: new Date(), updatedByUserId });
178178
};
179179

180-
export const updateLastAccessedAt = async function ({ userId, organizationId, lastAccessedAt }) {
180+
export const updateLastAccessedAt = async function ({ membershipId, lastAccessedAt }) {
181181
const knexConnection = DomainTransaction.getConnection();
182182

183-
await knexConnection(MEMBERSHIPS_TABLE)
184-
.where({ userId, organizationId })
185-
.update({ lastAccessedAt, updatedAt: new Date() });
183+
await knexConnection(MEMBERSHIPS_TABLE).where({ id: membershipId }).update({ lastAccessedAt, updatedAt: new Date() });
186184
};
187185

188186
const toDomain = (membershipData, userData = null, organizationData = null, organizationTags = null) => {

api/tests/team/acceptance/application/certification-center-membership/certification-center-membership.route.test.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -193,15 +193,15 @@ describe('Acceptance | Team | Application | Routes | certification-center-member
193193
const certificationCenterId = databaseBuilder.factory.buildCertificationCenter().id;
194194

195195
const userId = databaseBuilder.factory.buildUser().id;
196-
databaseBuilder.factory.buildCertificationCenterMembership({
196+
const certificationCenterMembershipId = databaseBuilder.factory.buildCertificationCenterMembership({
197197
certificationCenterId,
198198
userId,
199-
});
199+
}).id;
200200

201201
await databaseBuilder.commit();
202202
const request = {
203-
method: 'PATCH',
204-
url: `/api/certification-centers/${certificationCenterId}/certification-center-memberships/me`,
203+
method: 'POST',
204+
url: `/api/certification-center-memberships/${certificationCenterMembershipId}/access`,
205205
payload: {},
206206
headers: generateAuthenticatedUserRequestHeaders({ userId }),
207207
};

api/tests/team/acceptance/application/membership/membership.route.test.js

+6-5
Original file line numberDiff line numberDiff line change
@@ -407,25 +407,26 @@ describe('Acceptance | Team | Application | Route | membership', function () {
407407
});
408408
});
409409

410-
describe('PATCH /api/organizations/{id}/me', function () {
410+
describe('POST /api/memberships/{id}/access', function () {
411411
context('when user is one of the members of the organization', function () {
412412
it('updates user membership lastAccessedAt', async function () {
413413
// given
414414
const organizationId = databaseBuilder.factory.buildOrganization().id;
415415
const organizationMemberUserId = databaseBuilder.factory.buildUser().id;
416-
databaseBuilder.factory.buildMembership({
416+
const membershipId = databaseBuilder.factory.buildMembership({
417417
userId: organizationMemberUserId,
418418
organizationId,
419419
organizationRole: Membership.roles.MEMBER,
420420
lastAccessedAt: null,
421421
updatedAt: new Date('2020-01-01'),
422-
});
422+
disabledAt: null,
423+
}).id;
423424

424425
await databaseBuilder.commit();
425426

426427
const options = {
427-
method: 'PATCH',
428-
url: `/api/organizations/${organizationId}/me`,
428+
method: 'POST',
429+
url: `/api/memberships/${membershipId}/access`,
429430
payload: {},
430431
headers: generateAuthenticatedUserRequestHeaders({ userId: organizationMemberUserId }),
431432
};

api/tests/team/integration/application/membership/membership.controller.test.js

+6-6
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import { ForbiddenError } from '../../../../../src/shared/application/http-errors.js';
12
import { securityPreHandlers } from '../../../../../src/shared/application/security-pre-handlers.js';
23
import { InvalidMembershipOrganizationRoleError } from '../../../../../src/shared/domain/errors.js';
34
import { Membership } from '../../../../../src/shared/domain/models/index.js';
@@ -322,16 +323,15 @@ describe('Integration | Team | Application | Memberships | membership-controller
322323

323324
describe('#updateLastAccessedAt', function () {
324325
context('Error case', function () {
325-
describe('when user does not belong to organization', function () {
326+
describe('when user does not own the membership', function () {
326327
it('returns an HTTP response with status code 403', async function () {
327328
// given
328-
const organizationId = 1234;
329+
const membershipId = 1234;
329330
const userId = 5678;
330-
sinon
331-
.stub(securityPreHandlers, 'checkUserBelongsToOrganization')
332-
.callsFake((request, h) => h.response().code(403).takeover());
331+
sinon.stub(usecases, 'updateMembershipLastAccessedAt').throws(new ForbiddenError());
332+
333333
// when
334-
const response = await httpTestServer.request('PATCH', `/api/organizations/${organizationId}/me`, {
334+
const response = await httpTestServer.request('POST', `/api/memberships/${membershipId}/access`, {
335335
auth: { credentials: { userId } },
336336
});
337337

api/tests/team/integration/domain/usecases/update-certification-center-membership-last-accessed-at.usecase.test.js

+72-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,76 @@
1+
import { ForbiddenError } from '../../../../../src/shared/application/http-errors.js';
2+
import { NotFoundError } from '../../../../../src/shared/domain/errors.js';
13
import { CERTIFICATION_CENTER_MEMBERSHIP_ROLES } from '../../../../../src/shared/domain/models/CertificationCenterMembership.js';
24
import { usecases } from '../../../../../src/team/domain/usecases/index.js';
35
import { certificationCenterMembershipRepository } from '../../../../../src/team/infrastructure/repositories/certification-center-membership.repository.js';
4-
import { databaseBuilder, expect, knex, sinon } from '../../../../test-helper.js';
6+
import { catchErr, databaseBuilder, expect, knex, sinon } from '../../../../test-helper.js';
57

68
describe('Integration | Team | UseCases | update-certification-center-membership-last-accessed-at', function () {
9+
context('when the certification center membership does not exist', function () {
10+
it('throws a NotFoundError', async function () {
11+
// given
12+
const certificationCenterMembershipId = 4567;
13+
const userId = 1234;
14+
15+
const error = await catchErr(usecases.updateCertificationCenterMembershipLastAccessedAt)({
16+
userId,
17+
certificationCenterMembershipId,
18+
certificationCenterMembershipRepository,
19+
});
20+
21+
expect(error).to.be.instanceOf(NotFoundError);
22+
});
23+
});
24+
25+
context('when the user is not the owner of the certification center membership', function () {
26+
it('throws a ForbiddenError', async function () {
27+
// given
28+
const certificationCenterId = databaseBuilder.factory.buildCertificationCenter().id;
29+
const userId = databaseBuilder.factory.buildUser().id;
30+
31+
const certificationCenterMembershipId = databaseBuilder.factory.buildCertificationCenterMembership({
32+
certificationCenterId,
33+
userId,
34+
certificationCenterRole: CERTIFICATION_CENTER_MEMBERSHIP_ROLES.MEMBER,
35+
}).id;
36+
37+
await databaseBuilder.commit();
38+
39+
const error = await catchErr(usecases.updateCertificationCenterMembershipLastAccessedAt)({
40+
userId: userId + 1,
41+
certificationCenterMembershipId,
42+
certificationCenterMembershipRepository,
43+
});
44+
45+
expect(error).to.be.instanceOf(ForbiddenError);
46+
});
47+
});
48+
49+
context('when the certification center membership is disabled', function () {
50+
it('throws a ForbiddenError', async function () {
51+
// given
52+
const certificationCenterId = databaseBuilder.factory.buildCertificationCenter().id;
53+
const userId = databaseBuilder.factory.buildUser().id;
54+
55+
const certificationCenterMembershipId = databaseBuilder.factory.buildCertificationCenterMembership({
56+
certificationCenterId,
57+
userId,
58+
certificationCenterRole: CERTIFICATION_CENTER_MEMBERSHIP_ROLES.MEMBER,
59+
disabledAt: new Date(),
60+
}).id;
61+
62+
await databaseBuilder.commit();
63+
64+
const error = await catchErr(usecases.updateCertificationCenterMembershipLastAccessedAt)({
65+
userId,
66+
certificationCenterMembershipId,
67+
certificationCenterMembershipRepository,
68+
});
69+
70+
expect(error).to.be.instanceOf(ForbiddenError);
71+
});
72+
});
73+
774
it('updates certification center membership lastAccessedAt', async function () {
875
// given
976
const now = new Date('2021-01-02');
@@ -12,19 +79,20 @@ describe('Integration | Team | UseCases | update-certification-center-membership
1279
const certificationCenterId = databaseBuilder.factory.buildCertificationCenter().id;
1380
const userId = databaseBuilder.factory.buildUser().id;
1481

15-
databaseBuilder.factory.buildCertificationCenterMembership({
82+
const certificationCenterMembershipId = databaseBuilder.factory.buildCertificationCenterMembership({
1683
certificationCenterId,
1784
userId,
1885
certificationCenterRole: CERTIFICATION_CENTER_MEMBERSHIP_ROLES.MEMBER,
1986
lastAccessedAt: null,
20-
});
87+
disabledAt: null,
88+
}).id;
2189

2290
await databaseBuilder.commit();
2391

2492
// when
2593
await usecases.updateCertificationCenterMembershipLastAccessedAt({
2694
userId,
27-
certificationCenterId,
95+
certificationCenterMembershipId,
2896
certificationCenterMembershipRepository,
2997
});
3098

0 commit comments

Comments
 (0)