2023-11-20 01:47:07 +00:00
|
|
|
import { v4 as uuidv4 } from 'uuid';
|
|
|
|
import sha256 from 'js-sha256';
|
2024-03-25 06:11:55 +00:00
|
|
|
import { getOllamaModels } from '$lib/apis/ollama';
|
|
|
|
import { getOpenAIModels } from '$lib/apis/openai';
|
|
|
|
import { getLiteLLMModels } from '$lib/apis/litellm';
|
|
|
|
|
|
|
|
export const getModels = async (token: string) => {
|
|
|
|
let models = await Promise.all([
|
|
|
|
await getOllamaModels(token).catch((error) => {
|
|
|
|
console.log(error);
|
|
|
|
return null;
|
|
|
|
}),
|
|
|
|
await getOpenAIModels(token).catch((error) => {
|
|
|
|
console.log(error);
|
|
|
|
return null;
|
|
|
|
}),
|
|
|
|
await getLiteLLMModels(token).catch((error) => {
|
|
|
|
console.log(error);
|
|
|
|
return null;
|
|
|
|
})
|
|
|
|
]);
|
|
|
|
|
|
|
|
models = models
|
|
|
|
.filter((models) => models)
|
|
|
|
.reduce((a, e, i, arr) => a.concat(e, ...(i < arr.length - 1 ? [{ name: 'hr' }] : [])), []);
|
|
|
|
|
|
|
|
return models;
|
|
|
|
};
|
2023-11-20 01:47:07 +00:00
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
// Helper functions
|
|
|
|
//////////////////////////
|
|
|
|
|
2024-04-03 17:57:58 +00:00
|
|
|
export const sanitizeResponseContent = (content: string) => {
|
|
|
|
return content
|
|
|
|
.replace(/<\|[a-z]*$/, '')
|
|
|
|
.replace(/<\|[a-z]+\|$/, '')
|
|
|
|
.replace(/<$/, '')
|
|
|
|
.replaceAll(/<\|[a-z]+\|>/g, ' ')
|
|
|
|
.replaceAll(/<br\s?\/?>/gi, '\n')
|
|
|
|
.replaceAll('<', '<')
|
|
|
|
.trim();
|
|
|
|
};
|
|
|
|
|
|
|
|
export const revertSanitizedResponseContent = (content: string) => {
|
|
|
|
return content.replaceAll('<', '<');
|
|
|
|
};
|
|
|
|
|
2024-03-25 20:46:06 +00:00
|
|
|
export const capitalizeFirstLetter = (string) => {
|
|
|
|
return string.charAt(0).toUpperCase() + string.slice(1);
|
|
|
|
};
|
|
|
|
|
2023-11-20 01:47:07 +00:00
|
|
|
export const splitStream = (splitOn) => {
|
|
|
|
let buffer = '';
|
|
|
|
return new TransformStream({
|
|
|
|
transform(chunk, controller) {
|
|
|
|
buffer += chunk;
|
|
|
|
const parts = buffer.split(splitOn);
|
|
|
|
parts.slice(0, -1).forEach((part) => controller.enqueue(part));
|
|
|
|
buffer = parts[parts.length - 1];
|
|
|
|
},
|
|
|
|
flush(controller) {
|
|
|
|
if (buffer) controller.enqueue(buffer);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
export const convertMessagesToHistory = (messages) => {
|
2024-01-03 22:33:57 +00:00
|
|
|
const history = {
|
2023-11-20 01:47:07 +00:00
|
|
|
messages: {},
|
|
|
|
currentId: null
|
|
|
|
};
|
|
|
|
|
|
|
|
let parentMessageId = null;
|
|
|
|
let messageId = null;
|
|
|
|
|
|
|
|
for (const message of messages) {
|
|
|
|
messageId = uuidv4();
|
|
|
|
|
|
|
|
if (parentMessageId !== null) {
|
|
|
|
history.messages[parentMessageId].childrenIds = [
|
|
|
|
...history.messages[parentMessageId].childrenIds,
|
|
|
|
messageId
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
history.messages[messageId] = {
|
|
|
|
...message,
|
|
|
|
id: messageId,
|
|
|
|
parentId: parentMessageId,
|
|
|
|
childrenIds: []
|
|
|
|
};
|
|
|
|
|
|
|
|
parentMessageId = messageId;
|
|
|
|
}
|
|
|
|
|
|
|
|
history.currentId = messageId;
|
|
|
|
return history;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const getGravatarURL = (email) => {
|
|
|
|
// Trim leading and trailing whitespace from
|
|
|
|
// an email address and force all characters
|
|
|
|
// to lower case
|
|
|
|
const address = String(email).trim().toLowerCase();
|
|
|
|
|
|
|
|
// Create a SHA256 hash of the final string
|
|
|
|
const hash = sha256(address);
|
|
|
|
|
|
|
|
// Grab the actual image URL
|
|
|
|
return `https://www.gravatar.com/avatar/${hash}`;
|
|
|
|
};
|
2023-12-19 02:48:51 +00:00
|
|
|
|
2024-04-05 22:04:00 +00:00
|
|
|
export const canvasPixelTest = () => {
|
|
|
|
// Test a 1x1 pixel to potentially identify browser/plugin fingerprint blocking or spoofing
|
|
|
|
// Inspiration: https://github.com/kkapsner/CanvasBlocker/blob/master/test/detectionTest.js
|
2024-04-06 04:02:02 +00:00
|
|
|
const canvas = document.createElement('canvas');
|
2024-04-05 22:04:00 +00:00
|
|
|
const ctx = canvas.getContext('2d');
|
|
|
|
canvas.height = 1;
|
|
|
|
canvas.width = 1;
|
|
|
|
const imageData = new ImageData(canvas.width, canvas.height);
|
|
|
|
const pixelValues = imageData.data;
|
|
|
|
|
|
|
|
// Generate RGB test data
|
2024-04-06 04:02:02 +00:00
|
|
|
for (let i = 0; i < imageData.data.length; i += 1) {
|
|
|
|
if (i % 4 !== 3) {
|
2024-04-05 22:04:00 +00:00
|
|
|
pixelValues[i] = Math.floor(256 * Math.random());
|
2024-04-06 04:02:02 +00:00
|
|
|
} else {
|
2024-04-05 22:04:00 +00:00
|
|
|
pixelValues[i] = 255;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.putImageData(imageData, 0, 0);
|
|
|
|
const p = ctx.getImageData(0, 0, canvas.width, canvas.height).data;
|
|
|
|
|
|
|
|
// Read RGB data and fail if unmatched
|
2024-04-06 04:02:02 +00:00
|
|
|
for (let i = 0; i < p.length; i += 1) {
|
|
|
|
if (p[i] !== pixelValues[i]) {
|
|
|
|
console.log(
|
|
|
|
'canvasPixelTest: Wrong canvas pixel RGB value detected:',
|
|
|
|
p[i],
|
|
|
|
'at:',
|
|
|
|
i,
|
|
|
|
'expected:',
|
|
|
|
pixelValues[i]
|
|
|
|
);
|
|
|
|
console.log('canvasPixelTest: Canvas blocking or spoofing is likely');
|
2024-04-05 22:04:00 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2024-04-06 04:02:02 +00:00
|
|
|
};
|
2024-04-05 22:04:00 +00:00
|
|
|
|
2024-04-04 19:09:07 +00:00
|
|
|
export const generateInitialsImage = (name) => {
|
2024-04-05 03:07:52 +00:00
|
|
|
const canvas = document.createElement('canvas');
|
|
|
|
const ctx = canvas.getContext('2d');
|
|
|
|
canvas.width = 100;
|
|
|
|
canvas.height = 100;
|
2024-04-04 19:09:07 +00:00
|
|
|
|
2024-04-05 22:04:00 +00:00
|
|
|
if (!canvasPixelTest()) {
|
2024-04-06 04:02:02 +00:00
|
|
|
console.log(
|
|
|
|
'generateInitialsImage: failed pixel test, fingerprint evasion is likely. Using default image.'
|
|
|
|
);
|
2024-04-05 22:04:00 +00:00
|
|
|
return '/user.png';
|
|
|
|
}
|
|
|
|
|
2024-04-05 03:07:52 +00:00
|
|
|
ctx.fillStyle = '#F39C12';
|
|
|
|
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
2024-04-04 19:09:07 +00:00
|
|
|
|
2024-04-05 03:07:52 +00:00
|
|
|
ctx.fillStyle = '#FFFFFF';
|
|
|
|
ctx.font = '40px Helvetica';
|
|
|
|
ctx.textAlign = 'center';
|
|
|
|
ctx.textBaseline = 'middle';
|
2024-04-05 02:56:23 +00:00
|
|
|
|
2024-04-05 03:05:39 +00:00
|
|
|
const sanitizedName = name.trim();
|
2024-04-05 03:07:52 +00:00
|
|
|
const initials =
|
|
|
|
sanitizedName.length > 0
|
|
|
|
? sanitizedName[0] +
|
|
|
|
(sanitizedName.split(' ').length > 1
|
|
|
|
? sanitizedName[sanitizedName.lastIndexOf(' ') + 1]
|
|
|
|
: '')
|
|
|
|
: '';
|
2024-04-04 20:26:00 +00:00
|
|
|
|
2024-04-05 03:07:52 +00:00
|
|
|
ctx.fillText(initials.toUpperCase(), canvas.width / 2, canvas.height / 2);
|
2024-04-04 19:09:07 +00:00
|
|
|
|
2024-04-05 03:07:52 +00:00
|
|
|
return canvas.toDataURL();
|
2024-04-04 19:09:07 +00:00
|
|
|
};
|
|
|
|
|
2023-12-26 20:50:52 +00:00
|
|
|
export const copyToClipboard = (text) => {
|
2023-12-19 02:48:51 +00:00
|
|
|
if (!navigator.clipboard) {
|
2023-12-26 20:50:52 +00:00
|
|
|
const textArea = document.createElement('textarea');
|
2023-12-19 02:48:51 +00:00
|
|
|
textArea.value = text;
|
|
|
|
|
|
|
|
// Avoid scrolling to bottom
|
|
|
|
textArea.style.top = '0';
|
|
|
|
textArea.style.left = '0';
|
|
|
|
textArea.style.position = 'fixed';
|
|
|
|
|
|
|
|
document.body.appendChild(textArea);
|
|
|
|
textArea.focus();
|
|
|
|
textArea.select();
|
|
|
|
|
|
|
|
try {
|
2023-12-26 20:50:52 +00:00
|
|
|
const successful = document.execCommand('copy');
|
|
|
|
const msg = successful ? 'successful' : 'unsuccessful';
|
2023-12-19 02:48:51 +00:00
|
|
|
console.log('Fallback: Copying text command was ' + msg);
|
|
|
|
} catch (err) {
|
|
|
|
console.error('Fallback: Oops, unable to copy', err);
|
|
|
|
}
|
|
|
|
|
|
|
|
document.body.removeChild(textArea);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
navigator.clipboard.writeText(text).then(
|
|
|
|
function () {
|
|
|
|
console.log('Async: Copying to clipboard was successful!');
|
|
|
|
},
|
|
|
|
function (err) {
|
|
|
|
console.error('Async: Could not copy text: ', err);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
};
|
2023-12-26 21:10:50 +00:00
|
|
|
|
2024-02-25 19:55:15 +00:00
|
|
|
export const compareVersion = (latest, current) => {
|
2023-12-27 06:51:52 +00:00
|
|
|
return current === '0.0.0'
|
2023-12-27 06:52:53 +00:00
|
|
|
? false
|
2024-02-25 19:55:15 +00:00
|
|
|
: current.localeCompare(latest, undefined, {
|
2023-12-27 06:51:52 +00:00
|
|
|
numeric: true,
|
|
|
|
sensitivity: 'case',
|
|
|
|
caseFirst: 'upper'
|
|
|
|
}) < 0;
|
2023-12-26 21:10:50 +00:00
|
|
|
};
|
2024-01-02 08:55:28 +00:00
|
|
|
|
|
|
|
export const findWordIndices = (text) => {
|
|
|
|
const regex = /\[([^\]]+)\]/g;
|
2024-01-03 22:33:57 +00:00
|
|
|
const matches = [];
|
2024-01-02 08:55:28 +00:00
|
|
|
let match;
|
|
|
|
|
|
|
|
while ((match = regex.exec(text)) !== null) {
|
|
|
|
matches.push({
|
|
|
|
word: match[1],
|
|
|
|
startIndex: match.index,
|
|
|
|
endIndex: regex.lastIndex - 1
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return matches;
|
|
|
|
};
|
2024-01-07 08:57:10 +00:00
|
|
|
|
2024-01-08 07:43:32 +00:00
|
|
|
export const removeFirstHashWord = (inputString) => {
|
|
|
|
// Split the string into an array of words
|
|
|
|
const words = inputString.split(' ');
|
|
|
|
|
|
|
|
// Find the index of the first word that starts with #
|
|
|
|
const index = words.findIndex((word) => word.startsWith('#'));
|
|
|
|
|
|
|
|
// Remove the first word with #
|
|
|
|
if (index !== -1) {
|
|
|
|
words.splice(index, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Join the remaining words back into a string
|
|
|
|
const resultString = words.join(' ');
|
|
|
|
|
|
|
|
return resultString;
|
|
|
|
};
|
|
|
|
|
2024-01-08 09:32:55 +00:00
|
|
|
export const transformFileName = (fileName) => {
|
|
|
|
// Convert to lowercase
|
|
|
|
const lowerCaseFileName = fileName.toLowerCase();
|
|
|
|
|
|
|
|
// Remove special characters using regular expression
|
|
|
|
const sanitizedFileName = lowerCaseFileName.replace(/[^\w\s]/g, '');
|
|
|
|
|
|
|
|
// Replace spaces with dashes
|
|
|
|
const finalFileName = sanitizedFileName.replace(/\s+/g, '-');
|
|
|
|
|
|
|
|
return finalFileName;
|
|
|
|
};
|
|
|
|
|
2024-01-07 08:57:10 +00:00
|
|
|
export const calculateSHA256 = async (file) => {
|
|
|
|
// Create a FileReader to read the file asynchronously
|
|
|
|
const reader = new FileReader();
|
|
|
|
|
|
|
|
// Define a promise to handle the file reading
|
|
|
|
const readFile = new Promise((resolve, reject) => {
|
|
|
|
reader.onload = () => resolve(reader.result);
|
|
|
|
reader.onerror = reject;
|
|
|
|
});
|
|
|
|
|
|
|
|
// Read the file as an ArrayBuffer
|
|
|
|
reader.readAsArrayBuffer(file);
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Wait for the FileReader to finish reading the file
|
|
|
|
const buffer = await readFile;
|
|
|
|
|
|
|
|
// Convert the ArrayBuffer to a Uint8Array
|
|
|
|
const uint8Array = new Uint8Array(buffer);
|
|
|
|
|
|
|
|
// Calculate the SHA-256 hash using Web Crypto API
|
|
|
|
const hashBuffer = await crypto.subtle.digest('SHA-256', uint8Array);
|
|
|
|
|
|
|
|
// Convert the hash to a hexadecimal string
|
|
|
|
const hashArray = Array.from(new Uint8Array(hashBuffer));
|
|
|
|
const hashHex = hashArray.map((byte) => byte.toString(16).padStart(2, '0')).join('');
|
|
|
|
|
2024-01-07 09:40:36 +00:00
|
|
|
return `${hashHex}`;
|
2024-01-07 08:57:10 +00:00
|
|
|
} catch (error) {
|
|
|
|
console.error('Error calculating SHA-256 hash:', error);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
2024-01-17 08:01:11 +00:00
|
|
|
|
|
|
|
export const getImportOrigin = (_chats) => {
|
|
|
|
// Check what external service chat imports are from
|
2024-01-17 22:23:16 +00:00
|
|
|
if ('mapping' in _chats[0]) {
|
|
|
|
return 'openai';
|
|
|
|
}
|
|
|
|
return 'webui';
|
|
|
|
};
|
2024-01-17 08:01:11 +00:00
|
|
|
|
2024-01-17 22:23:16 +00:00
|
|
|
const convertOpenAIMessages = (convo) => {
|
2024-01-17 08:01:11 +00:00
|
|
|
// Parse OpenAI chat messages and create chat dictionary for creating new chats
|
2024-01-17 22:23:16 +00:00
|
|
|
const mapping = convo['mapping'];
|
2024-01-17 08:01:11 +00:00
|
|
|
const messages = [];
|
2024-01-17 22:23:16 +00:00
|
|
|
let currentId = '';
|
2024-01-19 18:22:28 +00:00
|
|
|
let lastId = null;
|
2024-01-17 08:01:11 +00:00
|
|
|
|
2024-01-17 22:23:16 +00:00
|
|
|
for (let message_id in mapping) {
|
|
|
|
const message = mapping[message_id];
|
2024-01-17 08:01:11 +00:00
|
|
|
currentId = message_id;
|
2024-01-19 18:22:28 +00:00
|
|
|
try {
|
2024-01-27 06:17:28 +00:00
|
|
|
if (
|
|
|
|
messages.length == 0 &&
|
|
|
|
(message['message'] == null ||
|
|
|
|
(message['message']['content']['parts']?.[0] == '' &&
|
|
|
|
message['message']['content']['text'] == null))
|
|
|
|
) {
|
2024-01-19 18:22:28 +00:00
|
|
|
// Skip chat messages with no content
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
const new_chat = {
|
|
|
|
id: message_id,
|
|
|
|
parentId: lastId,
|
|
|
|
childrenIds: message['children'] || [],
|
|
|
|
role: message['message']?.['author']?.['role'] !== 'user' ? 'assistant' : 'user',
|
2024-01-27 06:17:28 +00:00
|
|
|
content:
|
|
|
|
message['message']?.['content']?.['parts']?.[0] ||
|
|
|
|
message['message']?.['content']?.['text'] ||
|
|
|
|
'',
|
2024-01-19 18:22:28 +00:00
|
|
|
model: 'gpt-3.5-turbo',
|
|
|
|
done: true,
|
|
|
|
context: null
|
|
|
|
};
|
|
|
|
messages.push(new_chat);
|
|
|
|
lastId = currentId;
|
|
|
|
}
|
|
|
|
} catch (error) {
|
2024-01-27 06:17:28 +00:00
|
|
|
console.log('Error with', message, '\nError:', error);
|
2024-01-17 08:01:11 +00:00
|
|
|
}
|
2024-01-17 22:23:16 +00:00
|
|
|
}
|
2024-01-17 08:01:11 +00:00
|
|
|
|
|
|
|
let history = {};
|
2024-01-17 22:23:16 +00:00
|
|
|
messages.forEach((obj) => (history[obj.id] = obj));
|
2024-01-17 08:01:11 +00:00
|
|
|
|
|
|
|
const chat = {
|
2024-01-17 22:23:16 +00:00
|
|
|
history: {
|
|
|
|
currentId: currentId,
|
|
|
|
messages: history // Need to convert this to not a list and instead a json object
|
2024-01-17 08:01:11 +00:00
|
|
|
},
|
2024-01-17 22:47:56 +00:00
|
|
|
models: ['gpt-3.5-turbo'],
|
2024-01-17 22:23:16 +00:00
|
|
|
messages: messages,
|
|
|
|
options: {},
|
|
|
|
timestamp: convo['create_time'],
|
2024-01-17 22:47:56 +00:00
|
|
|
title: convo['title'] ?? 'New Chat'
|
2024-01-17 22:23:16 +00:00
|
|
|
};
|
|
|
|
return chat;
|
|
|
|
};
|
2024-01-17 08:01:11 +00:00
|
|
|
|
2024-01-19 18:22:28 +00:00
|
|
|
const validateChat = (chat) => {
|
|
|
|
// Because ChatGPT sometimes has features we can't use like DALL-E or migh have corrupted messages, need to validate
|
|
|
|
const messages = chat.messages;
|
|
|
|
|
2024-01-27 06:17:28 +00:00
|
|
|
// Check if messages array is empty
|
|
|
|
if (messages.length === 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Last message's children should be an empty array
|
|
|
|
const lastMessage = messages[messages.length - 1];
|
|
|
|
if (lastMessage.childrenIds.length !== 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// First message's parent should be null
|
|
|
|
const firstMessage = messages[0];
|
|
|
|
if (firstMessage.parentId !== null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Every message's content should be a string
|
|
|
|
for (let message of messages) {
|
|
|
|
if (typeof message.content !== 'string') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2024-01-19 18:22:28 +00:00
|
|
|
};
|
|
|
|
|
2024-01-17 22:23:16 +00:00
|
|
|
export const convertOpenAIChats = (_chats) => {
|
2024-01-17 08:01:11 +00:00
|
|
|
// Create a list of dictionaries with each conversation from import
|
2024-01-17 22:23:16 +00:00
|
|
|
const chats = [];
|
2024-01-19 18:22:28 +00:00
|
|
|
let failed = 0;
|
2024-01-17 22:23:16 +00:00
|
|
|
for (let convo of _chats) {
|
2024-01-17 22:47:56 +00:00
|
|
|
const chat = convertOpenAIMessages(convo);
|
|
|
|
|
2024-01-19 18:22:28 +00:00
|
|
|
if (validateChat(chat)) {
|
2024-01-17 22:47:56 +00:00
|
|
|
chats.push({
|
|
|
|
id: convo['id'],
|
|
|
|
user_id: '',
|
|
|
|
title: convo['title'],
|
|
|
|
chat: chat,
|
|
|
|
timestamp: convo['timestamp']
|
|
|
|
});
|
2024-01-27 06:17:28 +00:00
|
|
|
} else {
|
|
|
|
failed++;
|
|
|
|
}
|
2024-01-17 08:01:11 +00:00
|
|
|
}
|
2024-01-27 06:17:28 +00:00
|
|
|
console.log(failed, 'Conversations could not be imported');
|
2024-01-17 22:23:16 +00:00
|
|
|
return chats;
|
|
|
|
};
|
2024-01-27 06:17:28 +00:00
|
|
|
|
|
|
|
export const isValidHttpUrl = (string) => {
|
|
|
|
let url;
|
|
|
|
|
|
|
|
try {
|
|
|
|
url = new URL(string);
|
|
|
|
} catch (_) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return url.protocol === 'http:' || url.protocol === 'https:';
|
|
|
|
};
|
2024-02-11 03:20:56 +00:00
|
|
|
|
|
|
|
export const removeEmojis = (str) => {
|
|
|
|
// Regular expression to match emojis
|
|
|
|
const emojiRegex = /[\uD800-\uDBFF][\uDC00-\uDFFF]|\uD83C[\uDC00-\uDFFF]|\uD83D[\uDC00-\uDE4F]/g;
|
|
|
|
|
|
|
|
// Replace emojis with an empty string
|
|
|
|
return str.replace(emojiRegex, '');
|
|
|
|
};
|
|
|
|
|
|
|
|
export const extractSentences = (text) => {
|
|
|
|
// Split the paragraph into sentences based on common punctuation marks
|
|
|
|
const sentences = text.split(/(?<=[.!?])/);
|
|
|
|
|
|
|
|
return sentences
|
|
|
|
.map((sentence) => removeEmojis(sentence.trim()))
|
|
|
|
.filter((sentence) => sentence !== '');
|
|
|
|
};
|
2024-02-11 09:06:25 +00:00
|
|
|
|
|
|
|
export const blobToFile = (blob, fileName) => {
|
|
|
|
// Create a new File object from the Blob
|
|
|
|
const file = new File([blob], fileName, { type: blob.type });
|
|
|
|
return file;
|
|
|
|
};
|