mirror of
https://github.com/Significant-Gravitas/AutoGPT.git
synced 2026-01-08 22:58:01 -05:00
## Changes 🏗️ Standardize all the runtime environment checks on the Front-end and associated conditions to run against a single environment service where all the environment config is centralized and hence easier to manage. This helps prevent typos and bug when manually asserting against environment variables ( which are typed as `string` ), the helper functions are easier to read and re-use across the codebase. ## Checklist 📋 ### For code changes: - [x] I have clearly listed my changes in the PR description - [x] I have made a test plan - [x] I have tested my changes according to the test plan: - [x] Run the app and click around - [x] Everything is smooth - [x] Test on the CI and types are green ### For configuration changes: None 🙏🏽
195 lines
4.9 KiB
TypeScript
195 lines
4.9 KiB
TypeScript
"use client";
|
|
import { cn } from "@/lib/utils";
|
|
import { environment } from "@/services/environment";
|
|
import { useEffect, useRef, useState } from "react";
|
|
|
|
export interface TurnstileProps {
|
|
siteKey: string;
|
|
onVerify: (token: string) => void;
|
|
onExpire?: () => void;
|
|
onError?: (error: Error) => void;
|
|
action?: string;
|
|
className?: string;
|
|
id?: string;
|
|
shouldRender?: boolean;
|
|
setWidgetId?: (id: string | null) => void;
|
|
}
|
|
|
|
export function Turnstile({
|
|
siteKey,
|
|
onVerify,
|
|
onExpire,
|
|
onError,
|
|
action,
|
|
className,
|
|
id = "cf-turnstile",
|
|
shouldRender = true,
|
|
setWidgetId,
|
|
}: TurnstileProps) {
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
const widgetIdRef = useRef<string | null>(null);
|
|
const [loaded, setLoaded] = useState(false);
|
|
|
|
// Load the Turnstile script
|
|
useEffect(() => {
|
|
if (environment.isServerSide() || !shouldRender) return;
|
|
|
|
// Skip if already loaded
|
|
if (window.turnstile) {
|
|
setLoaded(true);
|
|
return;
|
|
}
|
|
|
|
const scriptSrc =
|
|
"https://challenges.cloudflare.com/turnstile/v0/api.js?render=explicit";
|
|
|
|
// If a script already exists, reuse it and attach listeners
|
|
const existingScript = Array.from(document.scripts).find(
|
|
(s) => s.src === scriptSrc,
|
|
);
|
|
|
|
if (existingScript) {
|
|
if (window.turnstile) {
|
|
setLoaded(true);
|
|
return;
|
|
}
|
|
|
|
const handleLoad: EventListener = () => {
|
|
setLoaded(true);
|
|
};
|
|
const handleError: EventListener = () => {
|
|
onError?.(new Error("Failed to load Turnstile script"));
|
|
};
|
|
|
|
existingScript.addEventListener("load", handleLoad);
|
|
existingScript.addEventListener("error", handleError);
|
|
|
|
return () => {
|
|
existingScript.removeEventListener("load", handleLoad);
|
|
existingScript.removeEventListener("error", handleError);
|
|
};
|
|
}
|
|
|
|
// Create a single script element if not present and keep it in the document
|
|
const script = document.createElement("script");
|
|
script.src = scriptSrc;
|
|
script.async = true;
|
|
script.defer = true;
|
|
|
|
const handleLoad: EventListener = () => {
|
|
setLoaded(true);
|
|
};
|
|
const handleError: EventListener = () => {
|
|
onError?.(new Error("Failed to load Turnstile script"));
|
|
};
|
|
|
|
script.addEventListener("load", handleLoad);
|
|
script.addEventListener("error", handleError);
|
|
|
|
document.head.appendChild(script);
|
|
|
|
return () => {
|
|
script.removeEventListener("load", handleLoad);
|
|
script.removeEventListener("error", handleError);
|
|
};
|
|
}, [onError, shouldRender]);
|
|
|
|
// Initialize and render the widget when script is loaded
|
|
useEffect(() => {
|
|
if (!loaded || !containerRef.current || !window.turnstile || !shouldRender)
|
|
return;
|
|
|
|
// Reset any existing widget
|
|
if (widgetIdRef.current && window.turnstile) {
|
|
try {
|
|
window.turnstile.reset(widgetIdRef.current);
|
|
} catch (err) {
|
|
console.warn("Failed to reset existing Turnstile widget:", err);
|
|
}
|
|
}
|
|
|
|
// Render a new widget
|
|
if (window.turnstile) {
|
|
widgetIdRef.current = window.turnstile.render(containerRef.current, {
|
|
sitekey: siteKey,
|
|
callback: (token: string) => {
|
|
onVerify(token);
|
|
},
|
|
"expired-callback": () => {
|
|
onExpire?.();
|
|
},
|
|
"error-callback": () => {
|
|
onError?.(new Error("Turnstile widget encountered an error"));
|
|
},
|
|
action,
|
|
});
|
|
|
|
// Notify the hook about the widget ID
|
|
setWidgetId?.(widgetIdRef.current);
|
|
}
|
|
|
|
return () => {
|
|
if (widgetIdRef.current && window.turnstile) {
|
|
try {
|
|
window.turnstile.remove(widgetIdRef.current);
|
|
} catch (err) {
|
|
console.warn("Failed to remove Turnstile widget:", err);
|
|
}
|
|
setWidgetId?.(null);
|
|
widgetIdRef.current = null;
|
|
}
|
|
};
|
|
}, [
|
|
loaded,
|
|
siteKey,
|
|
onVerify,
|
|
onExpire,
|
|
onError,
|
|
action,
|
|
shouldRender,
|
|
setWidgetId,
|
|
]);
|
|
|
|
// Method to reset the widget manually
|
|
useEffect(() => {
|
|
if (loaded && widgetIdRef.current && window.turnstile && shouldRender) {
|
|
window.turnstile.reset(widgetIdRef.current);
|
|
}
|
|
}, [loaded, shouldRender]);
|
|
|
|
// If shouldRender is false, don't render anything
|
|
if (!shouldRender) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<div
|
|
id={id}
|
|
ref={containerRef}
|
|
className={cn("my-4 flex items-center justify-center", className)}
|
|
/>
|
|
);
|
|
}
|
|
|
|
// Add TypeScript interface to Window to include turnstile property
|
|
declare global {
|
|
interface Window {
|
|
turnstile?: {
|
|
render: (
|
|
container: HTMLElement,
|
|
options: {
|
|
sitekey: string;
|
|
callback: (token: string) => void;
|
|
"expired-callback"?: () => void;
|
|
"error-callback"?: () => void;
|
|
action?: string;
|
|
},
|
|
) => string;
|
|
reset: (widgetId: string) => void;
|
|
remove: (widgetId: string) => void;
|
|
};
|
|
}
|
|
}
|
|
|
|
export default Turnstile;
|