Skip to content

Commit eea0294

Browse files
[FEATURE] S'assurer que les images de Modules de prod respectent les contraintes tech (PIX-17215)
#11842
2 parents cb28d5f + a706046 commit eea0294

File tree

13 files changed

+182
-25
lines changed

13 files changed

+182
-25
lines changed

api/src/devcomp/domain/models/element/Image.js

+20-6
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,28 @@
1+
import { DomainError } from '../../../../shared/domain/errors.js';
12
import { assertNotNullOrUndefined } from '../../../../shared/domain/models/asserts.js';
23
import { Element } from './Element.js';
34

45
class Image extends Element {
6+
static #VALID_PRODUCTION_HOSTNAME = 'assets.pix.org';
7+
58
/**
69
* @param{object} params
710
* @param{string} params.id
8-
* @param{string} url
9-
* @param{string} alt
10-
* @param{string} alternativeText
11-
* @param{string} legend
12-
* @param{string} licence
11+
* @param{string} params.url
12+
* @param{string} params.alt
13+
* @param{string} params.alternativeText
14+
* @param{string} params.legend
15+
* @param{string} params.licence
16+
* @param{boolean} params.isBeta
1317
*/
14-
constructor({ id, url, alt, alternativeText, legend, licence }) {
18+
constructor({ id, url, alt, alternativeText, legend, licence, isBeta = true }) {
1519
super({ id, type: 'image' });
1620

1721
assertNotNullOrUndefined(url, 'The URL is required for an image');
22+
if (!URL.canParse(url)) {
23+
throw new DomainError('The URL must be a valid URL for an image');
24+
}
25+
1826
assertNotNullOrUndefined(alt, 'The alt text is required for an image');
1927
assertNotNullOrUndefined(alternativeText, 'The alternative text is required for an image');
2028

@@ -23,6 +31,12 @@ class Image extends Element {
2331
this.alternativeText = alternativeText;
2432
this.legend = legend;
2533
this.licence = licence;
34+
35+
if (!isBeta) {
36+
if (URL.parse(url).hostname !== Image.#VALID_PRODUCTION_HOSTNAME) {
37+
throw new DomainError('The image URL must be from "assets.pix.org" when module is production ready');
38+
}
39+
}
2640
}
2741
}
2842

api/src/devcomp/infrastructure/datasources/learning-content/modules/bases-clavier-ordinateur-1.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "654c44dc-0560-4acc-9860-4a67c923577f",
33
"slug": "bases-clavier-1",
44
"title": "Les bases du clavier sur ordinateur 1/2",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>Dans ce module, découvrez ce qu’est un clavier et à quoi il sert. <span aria-hidden=\"true\">⌨️</span><br>Vous apprendrez à taper des mots simples et à modifier du texte.<br>Pour ce module, vous devez déjà savoir utiliser une souris d’ordinateur.</p>",

api/src/devcomp/infrastructure/datasources/learning-content/modules/bases-clavier-ordinateur-2.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "bb0a4ed3-1b49-4782-b867-05ade0868c4f",
33
"slug": "bases-clavier-2",
44
"title": "Les bases du clavier sur ordinateur 2/2",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>Pour taper certains caractères, il suffit d’appuyer une fois sur une touche. Pour d’autres c’est un peu plus compliqué ! <br>Dans ce module, vous apprendrez à taper des caractères en appuyant sur deux touches en même temps : les majuscules ou la ponctuation par exemple.</p>",

api/src/devcomp/infrastructure/datasources/learning-content/modules/ports-connexions-essentiels.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "12cd102f-9831-4264-8d5e-15a8f177594a",
33
"slug": "ports-connexions-essentiels",
44
"title": "Les ports de connexion d’un ordinateur",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>Pour savoir ce qu'on peut brancher ou non à un ordinateur, il faut bien connaître les principaux ports de connexion !</p>",

api/src/devcomp/infrastructure/datasources/learning-content/modules/tmp08ef.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "08ef1a47-b691-4138-b899-39f3512fa152",
33
"slug": "tmp08ef",
44
"title": "Derrière le prompt : comment fonctionnent les IA génératives ?",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>ChatGPT, Mistral, Dall-E : les systèmes d’Intelligence Artificielle (IA) générative sont de plus en plus utilisés par le grand public. Ce succès s’explique notamment par les prompts, ces commandes que l’on peut écrire simplement afin d’obtenir des résultats parfois bluffants. <span aria-hidden=\"true\">✨ </span> Mais ça n’a rien de magique… </p><br><p>Dans ce module, vous allez découvrir les coulisses du fonctionnement des prompts et comprendre comment les systèmes d’IA générative produisent leurs réponses en testant avec des exemples concrets.</p>",

api/src/devcomp/infrastructure/datasources/learning-content/modules/tri-multicritere-tableau.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "19468565-a56b-4aa5-9bf0-369e94bc85ea",
33
"slug": "tri-multicritere-tableau",
44
"title": "Trier un tableau selon plusieurs critères",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>Trier des données dans une feuille de calcul peut permettre de trouver efficacement des informations. Parfois, un tri sur une seule colonne ne suffit pas. <br>Dans ce module, vous allez apprendre à trier vos données en combinant plusieurs critères.<br></p>",

api/src/devcomp/infrastructure/datasources/learning-content/modules/utiliser-souris-ordinateur-1.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "e8cee13e-1d4d-47eb-bd26-d7ea6a10b1e6",
33
"slug": "utiliser-souris-ordinateur-1",
44
"title": "Utiliser une souris d'ordinateur - 1",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>Dans ce module, vous découvrirez ce qu’est une souris et à quoi elle sert.&nbsp;</p><p>Vous apprendrez à la déplacer et à cliquer pour interagir avec votre ordinateur.</p><p>Si vous n'avez jamais utilisé de souris, nous vous conseillons d'être accompagné pour ce module.</p><br>",

api/src/devcomp/infrastructure/datasources/learning-content/modules/utiliser-souris-ordinateur-2.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"id": "1f425bc6-7a35-4ceb-9634-a25da1e36233",
33
"slug": "utiliser-souris-ordinateur-2",
44
"title": "Utiliser une souris d'ordinateur - 2",
5-
"isBeta": false,
5+
"isBeta": true,
66
"details": {
77
"image": "https://images.pix.fr/modulix/placeholder-details.svg",
88
"description": "<p>La souris est composée de trois boutons : le bouton gauche, la molette et le bouton droit. Le bouton gauche permet de faire un clic gauche, c’est le plus utilisé. Mais il existe d’autres façons d’utiliser la souris.<br><br> Dans ce module, vous apprendrez à faire d’autres clics et à utiliser les deux autres boutons de la souris. 🖱️</p>",

api/src/devcomp/infrastructure/factories/module-factory.js

+6-5
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ export class ModuleFactory {
4747
.map((component) => {
4848
switch (component.type) {
4949
case 'element': {
50-
const element = ModuleFactory.#buildElement(component.element);
50+
const element = ModuleFactory.#buildElement(component.element, moduleData.isBeta);
5151
if (element) {
5252
return new ComponentElement({ element });
5353
} else {
@@ -60,7 +60,7 @@ export class ModuleFactory {
6060
return new Step({
6161
elements: step.elements
6262
.map((element) => {
63-
const domainElement = ModuleFactory.#buildElement(element);
63+
const domainElement = ModuleFactory.#buildElement(element, moduleData.isBeta);
6464
if (domainElement) {
6565
return domainElement;
6666
} else {
@@ -88,7 +88,7 @@ export class ModuleFactory {
8888
}
8989
}
9090

91-
static #buildElement(element) {
91+
static #buildElement(element, isBeta) {
9292
switch (element.type) {
9393
case 'custom':
9494
return ModuleFactory.#buildCustom(element);
@@ -99,7 +99,7 @@ export class ModuleFactory {
9999
case 'expand':
100100
return ModuleFactory.#buildExpand(element);
101101
case 'image':
102-
return ModuleFactory.#buildImage(element);
102+
return ModuleFactory.#buildImage(element, isBeta);
103103
case 'separator':
104104
return ModuleFactory.#buildSeparator(element);
105105
case 'text':
@@ -157,14 +157,15 @@ export class ModuleFactory {
157157
});
158158
}
159159

160-
static #buildImage(element) {
160+
static #buildImage(element, isBeta) {
161161
return new Image({
162162
id: element.id,
163163
url: element.url,
164164
alt: element.alt,
165165
alternativeText: element.alternativeText,
166166
legend: element.legend,
167167
licence: element.licence,
168+
isBeta,
168169
});
169170
}
170171

api/tests/devcomp/unit/domain/models/element/Element_test.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,12 @@ describe('Unit | Devcomp | Domain | Models | Element', function () {
3131
it('should instanciate non answerable elements', function () {
3232
// Given
3333
const text = new Text({ id: 'id', content: 'content' });
34-
const image = new Image({ id: 'id', url: 'url', alt: 'alt', alternativeText: 'alternativeText' });
34+
const image = new Image({
35+
id: 'id',
36+
url: 'https://assets.pix.org/modules/placeholder-details.svg',
37+
alt: 'alt',
38+
alternativeText: 'alternativeText',
39+
});
3540

3641
const nonAnswerableElements = [text, image];
3742

api/tests/devcomp/unit/domain/models/element/Image_test.js

+40-4
Original file line numberDiff line numberDiff line change
@@ -8,16 +8,17 @@ describe('Unit | Devcomp | Domain | Models | Element | Image', function () {
88
// when
99
const image = new Image({
1010
id: 'id',
11-
url: 'url',
11+
url: 'https://assets.pix.org/modules/placeholder-details.svg',
1212
alt: 'alt',
1313
alternativeText: 'alternativeText',
1414
legend: 'legend',
1515
licence: 'licence',
16+
isBeta: false,
1617
});
1718

1819
// then
1920
expect(image.id).to.equal('id');
20-
expect(image.url).to.equal('url');
21+
expect(image.url).to.equal('https://assets.pix.org/modules/placeholder-details.svg');
2122
expect(image.alt).to.equal('alt');
2223
expect(image.alternativeText).to.equal('alternativeText');
2324
expect(image.legend).to.equal('legend');
@@ -48,11 +49,22 @@ describe('Unit | Devcomp | Domain | Models | Element | Image', function () {
4849
});
4950
});
5051

51-
describe('An image without alt', function () {
52+
describe('An image with invalid url', function () {
5253
it('should throw an error', function () {
5354
// when
5455
const error = catchErrSync(() => new Image({ id: 'id', url: 'url' }))();
5556

57+
// then
58+
expect(error).to.be.instanceOf(DomainError);
59+
expect(error.message).to.equal('The URL must be a valid URL for an image');
60+
});
61+
});
62+
63+
describe('An image without alt', function () {
64+
it('should throw an error', function () {
65+
// when
66+
const error = catchErrSync(() => new Image({ id: 'id', url: 'https://images.pix.fr/coolcat.jpg' }))();
67+
5668
// then
5769
expect(error).to.be.instanceOf(DomainError);
5870
expect(error.message).to.equal('The alt text is required for an image');
@@ -62,11 +74,35 @@ describe('Unit | Devcomp | Domain | Models | Element | Image', function () {
6274
describe('An image without an alternative text', function () {
6375
it('should throw an error', function () {
6476
// when
65-
const error = catchErrSync(() => new Image({ id: 'id', url: 'url', alt: 'alt' }))();
77+
const error = catchErrSync(() => new Image({ id: 'id', url: 'https://images.pix.fr/coolcat.jpg', alt: 'alt' }))();
6678

6779
// then
6880
expect(error).to.be.instanceOf(DomainError);
6981
expect(error.message).to.equal('The alternative text is required for an image');
7082
});
7183
});
84+
85+
describe('When isBeta is false', function () {
86+
describe('and image URL is not from assets.pix.org', function () {
87+
it('should throw an error', function () {
88+
// given & when
89+
const error = catchErrSync(
90+
() =>
91+
new Image({
92+
id: 'id',
93+
url: 'https://images.pix.fr/coolcat.jpg',
94+
alt: 'alt',
95+
alternativeText: 'alternativeText',
96+
legend: 'legend',
97+
licence: 'licence',
98+
isBeta: false,
99+
}),
100+
)();
101+
102+
// then
103+
expect(error).to.be.instanceOf(DomainError);
104+
expect(error.message).to.equal('The image URL must be from "assets.pix.org" when module is production ready');
105+
});
106+
});
107+
});
72108
});

api/tests/devcomp/unit/infrastructure/factories/module-factory_test.js

+101
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ import { Grain } from '../../../../../src/devcomp/domain/models/Grain.js';
1414
import { Module } from '../../../../../src/devcomp/domain/models/module/Module.js';
1515
import { TransitionText } from '../../../../../src/devcomp/domain/models/TransitionText.js';
1616
import { ModuleFactory } from '../../../../../src/devcomp/infrastructure/factories/module-factory.js';
17+
import { DomainError } from '../../../../../src/shared/domain/errors.js';
1718
import { logger } from '../../../../../src/shared/infrastructure/utils/logger.js';
1819
import { catchErrSync, expect, sinon } from '../../../../test-helper.js';
1920
import { validateFlashcards } from '../../../shared/validateFlashcards.js';
@@ -336,6 +337,54 @@ describe('Unit | Devcomp | Infrastructure | Factories | Module ', function () {
336337
});
337338

338339
describe('With ComponentElement', function () {
340+
describe('When isBeta is false', function () {
341+
it('should throw a DomainError when Image element does not have a valid url', function () {
342+
// given
343+
const moduleData = {
344+
id: '6282925d-4775-4bca-b513-4c3009ec5886',
345+
slug: 'title',
346+
title: 'title',
347+
isBeta: false,
348+
details: {
349+
image: 'https://images.pix.fr/modulix/placeholder-details.svg',
350+
description: 'Description',
351+
duration: 5,
352+
level: 'Débutant',
353+
tabletSupport: 'comfortable',
354+
objectives: ['Objective 1'],
355+
},
356+
grains: [
357+
{
358+
id: 'f312c33d-e7c9-4a69-9ba0-913957b8f7dd',
359+
type: 'lesson',
360+
title: 'title',
361+
components: [
362+
{
363+
type: 'element',
364+
element: {
365+
id: '8d7687c8-4a02-4d7e-bf6c-693a6d481c78',
366+
type: 'image',
367+
url: 'https://images.pix.fr/modulix/didacticiel/ordi-spatial.svg',
368+
alt: 'Alternative',
369+
alternativeText: 'Alternative textuelle',
370+
legend: 'legend',
371+
licence: 'licence',
372+
},
373+
},
374+
],
375+
},
376+
],
377+
};
378+
379+
// when
380+
const error = catchErrSync(() => ModuleFactory.build(moduleData))();
381+
382+
// then
383+
expect(error).to.be.an.instanceOf(DomainError);
384+
expect(error.message).to.equal('The image URL must be from "assets.pix.org" when module is production ready');
385+
});
386+
});
387+
339388
it('should instantiate a Module with a ComponentElement which contains an Image Element', function () {
340389
// given
341390
const moduleData = {
@@ -927,6 +976,58 @@ describe('Unit | Devcomp | Infrastructure | Factories | Module ', function () {
927976
});
928977

929978
describe('With ComponentStepper', function () {
979+
describe('When isBeta is false', function () {
980+
it('should throw a DomainError when Image element does not have a valid url', function () {
981+
// given
982+
const moduleData = {
983+
id: '6282925d-4775-4bca-b513-4c3009ec5886',
984+
slug: 'title',
985+
title: 'title',
986+
isBeta: false,
987+
details: {
988+
image: 'https://images.pix.fr/modulix/placeholder-details.svg',
989+
description: 'Description',
990+
duration: 5,
991+
level: 'Débutant',
992+
tabletSupport: 'comfortable',
993+
objectives: ['Objective 1'],
994+
},
995+
grains: [
996+
{
997+
id: 'f312c33d-e7c9-4a69-9ba0-913957b8f7dd',
998+
type: 'lesson',
999+
title: 'title',
1000+
components: [
1001+
{
1002+
type: 'stepper',
1003+
steps: [
1004+
{
1005+
elements: [
1006+
{
1007+
id: '8d7687c8-4a02-4d7e-bf6c-693a6d481c78',
1008+
type: 'image',
1009+
url: 'https://images.pix.fr/modulix/didacticiel/ordi-spatial.svg',
1010+
alt: "Dessin détaillé dans l'alternative textuelle",
1011+
alternativeText: "Dessin d'un ordinateur dans un univers spatial.",
1012+
},
1013+
],
1014+
},
1015+
],
1016+
},
1017+
],
1018+
},
1019+
],
1020+
};
1021+
1022+
// when
1023+
const error = catchErrSync(() => ModuleFactory.build(moduleData))();
1024+
1025+
// then
1026+
expect(error).to.be.an.instanceOf(DomainError);
1027+
expect(error.message).to.equal('The image URL must be from "assets.pix.org" when module is production ready');
1028+
});
1029+
});
1030+
9301031
it('should instantiate a Module with a ComponentStepper which contains an Image Element', function () {
9311032
// given
9321033
const moduleData = {

api/tests/devcomp/unit/infrastructure/serializers/jsonapi/module-serializer_test.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -222,7 +222,7 @@ function getComponents() {
222222
new ComponentElement({
223223
element: new Image({
224224
id: '3',
225-
url: 'url',
225+
url: 'https://assets.pix.org/modules/placeholder-details.svg',
226226
alt: 'alt',
227227
alternativeText: 'alternativeText',
228228
licence: 'mon copyright',
@@ -399,7 +399,7 @@ function getAttributesComponents() {
399399
id: '3',
400400
isAnswerable: false,
401401
type: 'image',
402-
url: 'url',
402+
url: 'https://assets.pix.org/modules/placeholder-details.svg',
403403
legend: 'ma légende',
404404
licence: 'mon copyright',
405405
},

0 commit comments

Comments
 (0)