Mercurial Hosting > lang
changeset 6:025bb19b65b1
use claude
author | Franklin Schmidt <fschmidt@gmail.com> |
---|---|
date | Thu, 17 Jul 2025 23:19:22 -0600 |
parents | a970b7a01a74 |
children | 255c36830154 |
files | .hgignore src/chat.css src/chat.html.luan src/chat.js src/lib/Chat.luan src/lib/ai/Ai.luan src/lib/ai/claude/Chat.luan src/lib/ai/claude/Claude.luan src/private/Config_sample.luan |
diffstat | 9 files changed, 157 insertions(+), 6 deletions(-) [+] |
line wrap: on
line diff
--- a/.hgignore Wed Jul 16 15:08:14 2025 -0600 +++ b/.hgignore Thu Jul 17 23:19:22 2025 -0600 @@ -3,3 +3,4 @@ err local/ private/Config.luan +mine/
--- a/src/chat.css Wed Jul 16 15:08:14 2025 -0600 +++ b/src/chat.css Thu Jul 17 23:19:22 2025 -0600 @@ -10,7 +10,7 @@ [waiting-ai-icon] { width: 100px; - position: absolute; + position: fixed; top: 50%; left: 50%; transform: translate(-50%,-50%);
--- a/src/chat.html.luan Wed Jul 16 15:08:14 2025 -0600 +++ b/src/chat.html.luan Thu Jul 17 23:19:22 2025 -0600 @@ -39,8 +39,9 @@ @import "/chat.css?s=<%=started%>"; </style> <script> - let chatId = <%=chat_id%>; + let chatId = <%= chat.id %>; </script> + <script src="https://cdn.jsdelivr.net/npm/markdown-it@14.1.0/dist/markdown-it.min.js"></script> <script src="/chat.js?s=<%=started%>"></script> </head> <body> @@ -65,8 +66,8 @@ <textarea autofocus oninput="fixTextarea(event)" onkeydown="textareaKey(event)"></textarea> <button onclick="askAi()" title="Send"><img src="/images/send.svg"></button> </div> - <img waiting-ai-icon src="/images/spinner_green.gif"> </div> + <img waiting-ai-icon src="/images/spinner_green.gif"> <dialog rename> <h2>Rename Chat</h2> <form action="javascript:saveRenameChat()"> @@ -89,6 +90,9 @@ <button onclick="doDeleteChat(this)">Delete</button> </div> </dialog> + <script> + handleMarkdown(); + </script> </body> </html> <%
--- a/src/chat.js Wed Jul 16 15:08:14 2025 -0600 +++ b/src/chat.js Thu Jul 17 23:19:22 2025 -0600 @@ -35,11 +35,21 @@ document.querySelector('[waiting-ai-icon]').style.display = 'none'; } +// requires markdown-it +function handleMarkdown() { + let converter = window.markdownit(); + let divs = document.querySelectorAll('[role]'); + for( let div of divs ) { + div.innerHTML = converter.render(div.textContent); + } +} + function updateAi(html) { hideWaitingAiIcon(); document.querySelector('div[messages]').innerHTML = html; - //handleMarkdown(); + handleMarkdown(); document.querySelector('textarea').focus(); + window.scrollTo(0, document.body.scrollHeight); /* let scroll = aiDialog.querySelector('[scroll]'); setTimeout(function(){
--- a/src/lib/Chat.luan Wed Jul 16 15:08:14 2025 -0600 +++ b/src/lib/Chat.luan Thu Jul 17 23:19:22 2025 -0600 @@ -40,7 +40,7 @@ function Chat.new(chat) chat.updated = chat.updated or time_now() - chat.ai_name = chat.ai_name or "chatgpt" + chat.ai_name = chat.ai_name or "claude" chat.ai = Ai[chat.ai_name]["Chat.luan"] or error() function chat.save()
--- a/src/lib/ai/Ai.luan Wed Jul 16 15:08:14 2025 -0600 +++ b/src/lib/ai/Ai.luan Thu Jul 17 23:19:22 2025 -0600 @@ -5,7 +5,7 @@ local Ai = {} -local ais = {"chatgpt"} +local ais = {"claude"} local files = {"Chat.luan"} for _, ai in ipairs(ais) do local mods = {}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/lib/ai/claude/Chat.luan Thu Jul 17 23:19:22 2025 -0600 @@ -0,0 +1,98 @@ +local Luan = require "luan:Luan.luan" +local error = Luan.error +local ipairs = Luan.ipairs or error() +local type = Luan.type or error() +local String = require "luan:String.luan" +local starts_with = String.starts_with or error() +local Html = require "luan:Html.luan" +local html_encode = Html.encode or error() +local Parsers = require "luan:Parsers.luan" +local json_parse = Parsers.json_parse or error() +local json_string = Parsers.json_string or error() +local Claude = require "site:/lib/ai/claude/Claude.luan" +local claude_chat = Claude.chat or error() +local Logging = require "luan:logging/Logging.luan" +local logger = Logging.logger "claude/Chat" +local Thread = require "luan:Thread.luan" + + +local Chat = {} + +function Chat.output_messages_html(thread) + local messages = thread and json_parse(thread) or {nil} + for _, message in ipairs(messages) do + local role = message.role or error() + local who + if role=="assistant" then + who = "Claude" + elseif role=="user" then + who = "You" + else + error(role) + end + local function output(text) + if not starts_with( text, "[INTERNAL_UPDATE]" ) then +%> + <h3><%=who%></h3> + <div role="<%=role%>"><%=html_encode(text)%></div> +<% + end + end + local content = message.content or error() + if type(content) == "string" then + output(content) + else + for _, part in ipairs(content) do + if part.type=="text" then + local text = part.text or error() + output(text) + end + end + end + end_for +end + +function Chat.ask(thread,input) + local messages = thread and json_parse(thread) or {nil} + messages[#messages+1] = { + role = "user" + content = input + } +--[=[ + messages[#messages+1] = { + role = "assistant" + content = [[ +- 1 +- 2 +- 3 +- 4 +- 5 +- 6 +- 7 +- 8 +- 9 +- 10 +]] + Thread.sleep(5000) + } + if true then + return json_string(messages) + end +--]=] + local resultJson = claude_chat{ + messages = messages + } + local result = json_parse(resultJson) + -- logger.info(json_string(result)) + result.type == "message" or error() + result.role == "assistant" or error() + result.stop_reason == "end_turn" or result.stop_reason == "tool_use" or error() + local content = result.content or error() + messages[#messages+1] = { + role = "assistant" + content = content + } + return json_string(messages) +end + +return Chat
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/lib/ai/claude/Claude.luan Thu Jul 17 23:19:22 2025 -0600 @@ -0,0 +1,35 @@ +local Luan = require "luan:Luan.luan" +local error = Luan.error +local Io = require "luan:Io.luan" +local uri = Io.uri or error() +local Parsers = require "luan:Parsers.luan" +local json_string = Parsers.json_string or error() +local Config = require "site:/private/Config.luan" +local Logging = require "luan:logging/Logging.luan" +local logger = Logging.logger "claude/Claude" + + +local Claude = {} + +local url = "https://api.anthropic.com/v1/messages" +local headers = { + ["x-api-key"] = Config.claude.key or error() + ["anthropic-version"] = "2023-06-01" + ["Content-Type"] = "application/json" +} +local model = "claude-sonnet-4-0" +local max_tokens = 8192 + +function Claude.chat(content) + content.model = content.model or model + content.max_tokens = content.max_tokens or max_tokens + local options = { + method = "POST" + headers = headers + content = json_string(content) + } + local response = uri(url,options).read_text() + return response +end + +return Claude