https://calvinistparrot.com/api/parrot-chat
The Parrot Chat endpoint provides real-time conversational interactions by streaming responses. It handles creating chat sessions, processing user messages, maintaining context, and integrating multiple theological agents including a final review stage ("Calvin's Review"). This endpoint forms the backbone for Parrot Chat.
As with the Parrot QA API, the Parrot Chat endpoint supports multiple denominational modes to cater to various theological traditions. However, we will not compromise on the following essential doctrines:
For simplicity, I created this other endpoint that focuses on quick QA. Please check the Parrot QA API documentation for more information.
Send a JSON payload with these possible fields:
The API streams different event types as JSON objects:
{"type": "progress", "title": "Looking for articles", "content": "Searching for: predestination"}
{"type": "parrot", "content": "The doctrine of predestination..."}
{"type": "calvin", "content": "This explanation aligns with Reformed theology..."}
{"type": "gotQuestions", "content": "* [What is predestination?](https://www.gotquestions.org/predestination.html)"}
{"type": "done"}
POST /api/parrot-chat { "userId": "user123", "initialQuestion": "What is predestination?", "initialAnswer": "Predestination refers to...", "denomination": "reformed-baptist" }
POST /api/parrot-chat { "userId": "user123", "initialQuestion": "What is predestination?" }
Response:
{ "chatId": "chat123" }
Note: This endpoint only returns the chatId. The client should navigate to a new URL with this chatId (e.g.,
). When the chat page loads, it will automatically trigger the streaming process using the/main-chat/chat123
flag to process the initial question.isAutoTrigger
// In main-chat/page.tsx const handleStartNewChat = async () => { const response = await fetch('/api/parrot-chat', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ userId: "user123", initialQuestion: "What is predestination?" }), }); const { chatId } = await response.json(); router.push(`/main-chat/${chatId}`); // Navigate to chat page };
// In main-chat/[chatId]/page.tsx useEffect(() => { // When we detect only a user message with no response yet if (messages.length === 1 && messages[0].sender === "user" && !autoSentRef.current) { autoSentRef.current = true; // Auto-trigger the API call with the isAutoTrigger flag handleSendMessage({ message: messages[0].content, isAutoTrigger: true }); } }, [messages, handleSendMessage]); // Actual API call with isAutoTrigger flag const handleSendMessage = async ({ message, isAutoTrigger }) => { const response = await fetch("/api/parrot-chat", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ chatId: "chat123", // From URL params message: "What is predestination?", // Initial question isAutoTrigger: true // This tells the API not to store the message again }), }); // Handle streaming response... };
isAutoTrigger: true
, the API doesn't save the message again (it's already stored)POST /api/parrot-chat { "userId": "user123", "chatId": "chat123", "message": "How does it relate to free will?" }
GET /api/parrot-chat?chatId=chat123
Response:
{ "chat": { "id": "cm7p6rik1001emqp0slgxgu1j", "userId": "6754db6b00119ba9e0da", "conversationName": "Understanding Predestination", "denomination": "reformed-baptist", "createdAt": "2025-02-28T19:48:22.321Z", "modifiedAt": "2025-02-28T19:48:45.855Z" }, "messages": [ { "id": "cm7p6rimi001gmqp0liuisq27", "chatId": "cm7p6rik1001emqp0slgxgu1j", "sender": "user", "content": "What is predestination?", "timestamp": "2025-02-28T19:48:22.410Z" }, { "id": "cm7p6rqc0001imqp0h9gywt2x", "chatId": "cm7p6rik1001emqp0slgxgu1j", "sender": "gotQuestions", "content": "* [Providence and Predestination - Monergism](https://www.monergism.com/reformation-theology/blog/providence-and-predestination)\n* [What is predestination? - GotQuestions.org](https://www.gotquestions.org/predestination.html)\n* [Predestination and the Work of Jesus Considered | Monergism](https://www.monergism.com/predestination-and-work-jesus-considered)\n* [What is Predestination? - Monergism](https://www.monergism.com/what-predestination)\n* [What does the Bible say about predestination vs. free will?](https://www.gotquestions.org/predestination-vs-free-will.html)", "timestamp": "2025-02-28T19:48:32.401Z" }, { "id": "cm7p6rxq0001kmqp0kdxt3qvt", "chatId": "cm7p6rik1001emqp0slgxgu1j", "sender": "calvin", "content": "Your summary of predestination captures...", "timestamp": "2025-02-28T19:48:41.938Z" }, { "id": "cm7p6rzqs001mmqp0jrawhfi1", "chatId": "cm7p6rik1001emqp0slgxgu1j", "sender": "parrot", "content": "Predestination is...", "timestamp": "2025-02-28T19:48:44.596Z" } ] }
const handleStream = async (chatId: string, message: string) => { const response = await fetch('/api/parrot-chat', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ chatId, message }) }); const reader = response.body.getReader(); const decoder = new TextDecoder(); while (true) { const { value, done } = await reader.read(); if (done) break; const lines = decoder.decode(value).split('\n'); for (const line of lines) { if (!line.trim()) continue; const event = JSON.parse(line); switch (event.type) { case 'progress': updateProgress(event.title, event.content); break; case 'parrot': appendParrotMessage(event.content); break; case 'calvin': showCalvinReview(event.content); break; case 'gotQuestions': showReferences(event.content); break; case 'done': finishStream(); break; } } } };
For complete implementation examples, see:
The endpoint supports the following denomination:
Each mode tailors its responses according to distinct theological perspectives on secondary issues while sharing a common foundation on core doctrines.
For further questions or support, please reach out!
This is open source, so if you're interested in helping me development this, check out the GitHub repo.