9router/src/shared/components/KiroAuthModal.js
2026-01-27 11:35:28 +07:00

325 lines
12 KiB
JavaScript

"use client";
import { useState } from "react";
import PropTypes from "prop-types";
import { Modal, Button, Input } from "@/shared/components";
import { useCopyToClipboard } from "@/shared/hooks/useCopyToClipboard";
/**
* Kiro Auth Method Selection Modal
* Allows user to choose between multiple Kiro authentication methods:
* 1. AWS Builder ID (Device Code)
* 2. AWS IAM Identity Center/IDC (Device Code with custom startUrl/region)
* 3. Google Social Login (Manual callback)
* 4. GitHub Social Login (Manual callback)
* 5. Import Token (Paste refresh token)
*/
export default function KiroAuthModal({ isOpen, onMethodSelect, onClose }) {
const [selectedMethod, setSelectedMethod] = useState(null);
const [idcStartUrl, setIdcStartUrl] = useState("");
const [idcRegion, setIdcRegion] = useState("us-east-1");
const [refreshToken, setRefreshToken] = useState("");
const [error, setError] = useState(null);
const [importing, setImporting] = useState(false);
const { copied, copy } = useCopyToClipboard();
const handleMethodSelect = (method) => {
setSelectedMethod(method);
setError(null);
};
const handleBack = () => {
setSelectedMethod(null);
setError(null);
};
const handleImportToken = async () => {
if (!refreshToken.trim()) {
setError("Please enter a refresh token");
return;
}
setImporting(true);
setError(null);
try {
const res = await fetch("/api/oauth/kiro/import", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ refreshToken: refreshToken.trim() }),
});
const data = await res.json();
if (!res.ok) {
throw new Error(data.error || "Import failed");
}
// Success - close modal
onClose();
} catch (err) {
setError(err.message);
} finally {
setImporting(false);
}
};
const handleIdcContinue = () => {
if (!idcStartUrl.trim()) {
setError("Please enter your IDC start URL");
return;
}
onMethodSelect("idc", { startUrl: idcStartUrl.trim(), region: idcRegion });
};
const handleSocialLogin = (provider) => {
onMethodSelect("social", { provider });
};
return (
<Modal isOpen={isOpen} title="Connect Kiro" onClose={onClose} size="lg">
<div className="flex flex-col gap-4">
{/* Method Selection */}
{!selectedMethod && (
<div className="space-y-3">
<p className="text-sm text-text-muted mb-4">
Choose your authentication method:
</p>
{/* AWS Builder ID */}
<button
onClick={() => onMethodSelect("builder-id")}
className="w-full p-4 text-left border border-border rounded-lg hover:bg-sidebar transition-colors"
>
<div className="flex items-start gap-3">
<span className="material-symbols-outlined text-primary mt-0.5">shield</span>
<div className="flex-1">
<h3 className="font-semibold mb-1">AWS Builder ID</h3>
<p className="text-sm text-text-muted">
Recommended for most users. Free AWS account required.
</p>
</div>
</div>
</button>
{/* AWS IAM Identity Center (IDC) - HIDDEN */}
<button
onClick={() => handleMethodSelect("idc")}
className="hidden w-full p-4 text-left border border-border rounded-lg hover:bg-sidebar transition-colors"
>
<div className="flex items-start gap-3">
<span className="material-symbols-outlined text-primary mt-0.5">business</span>
<div className="flex-1">
<h3 className="font-semibold mb-1">AWS IAM Identity Center</h3>
<p className="text-sm text-text-muted">
For enterprise users with custom AWS IAM Identity Center.
</p>
</div>
</div>
</button>
{/* Google Social Login - HIDDEN */}
<button
onClick={() => handleMethodSelect("social-google")}
className="hidden w-full p-4 text-left border border-border rounded-lg hover:bg-sidebar transition-colors"
>
<div className="flex items-start gap-3">
<span className="material-symbols-outlined text-primary mt-0.5">account_circle</span>
<div className="flex-1">
<h3 className="font-semibold mb-1">Google Account</h3>
<p className="text-sm text-text-muted">
Login with your Google account (manual callback).
</p>
</div>
</div>
</button>
{/* GitHub Social Login - HIDDEN */}
<button
onClick={() => handleMethodSelect("social-github")}
className="hidden w-full p-4 text-left border border-border rounded-lg hover:bg-sidebar transition-colors"
>
<div className="flex items-start gap-3">
<span className="material-symbols-outlined text-primary mt-0.5">code</span>
<div className="flex-1">
<h3 className="font-semibold mb-1">GitHub Account</h3>
<p className="text-sm text-text-muted">
Login with your GitHub account (manual callback).
</p>
</div>
</div>
</button>
{/* Import Token */}
<button
onClick={() => handleMethodSelect("import")}
className="w-full p-4 text-left border border-border rounded-lg hover:bg-sidebar transition-colors"
>
<div className="flex items-start gap-3">
<span className="material-symbols-outlined text-primary mt-0.5">file_upload</span>
<div className="flex-1">
<h3 className="font-semibold mb-1">Import Token</h3>
<p className="text-sm text-text-muted">
Paste refresh token from Kiro IDE.
</p>
</div>
</div>
</button>
</div>
)}
{/* IDC Configuration */}
{selectedMethod === "idc" && (
<div className="space-y-4">
<div>
<label className="block text-sm font-medium mb-2">
IDC Start URL <span className="text-red-500">*</span>
</label>
<Input
value={idcStartUrl}
onChange={(e) => setIdcStartUrl(e.target.value)}
placeholder="https://your-org.awsapps.com/start"
className="font-mono text-sm"
/>
<p className="text-xs text-text-muted mt-1">
Your organization's AWS IAM Identity Center URL
</p>
</div>
<div>
<label className="block text-sm font-medium mb-2">
AWS Region
</label>
<Input
value={idcRegion}
onChange={(e) => setIdcRegion(e.target.value)}
placeholder="us-east-1"
className="font-mono text-sm"
/>
<p className="text-xs text-text-muted mt-1">
AWS region for your Identity Center (default: us-east-1)
</p>
</div>
{error && (
<p className="text-sm text-red-600">{error}</p>
)}
<div className="flex gap-2">
<Button onClick={handleIdcContinue} fullWidth>
Continue
</Button>
<Button onClick={handleBack} variant="ghost" fullWidth>
Back
</Button>
</div>
</div>
)}
{/* Social Login Info (Google) */}
{selectedMethod === "social-google" && (
<div className="space-y-4">
<div className="bg-amber-50 dark:bg-amber-900/20 p-4 rounded-lg border border-amber-200 dark:border-amber-800">
<div className="flex gap-2">
<span className="material-symbols-outlined text-amber-600 dark:text-amber-400">info</span>
<div className="flex-1 text-sm">
<p className="font-medium text-amber-900 dark:text-amber-100 mb-1">
Manual Callback Required
</p>
<p className="text-amber-800 dark:text-amber-200">
After login, you'll need to copy the callback URL from your browser and paste it back here.
</p>
</div>
</div>
</div>
<div className="flex gap-2">
<Button onClick={() => handleSocialLogin("google")} fullWidth>
Continue with Google
</Button>
<Button onClick={handleBack} variant="ghost" fullWidth>
Back
</Button>
</div>
</div>
)}
{/* Social Login Info (GitHub) */}
{selectedMethod === "social-github" && (
<div className="space-y-4">
<div className="bg-amber-50 dark:bg-amber-900/20 p-4 rounded-lg border border-amber-200 dark:border-amber-800">
<div className="flex gap-2">
<span className="material-symbols-outlined text-amber-600 dark:text-amber-400">info</span>
<div className="flex-1 text-sm">
<p className="font-medium text-amber-900 dark:text-amber-100 mb-1">
Manual Callback Required
</p>
<p className="text-amber-800 dark:text-amber-200">
After login, you'll need to copy the callback URL from your browser and paste it back here.
</p>
</div>
</div>
</div>
<div className="flex gap-2">
<Button onClick={() => handleSocialLogin("github")} fullWidth>
Continue with GitHub
</Button>
<Button onClick={handleBack} variant="ghost" fullWidth>
Back
</Button>
</div>
</div>
)}
{/* Import Token */}
{selectedMethod === "import" && (
<div className="space-y-4">
<div className="bg-blue-50 dark:bg-blue-900/20 p-3 rounded-lg border border-blue-200 dark:border-blue-800 mb-4">
<p className="text-sm text-blue-800 dark:text-blue-200">
💡 Please login to Kiro IDE first.
</p>
</div>
<div>
<label className="block text-sm font-medium mb-2">
Refresh Token <span className="text-red-500">*</span>
</label>
<Input
value={refreshToken}
onChange={(e) => setRefreshToken(e.target.value)}
placeholder="aorAAAAAG..."
className="font-mono text-sm"
type="password"
/>
<p className="text-xs text-text-muted mt-1">
Find it in Kiro IDE at: <code className="bg-sidebar px-1 rounded">~/.aws/sso/cache/kiro-auth-token.json</code>
</p>
</div>
{error && (
<div className="bg-red-50 dark:bg-red-900/20 p-3 rounded-lg">
<p className="text-sm text-red-600 dark:text-red-400">{error}</p>
</div>
)}
<div className="flex gap-2">
<Button onClick={handleImportToken} fullWidth disabled={importing}>
{importing ? "Importing..." : "Import Token"}
</Button>
<Button onClick={handleBack} variant="ghost" fullWidth>
Back
</Button>
</div>
</div>
)}
</div>
</Modal>
);
}
KiroAuthModal.propTypes = {
isOpen: PropTypes.bool.isRequired,
onMethodSelect: PropTypes.func.isRequired,
onClose: PropTypes.func.isRequired,
};