mirror of
https://github.com/labring/FastGPT.git
synced 2026-05-02 01:02:05 +08:00
c93c3937e1
* refactor: fastgpt object storage & global proxy (#6155) * feat: migrate to fastgpt storage sdk * chore: rename env variable * chore: move to sdk dir * docs: object storage * CHORE * chore: storage mocks * chore: update docker-compose * fix: global proxy agent * fix: update COS proxy * refactor: use fetch instead of http.request * fix: axios request base url * fix: axios proxy request behavior * fix: bumps axios * fix: patch axios for proxy * fix: replace axios with proxied axios * fix: upload txt file encoding * clean code * fix: use "minio" for minio adapter (#6205) * fix: use minio client to delete files when using minio vendor (#6206) * doc * feat: filter citations and add response button control (#6170) * feat: filter citations and add response button control * i18n * fix * fix test * perf: chat api code * fix: workflow edge overlap and auto-align in folded loop nodes (#6204) * fix: workflow edge overlap and auto-align in folded loop nodes * sort * fix * fix edge * fix icon * perf: s3 file name * perf: admin get app api * perf: catch user error * fix: refactor useOrg hook to use debounced search key (#6180) * chore: comment minio adapter (#6207) * chore: filename with suffix random id * perf: s3 storage code * fix: encode filename when copy object --------- Co-authored-by: archer <545436317@qq.com> * fix: node card link * json * perf: chat index; * index * chat item soft delete (#6216) * chat item soft delete * temp * fix * remove code * perf: delete chat item --------- Co-authored-by: archer <545436317@qq.com> * feat: select wheather filter sensitive info when export apps (#6222) * fix some bugs (#6210) * fix v4.14.5 bugs * type * fix * fix * custom feedback * fix * code * fix * remove invalid function --------- Co-authored-by: archer <545436317@qq.com> * perf: test * fix file default local upload (#6223) * docs: improve object storage introduction (#6224) * doc --------- Co-authored-by: roy <whoeverimf5@gmail.com> Co-authored-by: heheer <heheer@sealos.io> Co-authored-by: Finley Ge <32237950+FinleyGe@users.noreply.github.com>
159 lines
4.2 KiB
TypeScript
159 lines
4.2 KiB
TypeScript
import { getErrText } from '@fastgpt/global/common/error/utils';
|
|
import Papa from 'papaparse';
|
|
import { type AxiosProgressEvent } from 'axios';
|
|
import axios from 'axios';
|
|
import { parseS3UploadError } from '@fastgpt/global/common/error/s3';
|
|
|
|
export const loadFile2Buffer = ({ file, onError }: { file: File; onError?: (err: any) => void }) =>
|
|
new Promise<ArrayBuffer>((resolve, reject) => {
|
|
try {
|
|
let reader = new FileReader();
|
|
reader.readAsArrayBuffer(file);
|
|
reader.onload = async ({ target }) => {
|
|
if (!target?.result) {
|
|
onError?.('Load file error');
|
|
return reject('Load file error');
|
|
}
|
|
try {
|
|
resolve(target.result as ArrayBuffer);
|
|
} catch (err) {
|
|
console.log(err, 'Load file error');
|
|
onError?.(err);
|
|
|
|
reject(getErrText(err, 'Load file error'));
|
|
}
|
|
};
|
|
reader.onerror = (err) => {
|
|
console.log(err, 'Load file error');
|
|
onError?.(err);
|
|
|
|
reject(getErrText(err, 'Load file error'));
|
|
};
|
|
} catch (error) {
|
|
reject('The browser does not support file content reading');
|
|
}
|
|
});
|
|
|
|
export const readFileRawText = ({
|
|
file,
|
|
onError
|
|
}: {
|
|
file: File;
|
|
onError?: (err: any) => void;
|
|
}) => {
|
|
return new Promise<string>((resolve, reject) => {
|
|
try {
|
|
let reader = new FileReader();
|
|
reader.onload = async ({ target }) => {
|
|
if (!target?.result) {
|
|
onError?.('Load file error');
|
|
return reject('Load file error');
|
|
}
|
|
try {
|
|
resolve(target.result as string);
|
|
} catch (err) {
|
|
console.log(err, 'Load file error');
|
|
onError?.(err);
|
|
|
|
reject(getErrText(err, 'Load file error'));
|
|
}
|
|
};
|
|
reader.onerror = (err) => {
|
|
console.log(err, 'Load file error');
|
|
onError?.(err);
|
|
|
|
reject(getErrText(err, 'Load file error'));
|
|
};
|
|
detectFileEncoding(file).then((encoding) => {
|
|
console.log(encoding);
|
|
|
|
reader.readAsText(
|
|
file,
|
|
['iso-8859-1', 'windows-1252'].includes(encoding) ? 'gb2312' : 'utf-8'
|
|
);
|
|
});
|
|
} catch (error) {
|
|
reject('The browser does not support file content reading');
|
|
}
|
|
});
|
|
};
|
|
|
|
export const readCsvRawText = async ({ file }: { file: File }) => {
|
|
const rawText = await readFileRawText({ file });
|
|
const csvArr = Papa.parse(rawText).data as string[][];
|
|
return csvArr;
|
|
};
|
|
|
|
async function detectFileEncoding(file: File): Promise<string> {
|
|
const buffer = await loadFile2Buffer({ file });
|
|
const encoding = (() => {
|
|
const encodings = ['utf-8', 'iso-8859-1', 'windows-1252'];
|
|
for (let encoding of encodings) {
|
|
try {
|
|
const decoder = new TextDecoder(encoding, { fatal: true });
|
|
decoder.decode(buffer);
|
|
return encoding; // 如果解码成功,返回当前编码
|
|
} catch (e) {
|
|
// continue to try next encoding
|
|
}
|
|
}
|
|
return null; // 如果没有编码匹配,返回null
|
|
})();
|
|
|
|
return encoding || 'utf-8';
|
|
}
|
|
|
|
export const fileToBase64 = (file: File) => {
|
|
return new Promise<string>((resolve, reject) => {
|
|
const reader = new FileReader();
|
|
reader.readAsDataURL(file);
|
|
reader.onload = () => resolve(reader.result as string);
|
|
reader.onerror = (error) => reject(error);
|
|
});
|
|
};
|
|
|
|
export const base64ToFile = (base64: string, filename: string) => {
|
|
const arr = base64.split(',');
|
|
const mime = arr[0].match(/:(.*?);/)?.[1];
|
|
const bstr = atob(arr[1]);
|
|
let n = bstr.length;
|
|
const u8arr = new Uint8Array(n);
|
|
while (n--) {
|
|
u8arr[n] = bstr.charCodeAt(n);
|
|
}
|
|
return new File([u8arr], filename, { type: mime });
|
|
};
|
|
|
|
export const putFileToS3 = async ({
|
|
headers,
|
|
url,
|
|
file,
|
|
onSuccess,
|
|
onUploadProgress,
|
|
maxSize,
|
|
t
|
|
}: {
|
|
headers?: Record<string, string>;
|
|
url: string;
|
|
file: File;
|
|
onSuccess?: () => void;
|
|
onUploadProgress?: (progressEvent: AxiosProgressEvent) => void;
|
|
maxSize?: number;
|
|
t: any;
|
|
}) => {
|
|
try {
|
|
const res = await axios.put(url, file, {
|
|
headers: {
|
|
...headers
|
|
},
|
|
onUploadProgress,
|
|
timeout: 5 * 60 * 1000
|
|
});
|
|
if (res.status === 200) {
|
|
onSuccess?.();
|
|
}
|
|
} catch (error) {
|
|
Promise.reject(parseS3UploadError({ t, error, maxSize }));
|
|
}
|
|
};
|