mirror of
https://github.com/filecoin-project/slate.git
synced 2025-01-01 05:38:20 +03:00
441 lines
12 KiB
JavaScript
441 lines
12 KiB
JavaScript
import * as Websockets from "~/common/browser-websockets";
|
|
import * as Credentials from "~/common/credentials";
|
|
import * as Actions from "~/common/actions";
|
|
import * as Window from "~/common/window";
|
|
import * as Validations from "~/common/validations";
|
|
import * as Strings from "~/common/strings";
|
|
import * as Store from "~/common/store";
|
|
import * as FileUtilities from "~/common/file-utilities";
|
|
import * as Events from "~/common/custom-events";
|
|
|
|
import Cookies from "universal-cookie";
|
|
import JSZip from "jszip";
|
|
|
|
import { v4 as uuid } from "uuid";
|
|
import { saveAs } from "file-saver";
|
|
|
|
//NOTE(martina): this file is for utility *API-calling* functions
|
|
//For non API related utility functions, see common/utilities.js
|
|
//And for uploading related utility functions, see common/file-utilities.js
|
|
|
|
const cookies = new Cookies();
|
|
|
|
export const authenticate = async (state) => {
|
|
// NOTE(jim): Kills existing session cookie if there is one.
|
|
const jwt = cookies.get(Credentials.session.key);
|
|
|
|
if (jwt) {
|
|
cookies.remove(Credentials.session.key, {
|
|
path: "/",
|
|
maxAge: 3600 * 24 * 7,
|
|
sameSite: "strict",
|
|
});
|
|
}
|
|
|
|
let response = await Actions.signIn(state);
|
|
if (!response || response.error) {
|
|
return response;
|
|
}
|
|
|
|
if (response.token) {
|
|
// NOTE(jim):
|
|
// + One week.
|
|
// + Only requests to the same site.
|
|
// + Not using sessionStorage so the cookie doesn't leave when the browser dies.
|
|
cookies.set(Credentials.session.key, response.token, {
|
|
path: "/",
|
|
maxAge: 3600 * 24 * 7,
|
|
sameSite: "strict",
|
|
});
|
|
}
|
|
|
|
return response;
|
|
};
|
|
|
|
export const authenticateViaTwitter = (response) => {
|
|
// NOTE(jim): Kills existing session cookie if there is one.
|
|
const jwt = cookies.get(Credentials.session.key);
|
|
|
|
if (jwt) {
|
|
cookies.remove(Credentials.session.key, {
|
|
path: "/",
|
|
maxAge: 3600 * 24 * 7,
|
|
sameSite: "strict",
|
|
});
|
|
}
|
|
|
|
if (Events.hasError(response)) {
|
|
return false;
|
|
}
|
|
|
|
if (response.token) {
|
|
// NOTE(jim):
|
|
// + One week.
|
|
// + Only requests to the same site.
|
|
// + Not using sessionStorage so the cookie doesn't leave when the browser dies.
|
|
cookies.set(Credentials.session.key, response.token, {
|
|
path: "/",
|
|
maxAge: 3600 * 24 * 7,
|
|
sameSite: "strict",
|
|
});
|
|
}
|
|
|
|
return response;
|
|
};
|
|
|
|
// NOTE(jim): Signs a user out and redirects to the sign in screen.
|
|
export const signOut = async ({ viewer }) => {
|
|
let wsclient = Websockets.getClient();
|
|
if (wsclient) {
|
|
wsclient.send(JSON.stringify({ type: "UNSUBSCRIBE_VIEWER", data: viewer }));
|
|
|
|
await Websockets.deleteClient();
|
|
wsclient = null;
|
|
}
|
|
|
|
const jwt = cookies.get(Credentials.session.key);
|
|
if (jwt) {
|
|
cookies.remove(Credentials.session.key, {
|
|
path: "/",
|
|
maxAge: 3600 * 24 * 7,
|
|
sameSite: "strict",
|
|
});
|
|
}
|
|
|
|
window.location.replace("/_/auth");
|
|
return;
|
|
};
|
|
|
|
// NOTE(jim): Permanently deletes you, forever.
|
|
export const deleteMe = async ({ viewer }) => {
|
|
await Actions.updateSearch("delete-user");
|
|
|
|
let response = await Actions.deleteViewer();
|
|
|
|
if (Events.hasError(response)) {
|
|
return response;
|
|
}
|
|
|
|
await signOut({ viewer });
|
|
|
|
// let wsclient = Websockets.getClient();
|
|
// if (wsclient) {
|
|
// await Websockets.deleteClient();
|
|
// wsclient = null;
|
|
// }
|
|
|
|
return response;
|
|
};
|
|
|
|
export const hydrate = async () => {
|
|
const response = await Actions.hydrateAuthenticatedUser();
|
|
|
|
if (Events.hasError(response)) {
|
|
return null;
|
|
}
|
|
|
|
return JSON.parse(JSON.stringify(response.data));
|
|
};
|
|
|
|
export const formatPastedImages = ({ clipboardItems }) => {
|
|
let files = [];
|
|
let fileLoading = {};
|
|
for (let i = 0; i < clipboardItems.length; i++) {
|
|
// Note(Amine): skip content if it's not an image
|
|
if (clipboardItems[i].type.indexOf("image") === -1) continue;
|
|
const file = clipboardItems[i].getAsFile();
|
|
files.push(file);
|
|
fileLoading[`${file.lastModified}-${file.name}`] = {
|
|
name: file.name,
|
|
loaded: 0,
|
|
total: file.size,
|
|
};
|
|
}
|
|
return { fileLoading, toUpload: files };
|
|
};
|
|
|
|
export const formatDroppedFiles = async ({ dataTransfer }) => {
|
|
// NOTE(jim): If this is true, then drag and drop came from a slate object.
|
|
const data = dataTransfer.getData("slate-object-drag-data");
|
|
if (data) {
|
|
return;
|
|
}
|
|
|
|
const files = [];
|
|
let fileLoading = {};
|
|
if (dataTransfer.items && dataTransfer.items.length) {
|
|
for (var i = 0; i < dataTransfer.items.length; i++) {
|
|
const data = dataTransfer.items[i];
|
|
|
|
let file = null;
|
|
if (data.kind === "file") {
|
|
file = data.getAsFile();
|
|
} else if (data.kind == "string" && data.type == "text/uri-list") {
|
|
try {
|
|
const dataAsString = new Promise((resolve, reject) =>
|
|
data.getAsString((d) => resolve(d))
|
|
);
|
|
const resp = await fetch(await dataAsString);
|
|
const blob = resp.blob();
|
|
|
|
file = new File(blob, `data-${uuid()}`);
|
|
file.name = `data-${uuid()}`;
|
|
} catch (e) {
|
|
Events.dispatchMessage({
|
|
message: "File type not supported. Please try a different file",
|
|
});
|
|
|
|
return { error: true };
|
|
}
|
|
}
|
|
|
|
files.push(file);
|
|
fileLoading[`${file.lastModified}-${file.name}`] = {
|
|
name: file.name,
|
|
loaded: 0,
|
|
total: file.size,
|
|
};
|
|
}
|
|
}
|
|
|
|
if (!files.length) {
|
|
Events.dispatchMessage({ message: "File type not supported. Please try a different file" });
|
|
}
|
|
|
|
return { fileLoading, files, numFailed: dataTransfer.items.length - files.length };
|
|
};
|
|
|
|
export const formatUploadedFiles = ({ files }) => {
|
|
let toUpload = [];
|
|
let fileLoading = {};
|
|
for (let i = 0; i < files.length; i++) {
|
|
let file = files[i];
|
|
|
|
if (!file) {
|
|
continue;
|
|
}
|
|
|
|
toUpload.push(file);
|
|
fileLoading[`${file.lastModified}-${file.name}`] = {
|
|
name: file.name,
|
|
loaded: 0,
|
|
total: file.size,
|
|
};
|
|
}
|
|
|
|
if (!toUpload.length) {
|
|
Events.dispatchMessage({ message: "We could not find any files to upload." });
|
|
return false;
|
|
}
|
|
|
|
return { toUpload, fileLoading, numFailed: files.length - toUpload.length };
|
|
};
|
|
|
|
export const uploadImage = async (file, resources, excludeFromLibrary) => {
|
|
if (!file) {
|
|
Events.dispatchMessage({ message: "Something went wrong with the upload. Please try again" });
|
|
return;
|
|
}
|
|
|
|
if (!Validations.isPreviewableImage(file.type)) {
|
|
Events.dispatchMessage({ message: "Upload failed. Only images and gifs are allowed" });
|
|
return;
|
|
}
|
|
|
|
const response = await FileUtilities.upload({ file, routes: resources });
|
|
|
|
if (Events.hasError(response)) {
|
|
return false;
|
|
}
|
|
|
|
return response;
|
|
};
|
|
|
|
export const deleteFiles = async (fileIds = [], noAlert) => {
|
|
let ids = Array.isArray(fileIds) ? fileIds : [fileIds];
|
|
|
|
const response = await Actions.deleteFiles({ ids });
|
|
|
|
if (!noAlert) {
|
|
if (Events.hasError(response)) {
|
|
return false;
|
|
}
|
|
|
|
Events.dispatchMessage({ message: "Files successfully deleted!", status: "INFO" });
|
|
|
|
return response;
|
|
}
|
|
};
|
|
|
|
export const removeFromSlate = async ({ slate, ids }) => {
|
|
const response = await Actions.removeFileFromSlate({
|
|
slateId: slate.id,
|
|
ids,
|
|
});
|
|
|
|
if (Events.hasError(response)) {
|
|
return false;
|
|
}
|
|
|
|
return response;
|
|
};
|
|
|
|
// export const addToSlate = async ({ slate, files }) => {
|
|
// const addResponse = await Actions.addFileToSlate({ slate, files });
|
|
|
|
// if (Events.hasError(addResponse)) {
|
|
// return false;
|
|
// }
|
|
|
|
// const { added, skipped } = addResponse;
|
|
// let message = Strings.formatAsUploadMessage(added, skipped, true);
|
|
// Events.dispatchMessage({ message, status: !added ? null : "INFO" });
|
|
// return true;
|
|
// };
|
|
|
|
//NOTE(martina): save copy includes add to slate now. If it's already in the user's files but not in that slate, it'll skip the adding to files and just add to slate
|
|
export const saveCopy = async ({ files, slate }) => {
|
|
console.log("user behaviors save copy");
|
|
let response = await Actions.saveCopy({ files, slate });
|
|
if (Events.hasError(response)) {
|
|
return false;
|
|
}
|
|
let message = Strings.formatAsUploadMessage(response.data.added, response.data.skipped, slate);
|
|
Events.dispatchMessage({ message, status: !response.data.added ? null : "INFO" });
|
|
return response;
|
|
};
|
|
|
|
export const download = async (file) => {
|
|
Actions.createDownloadActivity({ file });
|
|
if (file.data.type === "application/unity") {
|
|
return await downloadZip(file);
|
|
}
|
|
let uri = Strings.getURLfromCID(file.cid);
|
|
Window.saveAs(uri, file.filename);
|
|
return { data: true };
|
|
};
|
|
|
|
export const downloadZip = async (file) => {
|
|
try {
|
|
const { data } = await Actions.getZipFilePaths(file);
|
|
const filesPaths = data.filesPaths.map((item) => item.replace(`/${file.id}/`, ""));
|
|
const baseUrl = Strings.getURLfromCID(file.cid);
|
|
const zipFileName = file.filename;
|
|
|
|
let zip = new JSZip();
|
|
|
|
await Promise.all(
|
|
filesPaths.map(async (filePath) => {
|
|
let url = `${baseUrl}/${filePath}`;
|
|
const blob = await Window.getBlobFromUrl(url);
|
|
|
|
zip.file(filePath, blob);
|
|
})
|
|
);
|
|
|
|
zip.generateAsync({ type: "blob" }).then((blob) => {
|
|
saveAs(blob, zipFileName);
|
|
});
|
|
|
|
return {
|
|
decorator: "UNITY_ZIP_DOWNLOAD_SUCCESS",
|
|
error: false,
|
|
};
|
|
} catch (e) {
|
|
return {
|
|
decorator: "UNITY_ZIP_DOWNLOAD_FAILED",
|
|
error: true,
|
|
};
|
|
}
|
|
};
|
|
|
|
const _nativeDownload = ({ url, onError }) => {
|
|
const iframe = document.createElement("iframe");
|
|
iframe.style.display = "none";
|
|
iframe.src = url;
|
|
|
|
const ERROR_MESSAGE = "SLATE_DOWNLOAD_ERROR";
|
|
const handleIframeErrors = (e) => {
|
|
if (e.data === ERROR_MESSAGE && onError) {
|
|
onError(e.data);
|
|
}
|
|
};
|
|
window.addEventListener("message", handleIframeErrors);
|
|
iframe.onload = (e) => window.removeEventListener("message", handleIframeErrors);
|
|
|
|
document.body.appendChild(iframe);
|
|
};
|
|
|
|
export const compressAndDownloadFiles = async ({ files, name = "slate.zip", resourceURI }) => {
|
|
Actions.createDownloadActivity({ files });
|
|
const errorMessage = "Something went wrong with the download. Please try again";
|
|
try {
|
|
if (!(files && files.length > 0)) {
|
|
Events.dispatchMessage({ message: "No files in collection to download" });
|
|
return;
|
|
}
|
|
Events.dispatchMessage({ message: "We're preparing your files to download", status: "INFO" });
|
|
let downloadFiles = [];
|
|
for (const file of files) {
|
|
if (file.type === "application/unity") {
|
|
const { data } = await Actions.getZipFilePaths(file);
|
|
const unityFiles = data.filesPaths.map((item) => ({
|
|
url: item.replace(`/${file.id}/`, `${Strings.getURLfromCID(file.cid)}/`),
|
|
name: item.replace(`/${file.id}/`, `/${file.filename}/`),
|
|
}));
|
|
|
|
downloadFiles = downloadFiles.concat(unityFiles);
|
|
continue;
|
|
}
|
|
|
|
downloadFiles.push({
|
|
name: file.filename,
|
|
url: Strings.getURLfromCID(file.cid),
|
|
});
|
|
}
|
|
|
|
const res = await Actions.createZipToken({ files: downloadFiles, resourceURI });
|
|
const downloadLink = Actions.downloadZip({ token: res.data.token, name, resourceURI });
|
|
await _nativeDownload({
|
|
url: downloadLink,
|
|
onError: (err) =>
|
|
Events.dispatchMessage({
|
|
message: errorMessage,
|
|
}),
|
|
});
|
|
} catch (e) {
|
|
console.error(e);
|
|
Events.dispatchMessage({ message: errorMessage });
|
|
}
|
|
};
|
|
|
|
// export const createSlate = async (data) => {
|
|
// let response = await Actions.createSlate({
|
|
// name: data.name,
|
|
// isPublic: data.isPublic,
|
|
// body: data.body,
|
|
// });
|
|
// return response;
|
|
// }
|
|
|
|
// export const createWalletAddress = async (data) => {
|
|
// let response = await Actions.updateViewer({
|
|
// type: "CREATE_FILECOIN_ADDRESS",
|
|
// address: {
|
|
// name: data.name,
|
|
// type: data.wallet_type,
|
|
// makeDefault: data.makeDefault,
|
|
// },
|
|
// });
|
|
// return response;
|
|
// }
|
|
|
|
// export const sendWalletAddressFilecoin = (data) => {
|
|
// let response = await Actions.sendFilecoin({
|
|
// source: data.source,
|
|
// target: data.target,
|
|
// amount: data.amount,
|
|
// });
|
|
// return response;
|
|
// }
|