diff --git a/packages/core/upload/admin/src/components/AssetCard/tests/ImageAssetCard.test.js b/packages/core/upload/admin/src/components/AssetCard/tests/ImageAssetCard.test.js
new file mode 100644
index 0000000000..4084832ff9
--- /dev/null
+++ b/packages/core/upload/admin/src/components/AssetCard/tests/ImageAssetCard.test.js
@@ -0,0 +1,414 @@
+import React from 'react';
+import { ThemeProvider, lightTheme } from '@strapi/parts';
+import { render as renderTL } from '@testing-library/react';
+import { ImageAssetCard } from '../ImageAssetCard';
+import en from '../../../translations/en.json';
+
+jest.mock('../../../utils', () => ({
+ ...jest.requireActual('../../../utils'),
+ getTrad: x => x,
+}));
+
+jest.mock('react-intl', () => ({
+ useIntl: () => ({ formatMessage: jest.fn(({ id }) => en[id]) }),
+}));
+
+describe('ImageAssetCard', () => {
+ it('snapshots the component', () => {
+ const { container } = renderTL(
+
+
+
+ );
+
+ expect(container).toMatchInlineSnapshot(`
+ .c21 {
+ border: 0;
+ -webkit-clip: rect(0 0 0 0);
+ clip: rect(0 0 0 0);
+ height: 1px;
+ margin: -1px;
+ overflow: hidden;
+ padding: 0;
+ position: absolute;
+ width: 1px;
+ }
+
+ .c0 {
+ background: #ffffff;
+ border-radius: 4px;
+ box-shadow: 0px 1px 4px rgba(33,33,52,0.1);
+ }
+
+ .c10 {
+ padding-top: 8px;
+ padding-right: 12px;
+ padding-bottom: 8px;
+ padding-left: 12px;
+ }
+
+ .c15 {
+ background: #f6f6f9;
+ color: #666687;
+ padding: 4px;
+ border-radius: 4px;
+ }
+
+ .c3 {
+ position: absolute;
+ top: 12px;
+ left: 12px;
+ }
+
+ .c5 {
+ position: absolute;
+ top: 12px;
+ right: 12px;
+ }
+
+ .c1 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-flex-direction: row;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-box-pack: center;
+ -webkit-justify-content: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ -webkit-align-items: center;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ }
+
+ .c11 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-flex-direction: row;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-align-items: center;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ }
+
+ .c9 {
+ margin: 0;
+ padding: 0;
+ max-height: 100%;
+ max-width: 100%;
+ }
+
+ .c8 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-pack: center;
+ -webkit-justify-content: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ height: 10.25rem;
+ width: 100%;
+ background: repeating-conic-gradient(#f6f6f9 0% 25%,transparent 0% 50%) 50% / 20px 20px;
+ }
+
+ .c12 {
+ font-weight: 500;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #32324d;
+ }
+
+ .c13 {
+ font-weight: 400;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #666687;
+ }
+
+ .c18 {
+ font-weight: 400;
+ font-size: 0.875rem;
+ line-height: 1.43;
+ color: #32324d;
+ }
+
+ .c19 {
+ font-weight: 600;
+ line-height: 1.14;
+ }
+
+ .c20 {
+ font-weight: 600;
+ font-size: 0.6875rem;
+ line-height: 1.45;
+ text-transform: uppercase;
+ }
+
+ .c16 {
+ display: inline-block;
+ }
+
+ .c17 {
+ margin-left: auto;
+ }
+
+ .c4 {
+ margin: 0;
+ height: 18px;
+ min-width: 18px;
+ border-radius: 4px;
+ border: 1px solid #c0c0cf;
+ -webkit-appearance: none;
+ background-color: #ffffff;
+ }
+
+ .c4:checked {
+ background-color: #4945ff;
+ border: 1px solid #4945ff;
+ }
+
+ .c4:checked:after {
+ content: '';
+ display: block;
+ position: relative;
+ background: url() no-repeat no-repeat center center;
+ width: 10px;
+ height: 10px;
+ left: 50%;
+ top: 50%;
+ -webkit-transform: translateX(-50%) translateY(-50%);
+ -ms-transform: translateX(-50%) translateY(-50%);
+ transform: translateX(-50%) translateY(-50%);
+ }
+
+ .c4:checked:disabled:after {
+ background: url() no-repeat no-repeat center center;
+ }
+
+ .c4:disabled {
+ background-color: #dcdce4;
+ border: 1px solid #c0c0cf;
+ }
+
+ .c4:indeterminate {
+ background-color: #4945ff;
+ border: 1px solid #4945ff;
+ }
+
+ .c4:indeterminate:after {
+ content: '';
+ display: block;
+ position: relative;
+ color: white;
+ height: 2px;
+ width: 10px;
+ background-color: #ffffff;
+ left: 50%;
+ top: 50%;
+ -webkit-transform: translateX(-50%) translateY(-50%);
+ -ms-transform: translateX(-50%) translateY(-50%);
+ transform: translateX(-50%) translateY(-50%);
+ }
+
+ .c4:indeterminate:disabled {
+ background-color: #dcdce4;
+ border: 1px solid #c0c0cf;
+ }
+
+ .c4:indeterminate:disabled:after {
+ background-color: #8e8ea9;
+ }
+
+ .c2 {
+ position: relative;
+ border-bottom: 1px solid #eaeaef;
+ }
+
+ .c6 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ cursor: pointer;
+ padding: 8px;
+ border-radius: 4px;
+ background: #ffffff;
+ border: 1px solid #dcdce4;
+ }
+
+ .c6 svg {
+ height: 12px;
+ width: 12px;
+ }
+
+ .c6 svg > g,
+ .c6 svg path {
+ fill: #ffffff;
+ }
+
+ .c6[aria-disabled='true'] {
+ pointer-events: none;
+ }
+
+ .c7 svg > g,
+ .c7 svg path {
+ fill: #8e8ea9;
+ }
+
+ .c7:hover svg > g,
+ .c7:hover svg path {
+ fill: #666687;
+ }
+
+ .c7:active svg > g,
+ .c7:active svg path {
+ fill: #a5a5ba;
+ }
+
+ .c7[aria-disabled='true'] {
+ background-color: #eaeaef;
+ }
+
+ .c7[aria-disabled='true'] svg path {
+ fill: #666687;
+ }
+
+ .c14 {
+ text-transform: uppercase;
+ }
+
+
+
+
+
+
+
+
+
+

+
+
+
+
+
+
+ hello.png
+
+
+
+ png
+
+ - 40✕40
+
+
+
+
+ Image
+
+
+
+
+
+
+
+ `);
+ });
+});
diff --git a/packages/core/upload/admin/src/components/AssetCard/tests/UnknownAssetCard.test.js b/packages/core/upload/admin/src/components/AssetCard/tests/UnknownAssetCard.test.js
new file mode 100644
index 0000000000..3e081de19c
--- /dev/null
+++ b/packages/core/upload/admin/src/components/AssetCard/tests/UnknownAssetCard.test.js
@@ -0,0 +1,444 @@
+import React from 'react';
+import { ThemeProvider, lightTheme } from '@strapi/parts';
+import { render as renderTL } from '@testing-library/react';
+import { UnknownAssetCard } from '../UnknownAssetCard';
+import en from '../../../translations/en.json';
+
+jest.mock('../../../utils', () => ({
+ ...jest.requireActual('../../../utils'),
+ getTrad: x => x,
+}));
+
+jest.mock('react-intl', () => ({
+ useIntl: () => ({ formatMessage: jest.fn(({ id }) => en[id]) }),
+}));
+
+describe('UnknownAssetCard', () => {
+ it('snapshots the component', () => {
+ const { container } = renderTL(
+
+
+
+ );
+
+ expect(container).toMatchInlineSnapshot(`
+ .c15 {
+ border: 0;
+ -webkit-clip: rect(0 0 0 0);
+ clip: rect(0 0 0 0);
+ height: 1px;
+ margin: -1px;
+ overflow: hidden;
+ padding: 0;
+ position: absolute;
+ width: 1px;
+ }
+
+ .c0 {
+ background: #ffffff;
+ border-radius: 4px;
+ box-shadow: 0px 1px 4px rgba(33,33,52,0.1);
+ }
+
+ .c11 {
+ padding-top: 8px;
+ padding-right: 12px;
+ padding-bottom: 8px;
+ padding-left: 12px;
+ }
+
+ .c3 {
+ position: absolute;
+ top: 12px;
+ left: 12px;
+ }
+
+ .c5 {
+ position: absolute;
+ top: 12px;
+ right: 12px;
+ }
+
+ .c1 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-flex-direction: row;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-box-pack: center;
+ -webkit-justify-content: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ -webkit-align-items: center;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ }
+
+ .c9 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-flex-direction: row;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-align-items: center;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ }
+
+ .c8 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-pack: center;
+ -webkit-justify-content: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ height: 10.25rem;
+ width: 100%;
+ background: repeating-conic-gradient(#f6f6f9 0% 25%,transparent 0% 50%) 50% / 20px 20px;
+ }
+
+ .c12 {
+ font-weight: 500;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #32324d;
+ }
+
+ .c13 {
+ font-weight: 400;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #666687;
+ }
+
+ .c4 {
+ margin: 0;
+ height: 18px;
+ min-width: 18px;
+ border-radius: 4px;
+ border: 1px solid #c0c0cf;
+ -webkit-appearance: none;
+ background-color: #ffffff;
+ }
+
+ .c4:checked {
+ background-color: #4945ff;
+ border: 1px solid #4945ff;
+ }
+
+ .c4:checked:after {
+ content: '';
+ display: block;
+ position: relative;
+ background: url() no-repeat no-repeat center center;
+ width: 10px;
+ height: 10px;
+ left: 50%;
+ top: 50%;
+ -webkit-transform: translateX(-50%) translateY(-50%);
+ -ms-transform: translateX(-50%) translateY(-50%);
+ transform: translateX(-50%) translateY(-50%);
+ }
+
+ .c4:checked:disabled:after {
+ background: url() no-repeat no-repeat center center;
+ }
+
+ .c4:disabled {
+ background-color: #dcdce4;
+ border: 1px solid #c0c0cf;
+ }
+
+ .c4:indeterminate {
+ background-color: #4945ff;
+ border: 1px solid #4945ff;
+ }
+
+ .c4:indeterminate:after {
+ content: '';
+ display: block;
+ position: relative;
+ color: white;
+ height: 2px;
+ width: 10px;
+ background-color: #ffffff;
+ left: 50%;
+ top: 50%;
+ -webkit-transform: translateX(-50%) translateY(-50%);
+ -ms-transform: translateX(-50%) translateY(-50%);
+ transform: translateX(-50%) translateY(-50%);
+ }
+
+ .c4:indeterminate:disabled {
+ background-color: #dcdce4;
+ border: 1px solid #c0c0cf;
+ }
+
+ .c4:indeterminate:disabled:after {
+ background-color: #8e8ea9;
+ }
+
+ .c2 {
+ position: relative;
+ border-bottom: 1px solid #eaeaef;
+ }
+
+ .c6 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ cursor: pointer;
+ padding: 8px;
+ border-radius: 4px;
+ background: #ffffff;
+ border: 1px solid #dcdce4;
+ }
+
+ .c6 svg {
+ height: 12px;
+ width: 12px;
+ }
+
+ .c6 svg > g,
+ .c6 svg path {
+ fill: #ffffff;
+ }
+
+ .c6[aria-disabled='true'] {
+ pointer-events: none;
+ }
+
+ .c7 svg > g,
+ .c7 svg path {
+ fill: #8e8ea9;
+ }
+
+ .c7:hover svg > g,
+ .c7:hover svg path {
+ fill: #666687;
+ }
+
+ .c7:active svg > g,
+ .c7:active svg path {
+ fill: #a5a5ba;
+ }
+
+ .c7[aria-disabled='true'] {
+ background-color: #eaeaef;
+ }
+
+ .c7[aria-disabled='true'] svg path {
+ fill: #666687;
+ }
+
+ .c14 {
+ text-transform: uppercase;
+ }
+
+ .c10 svg {
+ font-size: 3rem;
+ }
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ hello.png
+
+
+
+ png
+
+
+
+
+
+
+
+
+ `);
+ });
+});
diff --git a/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/FromUrlForm.js b/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/FromUrlForm.js
index 3136eb1f3e..b10672e636 100644
--- a/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/FromUrlForm.js
+++ b/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/FromUrlForm.js
@@ -1,4 +1,4 @@
-import React from 'react';
+import React, { useState } from 'react';
import PropTypes from 'prop-types';
import { Box } from '@strapi/parts/Box';
import { ModalFooter } from '@strapi/parts/ModalLayout';
@@ -11,12 +11,15 @@ import { getTrad, urlSchema } from '../../../utils';
import { urlsToAssets } from '../../../utils/urlsToAssets';
export const FromUrlForm = ({ onClose, onAddAsset }) => {
+ const [loading, setLoading] = useState(false);
const { formatMessage } = useIntl();
const handleSubmit = async ({ urls }) => {
+ setLoading(true);
const urlArray = urls.split(/\r?\n/);
const assets = await urlsToAssets(urlArray);
+ // no need to set the loading to false since the component unmounts
onAddAsset(assets);
};
@@ -53,7 +56,7 @@ export const FromUrlForm = ({ onClose, onAddAsset }) => {
+
}
diff --git a/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/FromUrlForm.test.js b/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/FromUrlForm.test.js
index 28f17fc4b4..3786165af2 100644
--- a/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/FromUrlForm.test.js
+++ b/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/FromUrlForm.test.js
@@ -3,7 +3,7 @@ import { ThemeProvider, lightTheme } from '@strapi/parts';
import { render as renderTL, screen, fireEvent, waitFor } from '@testing-library/react';
import { FromUrlForm } from '../FromUrlForm';
import en from '../../../../translations/en.json';
-import { mockAssets } from './server';
+import { server } from './server';
jest.mock('../../../../utils', () => ({
...jest.requireActual('../../../../utils'),
@@ -15,7 +15,9 @@ jest.mock('react-intl', () => ({
}));
describe('FromUrlForm', () => {
- beforeAll(mockAssets);
+ beforeAll(() => server.listen());
+ afterEach(() => server.resetHandlers());
+ afterAll(() => server.close());
it('snapshots the component with 4 URLs: 3 valid and one in failure', async () => {
const onAddAssetSpy = jest.fn();
@@ -60,9 +62,9 @@ describe('FromUrlForm', () => {
},
{
ext: 'lutin',
- mime: undefined,
+ mime: 'application/json',
source: 'url',
- type: 'unknown',
+ type: 'doc',
url: 'http://localhost:5000/not-working-like-cors.lutin',
},
];
diff --git a/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/server.js b/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/server.js
index cc076db0ac..93c33717c0 100644
--- a/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/server.js
+++ b/packages/core/upload/admin/src/components/UploadAssetDialog/AddAssetStep/tests/server.js
@@ -1,26 +1,40 @@
// mocking window.fetch since msw is not able to give back the res.url param
-export const mockAssets = () => {
- window.fetch = url => {
- if (url === 'http://localhost:5000/an-image.png') {
- const headers = { get: () => 'image/png' };
+import { rest } from 'msw';
+import { setupServer } from 'msw/node';
- return Promise.resolve({ url: 'http://localhost:5000/an-image.png', headers });
- }
+export const server = setupServer(
+ rest.get('*/an-image.png', (req, res, ctx) =>
+ res(ctx.set('Content-Type', 'image/png'), ctx.body())
+ ),
+ rest.get('*/a-pdf.pdf', (req, res, ctx) =>
+ res(ctx.set('Content-Type', 'application/pdf'), ctx.body())
+ ),
+ rest.get('*/a-video.mp4', (req, res, ctx) =>
+ res(ctx.set('Content-Type', 'video/mp4'), ctx.body())
+ ),
+ rest.get('*/not-working-like-cors.lutin', (req, res, ctx) => res(ctx.json({})))
+);
- if (url === 'http://localhost:5000/a-pdf.pdf') {
- const headers = { get: () => 'application/pdf' };
+// export const mockAssets = url => {
+// if (url === 'http://localhost:5000/an-image.png') {
+// const headers = { get: () => 'image/png' };
- return Promise.resolve({ url: 'http://localhost:5000/a-pdf.pdf', headers });
- }
+// return Promise.resolve({ url: 'http://localhost:5000/an-image.png', headers });
+// }
- if (url === 'http://localhost:5000/a-video.mp4') {
- const headers = { get: () => 'video/mp4' };
+// if (url === 'http://localhost:5000/a-pdf.pdf') {
+// const headers = { get: () => 'application/pdf' };
- return Promise.resolve({ url: 'http://localhost:5000/a-video.mp4', headers });
- }
+// return Promise.resolve({ url: 'http://localhost:5000/a-pdf.pdf', headers });
+// }
- // eslint-disable-next-line prefer-promise-reject-errors
- return Promise.reject('http://localhost:5000/not-working-like-cors.lutin');
- };
-};
+// if (url === 'http://localhost:5000/a-video.mp4') {
+// const headers = { get: () => 'video/mp4' };
+
+// return Promise.resolve({ url: 'http://localhost:5000/a-video.mp4', headers });
+// }
+
+// // eslint-disable-next-line prefer-promise-reject-errors
+// return Promise.reject('http://localhost:5000/not-working-like-cors.lutin');
+// };
diff --git a/packages/core/upload/admin/src/utils/urlsToAssets.js b/packages/core/upload/admin/src/utils/urlsToAssets.js
index 979be5f854..651c5aa35e 100644
--- a/packages/core/upload/admin/src/utils/urlsToAssets.js
+++ b/packages/core/upload/admin/src/utils/urlsToAssets.js
@@ -1,12 +1,16 @@
+import axios from 'axios';
import { AssetType, AssetSource } from '../constants';
export const urlsToAssets = async urls => {
const assetPromises = urls.map(url =>
- fetch(url)
- .then(res => ({ url: res.url, mime: res.headers.get('Content-Type') }))
- .catch(() => {
- throw url;
+ axios
+ .get(url, {
+ responseType: 'blob',
})
+ .then(res => ({
+ url: res.config.url,
+ mime: res.headers['content-type'],
+ }))
);
// Retrieve the assets metadata
const assetsResults = await Promise.allSettled(assetPromises);