Skip to content

Commit 42c6ea0

Browse files
committed
feat(api): check image production readiness
1 parent 3c54713 commit 42c6ea0

File tree

4 files changed

+63
-8
lines changed

4 files changed

+63
-8
lines changed

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

+15-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,10 @@
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
@@ -10,11 +13,16 @@ class Image extends Element {
1013
* @param{string} params.alternativeText
1114
* @param{string} params.legend
1215
* @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/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/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)