diff --git a/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.spec.tsx b/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.spec.tsx index c5945be6c8..42b6883b49 100644 --- a/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.spec.tsx +++ b/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.spec.tsx @@ -96,6 +96,8 @@ describe('', () => { modelName.DepositedFiles, depositedFile.id, file, + modelName.FileDepositories, + depositedFile.file_depository.id, ); }); diff --git a/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.tsx b/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.tsx index 115612e41b..35fbeb0351 100644 --- a/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.tsx +++ b/src/frontend/apps/lti_site/apps/deposit/components/Dashboard/DashboardStudent/UploadFiles/index.tsx @@ -91,7 +91,13 @@ export const UploadFiles = () => { size: file.size, filename: file.name, }); - addUpload(modelName.DepositedFiles, depositedFile.id, file); + addUpload( + modelName.DepositedFiles, + depositedFile.id, + file, + modelName.FileDepositories, + depositedFile.file_depository.id, + ); refreshDepositedFiles(); } catch (error) { if ((error as object).hasOwnProperty('size') && metadata.data) { diff --git a/src/frontend/apps/lti_site/components/LTIRoutes/index.tsx b/src/frontend/apps/lti_site/components/LTIRoutes/index.tsx index 6a550e012c..daae8f99ad 100644 --- a/src/frontend/apps/lti_site/components/LTIRoutes/index.tsx +++ b/src/frontend/apps/lti_site/components/LTIRoutes/index.tsx @@ -8,6 +8,7 @@ import { UploadForm, UploadHandlers, UploadManager, + UploadingObject, WithParams, builderFullScreenErrorRoute, modelName, @@ -130,11 +131,13 @@ export const LTIInnerRoutes = () => { path={UPLOAD_FORM_ROUTE.default} element={ - {({ objectId, objectType }) => + {({ objectId, objectType, parentType, parentId }) => objectId && objectType ? ( ) : ( ', () => { modelName.CLASSROOM_DOCUMENTS, classroomDocument.id, file, + modelName.CLASSROOMS, + '1', ); }); diff --git a/src/frontend/packages/lib_classroom/src/components/ClassroomWidgetProvider/widgets/SupportSharing/UploadDocuments/index.tsx b/src/frontend/packages/lib_classroom/src/components/ClassroomWidgetProvider/widgets/SupportSharing/UploadDocuments/index.tsx index 94ef2cca15..bade7a3567 100644 --- a/src/frontend/packages/lib_classroom/src/components/ClassroomWidgetProvider/widgets/SupportSharing/UploadDocuments/index.tsx +++ b/src/frontend/packages/lib_classroom/src/components/ClassroomWidgetProvider/widgets/SupportSharing/UploadDocuments/index.tsx @@ -127,7 +127,13 @@ export const UploadDocuments = ({ classroomId }: UploadDocumentsProps) => { size: file.size, classroom: classroomId, }); - addUpload(modelName.CLASSROOM_DOCUMENTS, document.id, file); + addUpload( + modelName.CLASSROOM_DOCUMENTS, + document.id, + file, + modelName.CLASSROOMS, + classroomId, + ); refreshClassroomDocuments(); } catch (error) { if ((error as object).hasOwnProperty('size') && metadata.data) { diff --git a/src/frontend/packages/lib_components/src/common/UploadField/index.tsx b/src/frontend/packages/lib_components/src/common/UploadField/index.tsx index 44bb4db099..8cbaee12ca 100644 --- a/src/frontend/packages/lib_components/src/common/UploadField/index.tsx +++ b/src/frontend/packages/lib_components/src/common/UploadField/index.tsx @@ -4,7 +4,10 @@ import Dropzone from 'react-dropzone'; import { defineMessages, useIntl } from 'react-intl'; import styled from 'styled-components'; -import { useUploadManager } from '@lib-components/common/UploadManager'; +import { + UploadingObject, + useUploadManager, +} from '@lib-components/common/UploadManager'; import { uploadableModelName } from '@lib-components/types/models'; import { DropzonePlaceholder } from './DropzonePlaceholder'; @@ -25,16 +28,23 @@ const DropzoneStyled = styled.div` export interface UploadFieldProps { objectType: uploadableModelName; objectId: string; + parentType?: Maybe; + parentId?: Maybe; } -export const UploadField = ({ objectType, objectId }: UploadFieldProps) => { +export const UploadField = ({ + objectType, + objectId, + parentType, + parentId, +}: UploadFieldProps) => { const { addUpload } = useUploadManager(); const [file, setFile] = useState>(undefined); const intl = useIntl(); const onDrop = (files: File[]) => { setFile(files[0]); - addUpload(objectType, objectId, files[0]); + addUpload(objectType, objectId, files[0], parentType, parentId); }; return ( diff --git a/src/frontend/packages/lib_components/src/common/UploadForm/index.tsx b/src/frontend/packages/lib_components/src/common/UploadForm/index.tsx index fdae484570..c048ddd1af 100644 --- a/src/frontend/packages/lib_components/src/common/UploadForm/index.tsx +++ b/src/frontend/packages/lib_components/src/common/UploadForm/index.tsx @@ -14,6 +14,7 @@ import { Loader } from '@lib-components/common/Loader'; import { UploadField } from '@lib-components/common/UploadField'; import { UploadManagerStatus, + UploadingObject, useUploadManager, } from '@lib-components/common/UploadManager'; import { builderDashboardRoute } from '@lib-components/data/routes'; @@ -113,9 +114,16 @@ const UploadFormBack = styled.div` export interface UploadFormProps { objectId: UploadableObject['id']; objectType: uploadableModelName; + parentType?: Maybe; + parentId?: Maybe; } -export const UploadForm = ({ objectId, objectType }: UploadFormProps) => { +export const UploadForm = ({ + objectId, + objectType, + parentType, + parentId, +}: UploadFormProps) => { const appData = useAppConfig(); const { uploadManagerState, resetUpload } = useUploadManager(); const objectStatus = uploadManagerState[objectId]?.status; @@ -199,7 +207,9 @@ export const UploadForm = ({ objectId, objectType }: UploadFormProps) => { /> - + diff --git a/src/frontend/packages/lib_components/src/common/UploadManager/index.spec.tsx b/src/frontend/packages/lib_components/src/common/UploadManager/index.spec.tsx index 9d0c97ff5d..923acf9214 100644 --- a/src/frontend/packages/lib_components/src/common/UploadManager/index.spec.tsx +++ b/src/frontend/packages/lib_components/src/common/UploadManager/index.spec.tsx @@ -109,6 +109,99 @@ describe('', () => { } }); + it('uploads the file with a parent path', async () => { + const objectType = modelName.THUMBNAILS; + const objectId = uuidv4(); + const parentType = modelName.VIDEOS; + const parentId = uuidv4(); + const file = new File(['(⌐□_□)'], 'course.jpg', { type: 'image/jpeg' }); + + const initiateUploadDeferred = new Deferred(); + const mockInitiateUpload = fetchMock.mock( + `/api/videos/${parentId}/thumbnails/${objectId}/initiate-upload/`, + initiateUploadDeferred.promise, + { method: 'POST' }, + ); + + const fileUploadDeferred = new Deferred(); + xhrMock.post( + 'https://s3.aws.example.com/', + () => fileUploadDeferred.promise, + ); + + render( + + + , + ); + + { + const { addUpload, uploadManagerState } = getLatestHookValues(); + expect(uploadManagerState).toEqual({}); + act(() => addUpload(objectType, objectId, file, parentType, parentId)); + } + { + const { uploadManagerState } = getLatestHookValues(); + expect(uploadManagerState).toEqual({ + [objectId]: { + objectId, + objectType, + file, + progress: 0, + status: UploadManagerStatus.INIT, + parentType, + parentId, + }, + }); + expect(mockInitiateUpload.calls()).toHaveLength(1); + } + { + await act(async () => { + initiateUploadDeferred.resolve({ + fields: { + key: 'foo', + }, + url: 'https://s3.aws.example.com/', + }); + // We need to wait for the upload to complete before we can check the state + await new Promise((resolve) => setTimeout(resolve, 100)); + }); + const { uploadManagerState } = getLatestHookValues(); + expect(uploadManagerState).toEqual({ + [objectId]: { + objectId, + objectType, + file, + progress: 0, + status: UploadManagerStatus.UPLOADING, + parentType, + parentId, + }, + }); + } + { + await act(async () => { + fileUploadDeferred.resolve( + new MockResponse().body('form data body').status(204), + ); + // We need to wait for the upload to complete before we can check the state + await new Promise((resolve) => setTimeout(resolve, 100)); + }); + const { uploadManagerState } = getLatestHookValues(); + expect(uploadManagerState).toEqual({ + [objectId]: { + objectId, + objectType, + file, + progress: 0, + status: UploadManagerStatus.SUCCESS, + parentType, + parentId, + }, + }); + } + }); + it('reports the error and does not upload to AWS when it fails to get the policy', async () => { const objectType = modelName.VIDEOS; const objectId = uuidv4(); diff --git a/src/frontend/packages/lib_components/src/common/UploadManager/index.tsx b/src/frontend/packages/lib_components/src/common/UploadManager/index.tsx index 4d482fb5e7..9a513da7d6 100644 --- a/src/frontend/packages/lib_components/src/common/UploadManager/index.tsx +++ b/src/frontend/packages/lib_components/src/common/UploadManager/index.tsx @@ -1,3 +1,4 @@ +import { Maybe } from '@lib-common/types'; import React, { createContext, useCallback, @@ -31,6 +32,12 @@ export interface UploadingObject { progress: number; status: UploadManagerStatus; message?: string; + parentType?: + | modelName.VIDEOS + | MarkdownDocumentModelName.MARKDOWN_DOCUMENTS + | ClassroomModelName.CLASSROOMS + | FileDepositoryModelName.FileDepositories; + parentId?: string; } export interface UploadManagerState { @@ -82,7 +89,7 @@ export const UploadManager = ({ Object.values(uploadManagerState) .filter(({ status }) => status === UploadManagerStatus.INIT) - .forEach(({ file, objectId, objectType }) => { + .forEach(({ file, objectId, objectType, parentId, parentType }) => { (async () => { let presignedPost: AWSPresignedPost; try { @@ -92,6 +99,8 @@ export const UploadManager = ({ file.name, file.type, file.size, + parentType, + parentId, ); } catch (error) { if ((error as ApiException).type === 'SizeError') { @@ -196,7 +205,13 @@ export const useUploadManager = () => { useContext(UploadManagerContext); const addUpload = useCallback( - (objectType: uploadableModelName, objectId: string, file: File) => { + ( + objectType: uploadableModelName, + objectId: string, + file: File, + parentType?: Maybe, + parentId?: Maybe, + ) => { setUploadState((state) => ({ ...state, [objectId]: { @@ -205,6 +220,8 @@ export const useUploadManager = () => { file, progress: 0, status: UploadManagerStatus.INIT, + parentType, + parentId, }, })); }, diff --git a/src/frontend/packages/lib_components/src/data/sideEffects/initiateUpload/index.ts b/src/frontend/packages/lib_components/src/data/sideEffects/initiateUpload/index.ts index f0f4d02879..fb74d4be7d 100644 --- a/src/frontend/packages/lib_components/src/data/sideEffects/initiateUpload/index.ts +++ b/src/frontend/packages/lib_components/src/data/sideEffects/initiateUpload/index.ts @@ -1,3 +1,6 @@ +import { Maybe } from '@lib-common/types'; + +import { UploadingObject } from '@lib-components/common'; import { fetchWrapper } from '@lib-components/common/queries/fetchWrapper'; import { useJwt } from '@lib-components/hooks/stores'; import { API_ENDPOINT } from '@lib-components/settings'; @@ -10,6 +13,11 @@ import { UploadableObject } from '@lib-components/types/tracks'; * policy to authenticate this upload with S3. * @param objectType The kind of object for which we're uploading a file (model name). * @param objectId The ID of the object for which we're uploading a file. + * @param filename The name of the file we're uploading. + * @param mimetype The mimetype of the file we're uploading. + * @param size The size of the file we're uploading. + * @param parentType The kind of parent object for which we're uploading a file (nullable model name). + * @param parentId The ID of the parent object for which we're uploading a file (nullable). */ export const initiateUpload = async ( objectType: uploadableModelName, @@ -17,22 +25,25 @@ export const initiateUpload = async ( filename: string, mimetype: string, size: number, + parentType?: Maybe, + parentId?: Maybe, ) => { - const response = await fetchWrapper( - `${API_ENDPOINT}/${objectType}/${objectId}/initiate-upload/`, - { - body: JSON.stringify({ - filename, - mimetype, - size, - }), - headers: { - Authorization: `Bearer ${useJwt.getState().getJwt() ?? ''}`, - 'Content-Type': 'application/json', - }, - method: 'POST', + let input = `${API_ENDPOINT}/${objectType}/${objectId}/initiate-upload/`; + if (parentId && parentType) { + input = `${API_ENDPOINT}/${parentType}/${parentId}/${objectType}/${objectId}/initiate-upload/`; + } + const response = await fetchWrapper(input, { + body: JSON.stringify({ + filename, + mimetype, + size, + }), + headers: { + Authorization: `Bearer ${useJwt.getState().getJwt() ?? ''}`, + 'Content-Type': 'application/json', }, - ); + method: 'POST', + }); if (!response.ok) { const contentType = response.headers.get('content-type'); diff --git a/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.spec.tsx b/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.spec.tsx index c021391361..efa7396958 100644 --- a/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.spec.tsx +++ b/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.spec.tsx @@ -551,7 +551,7 @@ describe('', () => { // Drop an image const markdownImageId = '5459a5b2-2f81-11ed-ab8f-47c92ec0ac16'; fetchMock.postOnce( - `/api/markdown-images/`, + `/api/markdown-documents/1/markdown-images/`, markdownImageMockFactory({ id: markdownImageId, active_stamp: null, @@ -562,7 +562,7 @@ describe('', () => { }), ); fetchMock.postOnce( - `/api/markdown-images/${markdownImageId}/initiate-upload/`, + `/api/markdown-documents/1/markdown-images/${markdownImageId}/initiate-upload/`, { fields: { key: 'foo', @@ -589,7 +589,7 @@ describe('', () => { // Image is uploaded fetchMock.get( - `/api/markdown-images/${markdownImageId}/`, + `/api/markdown-documents/1/markdown-images/${markdownImageId}/`, markdownImageMockFactory({ id: markdownImageId, is_ready_to_show: true, diff --git a/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.tsx b/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.tsx index 275adf1d18..16e7547066 100644 --- a/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.tsx +++ b/src/frontend/packages/lib_markdown/src/components/MarkdownEditor/index.tsx @@ -125,7 +125,10 @@ export const MarkdownEditor = ({ markdownDocumentId }: MarkdownEditorProps) => { ); }; - const { addImageUpload } = useImageUploadManager(onImageUploadFinished); + const { addImageUpload } = useImageUploadManager( + markdownDocumentId, + onImageUploadFinished, + ); // note: we don't want to fetch the markdown document regularly to prevent // any editor update while the user has not saved her document. diff --git a/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.spec.tsx b/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.spec.tsx index 3d77fb25d0..7ae39b0c36 100644 --- a/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.spec.tsx +++ b/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.spec.tsx @@ -381,7 +381,7 @@ describe('', () => { const markdownText = fs.readFileSync(file, { encoding: 'utf8' }); fetchMock.getOnce( - '/api/markdown-images/981a52a2-7caf-49a5-bb36-1d8512152214/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/981a52a2-7caf-49a5-bb36-1d8512152214/`, markdownImageMockFactory({ id: '981a52a2-7caf-49a5-bb36-1d8512152214', url: 'https://s3.link/easy-to-find.png', @@ -389,7 +389,7 @@ describe('', () => { ); fetchMock.getOnce( - '/api/markdown-images/066036cc-2dde-11ed-89f4-afcf72a20b4c/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/066036cc-2dde-11ed-89f4-afcf72a20b4c/`, markdownImageMockFactory({ id: '066036cc-2dde-11ed-89f4-afcf72a20b4c', url: null, diff --git a/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.tsx b/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.tsx index b4aa71d05a..c824071212 100644 --- a/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.tsx +++ b/src/frontend/packages/lib_markdown/src/components/MdxRenderer/index.tsx @@ -90,7 +90,10 @@ export const MdxRenderer = ({ remarkLatexPlugin(markdownDocumentId), remarkMermaidPlugin, remarkMath, - remarkLocallyHostedImagePlugin(localImagesUrlCache.current), + remarkLocallyHostedImagePlugin( + markdownDocumentId, + localImagesUrlCache.current, + ), ]; const rehypePlugins: PluggableList = [ options?.useMathjax ? rehypeMathjax : rehypeKatex, diff --git a/src/frontend/packages/lib_markdown/src/components/MdxRenderer/remarkLocallyHostedImagePlugin.tsx b/src/frontend/packages/lib_markdown/src/components/MdxRenderer/remarkLocallyHostedImagePlugin.tsx index 03ea4a3ea9..aba6c03bef 100644 --- a/src/frontend/packages/lib_markdown/src/components/MdxRenderer/remarkLocallyHostedImagePlugin.tsx +++ b/src/frontend/packages/lib_markdown/src/components/MdxRenderer/remarkLocallyHostedImagePlugin.tsx @@ -11,6 +11,7 @@ import { fetchOneMarkdownImage } from '@lib-markdown/data/queries'; import { MarkdownImageCache } from './types'; const remarkLocallyHostedImagePlugin = ( + markdownDocumentId: string, localImagesUrlCache: MarkdownImageCache, ) => { // `markdownDocumentId` is mandatory to allow API calls @@ -41,7 +42,10 @@ const remarkLocallyHostedImagePlugin = ( } if (image.url.startsWith('/uploaded/image/')) { - const response = await fetchOneMarkdownImage(imageId); + const response = await fetchOneMarkdownImage( + markdownDocumentId, + imageId, + ); if (response.url) { localImagesUrlCache[imageId] = { url: response.url, diff --git a/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.spec.tsx b/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.spec.tsx index bb90e94e99..fa4f92c2bf 100644 --- a/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.spec.tsx +++ b/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.spec.tsx @@ -23,13 +23,17 @@ jest.mock('lib-components', () => ({ describe('useImageUploadManager', () => { let getLatestUseImageUploadManagerHookValues: () => any = () => {}; let getLatestUseUploadManagerHookValues: () => any = () => {}; + const markdownDocumentId = 'truc'; const onImageUploadFinished = jest.fn(); const TestComponent = () => { const uploadManager = useUploadManager(); getLatestUseUploadManagerHookValues = () => uploadManager; - const imageUploadManager = useImageUploadManager(onImageUploadFinished); + const imageUploadManager = useImageUploadManager( + markdownDocumentId, + onImageUploadFinished, + ); getLatestUseImageUploadManagerHookValues = () => imageUploadManager; return null; }; @@ -49,8 +53,8 @@ describe('useImageUploadManager', () => { const initiateUploadDeferred = new Deferred(); - const mockcreateMarkdownImage = fetchMock.postOnce( - `/api/markdown-images/`, + const mockCreateMarkdownImage = fetchMock.postOnce( + `/api/markdown-documents/${markdownDocumentId}/markdown-images/`, markdownImageMockFactory({ id: objectId, active_stamp: null, @@ -62,7 +66,7 @@ describe('useImageUploadManager', () => { ); const mockInitiateUpload = fetchMock.postOnce( - `/api/markdown-images/${objectId}/initiate-upload/`, + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${objectId}/initiate-upload/`, initiateUploadDeferred.promise, ); @@ -93,15 +97,19 @@ describe('useImageUploadManager', () => { file, progress: 0, status: UploadManagerStatus.INIT, + parentType: modelName.MARKDOWN_DOCUMENTS, + parentId: markdownDocumentId, }, }); expect( mockInitiateUpload.calls( - `/api/markdown-images/${objectId}/initiate-upload/`, + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${objectId}/initiate-upload/`, ), ).toHaveLength(1); expect( - mockcreateMarkdownImage.calls(`/api/markdown-images/`), + mockCreateMarkdownImage.calls( + `/api/markdown-documents/${markdownDocumentId}/markdown-images/`, + ), ).toHaveLength(1); } { @@ -122,6 +130,8 @@ describe('useImageUploadManager', () => { file, progress: 0, status: UploadManagerStatus.UPLOADING, + parentType: modelName.MARKDOWN_DOCUMENTS, + parentId: markdownDocumentId, }, }); expect(screen.getByRole('status')).toHaveTextContent('course.gif0%'); @@ -129,7 +139,7 @@ describe('useImageUploadManager', () => { // When status will turn to SUCCESS the image polling will start fetchMock.get( - `/api/markdown-images/${objectId}/`, + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${objectId}/`, markdownImageMockFactory({ id: objectId, is_ready_to_show: false, @@ -152,6 +162,8 @@ describe('useImageUploadManager', () => { file, progress: 0, status: UploadManagerStatus.SUCCESS, + parentType: modelName.MARKDOWN_DOCUMENTS, + parentId: markdownDocumentId, }, }); expect(screen.getByRole('status')).toHaveTextContent( @@ -162,7 +174,7 @@ describe('useImageUploadManager', () => { { act(() => { fetchMock.get( - `/api/markdown-images/${objectId}/`, + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${objectId}/`, markdownImageMockFactory({ id: objectId, is_ready_to_show: true, diff --git a/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.tsx b/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.tsx index 8a1f04d203..15a5d751fe 100644 --- a/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.tsx +++ b/src/frontend/packages/lib_markdown/src/components/useImageUploadManager/index.tsx @@ -33,6 +33,7 @@ const toasterStyle = { }; export const useImageUploadManager = ( + markdownDocumentId: string, onImageUploadFinished: (imageId: string, imageFileName: string) => void, ) => { const intl = useIntl(); @@ -55,7 +56,7 @@ export const useImageUploadManager = ( // Once the update is done, the file will be processed, we have to wait for the processing // to be done too, hence the polling. await toast.promise( - pollForMarkdownImage(imageId), + pollForMarkdownImage(markdownDocumentId, imageId), { loading: intl.formatMessage(messages.processing, { imageName: uploadingObject.file.name, @@ -83,16 +84,28 @@ export const useImageUploadManager = ( resetUpload(imageId); } }); - }, [intl, onImageUploadFinished, resetUpload, uploadManagerState]); + }, [ + intl, + markdownDocumentId, + onImageUploadFinished, + resetUpload, + uploadManagerState, + ]); const addImageUpload = useCallback( async (file: File) => { - const response = await createMarkdownImage(); + const response = await createMarkdownImage(markdownDocumentId); const markdownImageId = response.id; - addUpload(modelName.MARKDOWN_IMAGES, markdownImageId, file); + addUpload( + modelName.MARKDOWN_IMAGES, + markdownImageId, + file, + modelName.MARKDOWN_DOCUMENTS, + markdownDocumentId, + ); return markdownImageId; }, - [addUpload], + [addUpload, markdownDocumentId], ); return { addImageUpload }; diff --git a/src/frontend/packages/lib_markdown/src/data/queries/index.tsx b/src/frontend/packages/lib_markdown/src/data/queries/index.tsx index 3463bf7321..488c6bf3ab 100644 --- a/src/frontend/packages/lib_markdown/src/data/queries/index.tsx +++ b/src/frontend/packages/lib_markdown/src/data/queries/index.tsx @@ -212,10 +212,16 @@ export const markdownRenderLatex = ( }; // It has to be called outside hook context -export const fetchOneMarkdownImage = (markdownImageId: string): any => { +export const fetchOneMarkdownImage = ( + markdownDocumentId: string, + markdownImageId: string, +): any => { return fetchOne({ meta: undefined, pageParam: undefined, - queryKey: ['markdown-images', markdownImageId], + queryKey: [ + `markdown-documents/${markdownDocumentId}/markdown-images`, + markdownImageId, + ], }); }; diff --git a/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.spec.tsx b/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.spec.tsx index bc17d7b4e4..a008009193 100644 --- a/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.spec.tsx +++ b/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.spec.tsx @@ -1,9 +1,11 @@ import fetchMock from 'fetch-mock'; import { useJwt } from 'lib-components'; +import { v4 as uuidv4 } from 'uuid'; import { createMarkdownImage } from './index'; describe('createMarkdownImage', () => { + const markdownDocumentId = uuidv4(); beforeEach(() => { useJwt.setState({ jwt: 'token', @@ -13,17 +15,20 @@ describe('createMarkdownImage', () => { afterEach(() => fetchMock.restore()); it('creates a new shared live media and returns it', async () => { - fetchMock.mock('/api/markdown-images/', { - active_stamp: null, - filename: null, - id: '5570eb90-764e-4300-b92e-d3426e9046d2', - is_ready_to_show: false, - upload_state: 'pending', - url: null, - markdown_document: '72f53735-3283-456c-a562-4e1b59e2a686', - }); + fetchMock.mock( + `/api/markdown-documents/${markdownDocumentId}/markdown-images/`, + { + active_stamp: null, + filename: null, + id: '5570eb90-764e-4300-b92e-d3426e9046d2', + is_ready_to_show: false, + upload_state: 'pending', + url: null, + markdown_document: markdownDocumentId, + }, + ); - const markdownImage = await createMarkdownImage(); + const markdownImage = await createMarkdownImage(markdownDocumentId); const fetchArgs = fetchMock.lastCall()![1]!; @@ -34,7 +39,7 @@ describe('createMarkdownImage', () => { is_ready_to_show: false, upload_state: 'pending', url: null, - markdown_document: '72f53735-3283-456c-a562-4e1b59e2a686', + markdown_document: markdownDocumentId, }); expect(fetchArgs.headers).toEqual({ Authorization: 'Bearer token', @@ -45,19 +50,22 @@ describe('createMarkdownImage', () => { it('throws when it fails to create the Markdown image (request failure)', async () => { fetchMock.mock( - '/api/markdown-images/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/`, Promise.reject(new Error('Failed to perform the request')), ); - await expect(createMarkdownImage()).rejects.toThrow( + await expect(createMarkdownImage(markdownDocumentId)).rejects.toThrow( 'Failed to perform the request', ); }); it('throws when it fails to create the Markdown image (API error)', async () => { - fetchMock.mock('/api/markdown-images/', 400); + fetchMock.mock( + `/api/markdown-documents/${markdownDocumentId}/markdown-images/`, + 400, + ); - await expect(createMarkdownImage()).rejects.toThrow( + await expect(createMarkdownImage(markdownDocumentId)).rejects.toThrow( 'Failed to create a new markdown image.', ); }); diff --git a/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.ts b/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.ts index 7d19d3ae7f..4ac7a4605d 100644 --- a/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.ts +++ b/src/frontend/packages/lib_markdown/src/data/sideEffects/createMarkdownImage/index.ts @@ -8,9 +8,9 @@ import { useJwt, } from 'lib-components'; -export const createMarkdownImage = async () => { +export const createMarkdownImage = async (markdownDocumentId: string) => { const response = await fetchWrapper( - `${API_ENDPOINT}/${modelName.MARKDOWN_IMAGES}/`, + `${API_ENDPOINT}/${modelName.MARKDOWN_DOCUMENTS}/${markdownDocumentId}/${modelName.MARKDOWN_IMAGES}/`, { headers: { Authorization: `Bearer ${useJwt.getState().getJwt()}`, diff --git a/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.spec.tsx b/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.spec.tsx index 4c22cce959..0500208fc1 100644 --- a/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.spec.tsx +++ b/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.spec.tsx @@ -2,6 +2,7 @@ import { waitFor } from '@testing-library/react'; import fetchMock from 'fetch-mock'; import { markdownImageMockFactory } from 'index'; import { report } from 'lib-components'; +import { v4 as uuidv4 } from 'uuid'; import { pollForMarkdownImage } from './index'; @@ -10,6 +11,9 @@ jest.mock('lib-components', () => ({ report: jest.fn(), })); +const markdownDocumentId = uuidv4(); +const markdownImageId = uuidv4(); + describe('pollForMarkdownImage', () => { beforeEach(() => { jest.clearAllMocks(); @@ -22,10 +26,10 @@ describe('pollForMarkdownImage', () => { it('polls the image, backing off until it is ready and resolves with a success', async () => { fetchMock.mock( - '/api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/`, JSON.stringify( markdownImageMockFactory({ - id: 'c43f0c8f-4d3b-4219-86c3-86367b2b88cc', + id: markdownImageId, is_ready_to_show: false, }), ), @@ -33,14 +37,15 @@ describe('pollForMarkdownImage', () => { ); const promise = pollForMarkdownImage( - 'c43f0c8f-4d3b-4219-86c3-86367b2b88cc', + markdownDocumentId, + markdownImageId, 1, ); await waitFor(() => { expect( fetchMock.calls( - '/api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/`, { method: 'GET', }, @@ -49,11 +54,11 @@ describe('pollForMarkdownImage', () => { }); const markdownImage = markdownImageMockFactory({ - id: 'c43f0c8f-4d3b-4219-86c3-86367b2b88cc', + id: markdownImageId, is_ready_to_show: true, }); fetchMock.mock( - '/api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/`, JSON.stringify(markdownImage), { method: 'GET', @@ -64,7 +69,7 @@ describe('pollForMarkdownImage', () => { await waitFor(() => { expect( fetchMock.calls( - '/api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/`, { method: 'GET', }, @@ -77,26 +82,26 @@ describe('pollForMarkdownImage', () => { it('polls non-existing image', async () => { fetchMock.mock( - '/api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/`, 404, { method: 'GET' }, ); await expect(async () => { - await pollForMarkdownImage('c43f0c8f-4d3b-4219-86c3-86367b2b88cc'); + await pollForMarkdownImage(markdownDocumentId, markdownImageId); }).rejects.toThrow( - 'Failed to get /api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/.', + `Failed to get /api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/.`, ); expect(report).toHaveBeenCalledWith( Error( - 'Failed to get /api/markdown-images/c43f0c8f-4d3b-4219-86c3-86367b2b88cc/.', + `Failed to get /api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/.`, ), ); }); it('resolves with a failure and reports it when it fails to poll the image', async () => { fetchMock.mock( - '/api/markdown-images/15cf570a-5dc6-421a-9856-59e1b008a6fb/', + `/api/markdown-documents/${markdownDocumentId}/markdown-images/${markdownImageId}/`, Promise.reject(new Error('Failed to get the image')), { method: 'GET', @@ -105,7 +110,7 @@ describe('pollForMarkdownImage', () => { ); await expect(async () => { - await pollForMarkdownImage('15cf570a-5dc6-421a-9856-59e1b008a6fb'); + await pollForMarkdownImage(markdownDocumentId, markdownImageId); }).rejects.toThrow('Failed to get the image'); expect(report).toHaveBeenCalledWith(Error('Failed to get the image')); diff --git a/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.tsx b/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.tsx index a529c35e4e..c04d675a50 100644 --- a/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.tsx +++ b/src/frontend/packages/lib_markdown/src/data/sideEffects/pollForMarkdownImage/index.tsx @@ -6,12 +6,13 @@ import { MarkdownImage, report } from 'lib-components'; import { fetchOneMarkdownImage } from '@lib-markdown/data/queries'; export async function pollForMarkdownImage( - resourceId: string, + documentId: string, + imageId: string, timer = 15, counter = 1, ): Promise { try { - const image = await fetchOneMarkdownImage(resourceId); + const image = await fetchOneMarkdownImage(documentId, imageId); if (image.is_ready_to_show) { return image; @@ -19,7 +20,7 @@ export async function pollForMarkdownImage( counter++; timer = timer * counter; await new Promise((resolve) => window.setTimeout(resolve, 100 * timer)); - return await pollForMarkdownImage(resourceId, timer, counter); + return await pollForMarkdownImage(documentId, imageId, timer, counter); } } catch (error) { report(error); diff --git a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.spec.tsx b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.spec.tsx index 6ae7a99690..77f53d972a 100644 --- a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.spec.tsx +++ b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.spec.tsx @@ -174,6 +174,8 @@ describe('', () => { modelName.TIMEDTEXTTRACKS, mockTimedTextTrack.id, file, + modelName.VIDEOS, + mockedVideo.id, ), ); }); diff --git a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.tsx b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.tsx index fcaf76a55c..07deddf29d 100644 --- a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.tsx +++ b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/LocalizedTimedTextTrackUpload/index.tsx @@ -103,6 +103,8 @@ export const LocalizedTimedTextTrackUpload = ({ modelName.TIMEDTEXTTRACKS, timedTextTrackId, event.target.files[0], + modelName.VIDEOS, + video.id, ); } catch (error) { if ((error as object).hasOwnProperty('size') && metadata.data) { diff --git a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.spec.tsx b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.spec.tsx index 08a0d23519..519e9557ff 100644 --- a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.spec.tsx +++ b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.spec.tsx @@ -153,6 +153,8 @@ describe('', () => { modelName.SHAREDLIVEMEDIAS, mockedSharedLiveMedia.id, file, + modelName.VIDEOS, + mockedVideo.id, ); }); @@ -386,6 +388,8 @@ describe('', () => { modelName.SHAREDLIVEMEDIAS, mockedSharedLiveMedia.id, file, + modelName.VIDEOS, + mockedVideo.id, ); }); diff --git a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.tsx b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.tsx index 4506bd9c24..637ec548d5 100644 --- a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.tsx +++ b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/SharedLiveMedia/index.tsx @@ -107,6 +107,8 @@ export const SharedLiveMedia = ({ isLive, isTeacher }: SharedMediaProps) => { modelName.SHAREDLIVEMEDIAS, sharedLiveMediaId, event.target.files[0], + modelName.VIDEOS, + video.id, ); } }; diff --git a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.spec.tsx b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.spec.tsx index a459ab957d..beb0a0280a 100644 --- a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.spec.tsx +++ b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.spec.tsx @@ -149,6 +149,8 @@ describe('', () => { modelName.THUMBNAILS, mockedThumbnail.id, file, + modelName.VIDEOS, + mockedVideo.id, ); }); diff --git a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.tsx b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.tsx index 32771c5ec3..d2e7331e96 100644 --- a/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.tsx +++ b/src/frontend/packages/lib_video/src/components/common/VideoWidgetProvider/widgets/WidgetThumbnail/index.tsx @@ -93,7 +93,13 @@ export const WidgetThumbnail = ({ isLive = true }: WidgetThumbnailProps) => { } else { thumbnailId = thumbnail.id; } - addUpload(modelName.THUMBNAILS, thumbnailId, event.target.files[0]); + addUpload( + modelName.THUMBNAILS, + thumbnailId, + event.target.files[0], + modelName.VIDEOS, + video.id, + ); } catch (error) { if ( (error as object).hasOwnProperty('size') &&