mirror of
https://github.com/StanGirard/quivr.git
synced 2025-01-05 18:57:48 +03:00
f69c64ead1
* feature: responsive navbar * style: nav links hover animatiosn * style: better Input Fields * refactor: use form submit instead of button onclick * feature: loading states * feature: log out confirmation * feature: basic toast * feature: Toast variants * fix: use global toast provider * feature: use toast instead of alert for auth routes * fix(mobile): nav menu close on route change * fix: field dark mode * feature: redirect when login and logout * refactor: group auth routes * refactor: use @/app imports * style: use Field on /upload * fix: forward ref * feature: Multi toast * feature: add toasts to /upload * refactor: new login in auth group * chore: quote * chore(pnpm): removed * feature: toasty animations * fix: build errors and warnings * chore: remove irrelevant comments * fix: use unique ids for toasts --------- Co-authored-by: Stan Girard <girard.stanislas@gmail.com>
281 lines
8.1 KiB
TypeScript
281 lines
8.1 KiB
TypeScript
"use client";
|
|
import {
|
|
Dispatch,
|
|
SetStateAction,
|
|
useCallback,
|
|
useState,
|
|
useRef,
|
|
useEffect,
|
|
} from "react";
|
|
import { FileRejection, useDropzone } from "react-dropzone";
|
|
import axios from "axios";
|
|
import { Message } from "@/lib/types";
|
|
import Button from "../components/ui/Button";
|
|
import { MdClose } from "react-icons/md";
|
|
import { AnimatePresence, motion } from "framer-motion";
|
|
import Link from "next/link";
|
|
import Card from "../components/ui/Card";
|
|
import PageHeading from "../components/ui/PageHeading";
|
|
import { useSupabase } from "../supabase-provider";
|
|
import { redirect } from "next/navigation";
|
|
import Field from "../components/ui/Field";
|
|
import Toast, { ToastRef } from "../components/ui/Toast";
|
|
|
|
export default function UploadPage() {
|
|
const [message, setMessage] = useState<Message | null>(null);
|
|
const [isPending, setIsPending] = useState(false);
|
|
const [files, setFiles] = useState<File[]>([]);
|
|
const [pendingFileIndex, setPendingFileIndex] = useState<number>(0);
|
|
const urlInputRef = useRef<HTMLInputElement | null>(null);
|
|
const { supabase, session } = useSupabase();
|
|
if (session === null) {
|
|
redirect("/login");
|
|
}
|
|
|
|
const messageToast = useRef<ToastRef>(null);
|
|
|
|
useEffect(() => {
|
|
if (!message) return;
|
|
messageToast.current?.publish({
|
|
variant:
|
|
message.type === "error"
|
|
? "danger"
|
|
: message.type === "warning"
|
|
? "neutral"
|
|
: "success",
|
|
text: message.text,
|
|
});
|
|
}, [message]);
|
|
|
|
const crawlWebsite = useCallback(async () => {
|
|
// Validate URL
|
|
const url = urlInputRef.current ? urlInputRef.current.value : null;
|
|
if (!url || !isValidUrl(url)) {
|
|
// Assuming you have a function to validate URLs
|
|
setMessage({
|
|
type: "error",
|
|
text: "Invalid URL",
|
|
});
|
|
return;
|
|
}
|
|
|
|
// Configure parameters
|
|
const config = {
|
|
url: url,
|
|
js: false,
|
|
depth: 1,
|
|
max_pages: 100,
|
|
max_time: 60,
|
|
};
|
|
|
|
try {
|
|
const response = await axios.post(
|
|
`${process.env.NEXT_PUBLIC_BACKEND_URL}/crawl`,
|
|
config,
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${session.access_token}`,
|
|
},
|
|
}
|
|
);
|
|
|
|
setMessage({
|
|
type: response.data.type,
|
|
text: response.data.message,
|
|
});
|
|
} catch (error: any) {
|
|
setMessage({
|
|
type: "error",
|
|
text: "Failed to crawl website: " + error.toString(),
|
|
});
|
|
}
|
|
}, [session.access_token]);
|
|
|
|
const upload = useCallback(
|
|
async (file: File) => {
|
|
const formData = new FormData();
|
|
formData.append("file", file);
|
|
try {
|
|
const response = await axios.post(
|
|
`${process.env.NEXT_PUBLIC_BACKEND_URL}/upload`,
|
|
formData,
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${session.access_token}`,
|
|
},
|
|
}
|
|
);
|
|
|
|
setMessage({
|
|
type: response.data.type,
|
|
text:
|
|
(response.data.type === "success"
|
|
? "File uploaded successfully: "
|
|
: "") + JSON.stringify(response.data.message),
|
|
});
|
|
} catch (error: any) {
|
|
setMessage({
|
|
type: "error",
|
|
text: "Failed to upload file: " + error.toString(),
|
|
});
|
|
}
|
|
},
|
|
[session.access_token]
|
|
);
|
|
|
|
const onDrop = (acceptedFiles: File[], fileRejections: FileRejection[]) => {
|
|
if (fileRejections.length > 0) {
|
|
setMessage({ type: "error", text: "File too big." });
|
|
return;
|
|
}
|
|
setMessage(null);
|
|
for (let i = 0; i < acceptedFiles.length; i++) {
|
|
const file = acceptedFiles[i];
|
|
const isAlreadyInFiles =
|
|
files.filter((f) => f.name === file.name && f.size === file.size)
|
|
.length > 0;
|
|
if (isAlreadyInFiles) {
|
|
setMessage({ type: "warning", text: `${file.name} was already added` });
|
|
acceptedFiles.splice(i, 1);
|
|
}
|
|
}
|
|
setFiles((files) => [...files, ...acceptedFiles]);
|
|
};
|
|
|
|
const uploadAllFiles = async () => {
|
|
setIsPending(true);
|
|
setMessage(null);
|
|
// files.forEach((file) => upload(file));
|
|
for (const file of files) {
|
|
await upload(file);
|
|
setPendingFileIndex((i) => i + 1);
|
|
}
|
|
setPendingFileIndex(0);
|
|
setIsPending(false);
|
|
};
|
|
|
|
const { getRootProps, getInputProps, isDragActive, open } = useDropzone({
|
|
onDrop,
|
|
noClick: true,
|
|
maxSize: 100000000, // 1 MB
|
|
});
|
|
|
|
return (
|
|
<main>
|
|
<section
|
|
{...getRootProps()}
|
|
// className="w-full h-full min-h-screen text-center flex flex-col items-center gap-5 pt-32 outline-none"
|
|
className="w-full outline-none pt-32 flex flex-col gap-5 items-center justify-center p-6"
|
|
>
|
|
<PageHeading
|
|
title="Upload Knowledge"
|
|
subtitle="Text, document, spreadsheet, presentation, audio, video, and URLs supported"
|
|
/>
|
|
{/* Wrap the cards in a flex container */}
|
|
<div className="flex justify-center gap-5">
|
|
{/* Assign a width of 50% to each card */}
|
|
<Card className="w-1/2">
|
|
<input {...getInputProps()} />
|
|
<div className="text-center mt-2 p-6 max-w-sm w-full flex flex-col gap-5 items-center">
|
|
{files.length > 0 ? (
|
|
<AnimatePresence>
|
|
{files.map((file, i) => (
|
|
<FileComponent
|
|
key={file.name + file.size}
|
|
file={file}
|
|
setFiles={setFiles}
|
|
/>
|
|
))}
|
|
</AnimatePresence>
|
|
) : null}
|
|
|
|
{isDragActive ? (
|
|
<p className="text-blue-600">Drop the files here...</p>
|
|
) : (
|
|
<button
|
|
onClick={open}
|
|
className="opacity-50 cursor-pointer hover:opacity-100 hover:underline transition-opacity"
|
|
>
|
|
Drag and drop files here, or click to browse
|
|
</button>
|
|
)}
|
|
</div>
|
|
</Card>
|
|
{/* Assign a width of 50% to each card */}
|
|
<Card className="w-1/2">
|
|
<div className="text-center mt-2 p-6 max-w-sm w-full flex flex-col gap-5 items-center">
|
|
<Field
|
|
name="crawlurl"
|
|
ref={urlInputRef}
|
|
type="text"
|
|
placeholder="Enter a website URL"
|
|
/>
|
|
<button
|
|
onClick={crawlWebsite}
|
|
className="opacity-50 cursor-pointer hover:opacity-100 hover:underline transition-opacity"
|
|
>
|
|
Crawl
|
|
</button>
|
|
</div>
|
|
</Card>
|
|
</div>
|
|
<div className="flex flex-col items-center justify-center gap-5">
|
|
<Button isLoading={isPending} onClick={uploadAllFiles} className="">
|
|
{isPending ? `Uploading ${files[pendingFileIndex].name}` : "Upload"}
|
|
</Button>
|
|
<Link href={"/chat"}>
|
|
<Button variant={"secondary"} className="py-3">
|
|
Chat
|
|
</Button>
|
|
</Link>
|
|
</div>
|
|
</section>
|
|
<Toast ref={messageToast} />
|
|
</main>
|
|
);
|
|
}
|
|
|
|
const FileComponent = ({
|
|
file,
|
|
setFiles,
|
|
}: {
|
|
file: File;
|
|
setFiles: Dispatch<SetStateAction<File[]>>;
|
|
}) => {
|
|
return (
|
|
<motion.div
|
|
initial={{ x: "-10%", opacity: 0 }}
|
|
animate={{ x: "0%", opacity: 1 }}
|
|
exit={{ x: "10%", opacity: 0 }}
|
|
className="flex py-2 dark:bg-black border-b border-black/10 dark:border-white/25 w-full text-left leading-none"
|
|
>
|
|
<div className="flex flex-col gap-1 flex-1">
|
|
<span className="flex-1 mr-10">{file.name}</span>
|
|
<span className="text-xs opacity-50">
|
|
{(file.size / 1000).toFixed(3)} kb
|
|
</span>
|
|
</div>
|
|
<button
|
|
role="remove file"
|
|
className="ml-5 text-xl text-red-500 px-5"
|
|
onClick={() =>
|
|
setFiles((files) =>
|
|
files.filter((f) => f.name !== file.name || f.size !== file.size)
|
|
)
|
|
}
|
|
>
|
|
<MdClose />
|
|
</button>
|
|
</motion.div>
|
|
);
|
|
};
|
|
|
|
function isValidUrl(string: string) {
|
|
try {
|
|
new URL(string);
|
|
return true;
|
|
} catch (_) {
|
|
return false;
|
|
}
|
|
}
|