// services import { APIService } from "services/api.service"; // helpers import { API_BASE_URL } from "helpers/common.helper"; import axios from "axios"; export interface UnSplashImage { id: string; created_at: Date; updated_at: Date; promoted_at: Date; width: number; height: number; color: string; blur_hash: string; description: null; alt_description: string; urls: UnSplashImageUrls; [key: string]: any; } export interface UnSplashImageUrls { raw: string; full: string; regular: string; small: string; thumb: string; small_s3: string; } export class FileService extends APIService { private cancelSource: any; constructor() { super(API_BASE_URL); this.uploadFile = this.uploadFile.bind(this); this.deleteImage = this.deleteImage.bind(this); this.restoreImage = this.restoreImage.bind(this); this.cancelUpload = this.cancelUpload.bind(this); } async attachAssetToIssue(workspaceSlug: string, projectId: string, issueId: string, assetId: string): Promise { const attachUrl = `/api/workspaces/${workspaceSlug}/projects/${projectId}/issues/${issueId}/attachments/${assetId}`; return this.get(attachUrl, { responseType: "blob", headers: this.getHeaders(), }) .then(async (response) => response?.data) .catch((error) => { console.log(error); throw error?.response?.data; }); } getAttachAssetToIssueFile(workspaceSlug: string, projectId: string, issueId: string) { return async (assetId: string) => { try { const data = await this.attachAssetToIssue(workspaceSlug, projectId, issueId, assetId); return data as Blob; } catch (e) { console.error(e); } }; } async uploadFile(workspaceSlug: string, projectId: string, issueId: string, file: FormData): Promise { this.cancelSource = axios.CancelToken.source(); return this.post(`/api/workspaces/${workspaceSlug}/file-assets/`, file, { headers: { ...this.getHeaders(), "Content-Type": "multipart/form-data", }, cancelToken: this.cancelSource.token, }) .then(async (response) => response?.data.asset) .catch((error) => { if (axios.isCancel(error)) { console.log(error.message); } else { console.log(error); throw error?.response?.data; } }); } cancelUpload() { this.cancelSource.cancel("Upload cancelled"); } getUploadFileFunction(workspaceSlug: string, projectId: string, issueId: string): (file: File) => Promise { return async (file: File) => { try { const formData = new FormData(); formData.append("asset", file); formData.append("attributes", JSON.stringify({})); const data = await this.uploadFile(workspaceSlug, projectId, issueId, formData); return data; } catch (e) { console.error(e); } }; } getDeleteImageFunction(workspaceId: string) { return async (src: string) => { try { const assetUrlWithWorkspaceId = `${workspaceId}/${this.extractAssetIdFromUrl(src, workspaceId)}`; const data = await this.deleteImage(assetUrlWithWorkspaceId); return data; } catch (e) { console.error(e); } }; } getRestoreImageFunction(workspaceId: string) { return async (src: string) => { try { const assetUrlWithWorkspaceId = `${workspaceId}/${this.extractAssetIdFromUrl(src, workspaceId)}`; const data = await this.restoreImage(assetUrlWithWorkspaceId); return data; } catch (e) { console.error(e); } }; } extractAssetIdFromUrl(src: string, workspaceId: string): string { const indexWhereAssetIdStarts = src.indexOf(workspaceId) + workspaceId.length + 1; if (indexWhereAssetIdStarts === -1) { throw new Error("Workspace ID not found in source string"); } const assetUrl = src.substring(indexWhereAssetIdStarts); return assetUrl; } async deleteImage(assetUrlWithWorkspaceId: string): Promise { return this.delete(`/api/workspaces/file-assets/${assetUrlWithWorkspaceId}/`) .then((response) => response?.status) .catch((error) => { throw error?.response?.data; }); } async restoreImage(assetUrlWithWorkspaceId: string): Promise { return this.post(`/api/workspaces/file-assets/${assetUrlWithWorkspaceId}/restore/`, { headers: this.getHeaders(), "Content-Type": "application/json", }) .then((response) => response?.status) .catch((error) => { throw error?.response?.data; }); } async deleteFile(workspaceId: string, assetUrl: string): Promise { const lastIndex = assetUrl.lastIndexOf("/"); const assetId = assetUrl.substring(lastIndex + 1); return this.delete(`/api/workspaces/file-assets/${workspaceId}/${assetId}/`) .then((response) => response?.data) .catch((error) => { throw error?.response?.data; }); } async uploadUserFile(file: FormData): Promise { return this.post(`/api/users/file-assets/`, file, { headers: { ...this.getHeaders(), "Content-Type": "multipart/form-data", }, }) .then((response) => response?.data) .catch((error) => { throw error?.response?.data; }); } async deleteUserFile(assetUrl: string): Promise { const lastIndex = assetUrl.lastIndexOf("/"); const assetId = assetUrl.substring(lastIndex + 1); return this.delete(`/api/users/file-assets/${assetId}`) .then((response) => response?.data) .catch((error) => { throw error?.response?.data; }); } async getUnsplashImages(query?: string): Promise { return this.get(`/api/unsplash/`, { params: { query, }, }) .then((res) => res?.data?.results ?? res?.data) .catch((err) => { throw err?.response?.data; }); } async getProjectCoverImages(): Promise { return this.get(`/api/project-covers/`) .then((res) => res?.data) .catch((err) => { throw err?.response?.data; }); } }