Files
store/agent.sune
2025-09-09 10:27:51 -07:00

1 line
19 KiB
JSON
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
[{"id":"cjnh9pp","name":"Agent","pinned":false,"avatar":"","url":"gh://sune-org/store/agent.sune","updatedAt":1757438872367,"settings":{"model":"or:z-ai/glm-4.5","temperature":"","top_p":".95","top_k":"","frequency_penalty":"","repetition_penalty":"","min_p":"","top_a":"","verbosity":"","reasoning_effort":"default","system_prompt":"You are a live agent. Depending on the task, you may return partial html (preferred) or JavaScript. If returning html realize that you are inserting into already existing html. Tailwind, lucide, dom-cash, alpinejs are available. When you return code in triple backticks, it runs immediately.\nAccessing individual API keys:\nwindow.USER.apiKeyGoogle\nwindow.USER.apiKeyCloudflare\nwindow.USER.githubToken\nYou can use this command to pass information to the next chat, for example when you need to know the return value of something, put it here and the next user chat will output it:\n```js\nwindow.USER.log(str)\n```\nTypically when you send your response, the chat is concluded. If required anywhere in your response put <3 as in the heart of the conversation needs to stay alive for another round. The <3 will ensure the conversation doesnt end.","html":"<div class=\"p-3 sm:p-4\">\n <div id=\"agent-panel-root\" class=\"mx-auto w-full max-w-3xl rounded-xl border border-gray-200 bg-white p-3 shadow-sm\">\n <div class=\"flex items-center justify-between gap-3\">\n <div class=\"flex items-center gap-2\">\n <i data-lucide=\"brain-circuit\" class=\"h-5 w-5 text-gray-600\"></i>\n <span class=\"text-sm font-medium text-gray-800\">Agent Control</span>\n </div>\n <span id=\"agent-panel-version\" class=\"text-xs text-gray-400\"></span>\n </div>\n\n <div class=\"mt-3 flex items-center justify-between rounded-lg bg-gray-100 p-2\">\n <label id=\"agent-panel-status\" class=\"cursor-pointer text-sm text-gray-700\"></label>\n <button type=\"button\" role=\"switch\" aria-checked=\"false\" id=\"agent-panel-toggle\" class=\"relative inline-flex h-6 w-11 flex-shrink-0 cursor-pointer rounded-full border-2 border-transparent bg-gray-300 transition-colors duration-200 ease-in-out focus:outline-none\">\n <span id=\"agent-panel-toggle-handle\" class=\"pointer-events-none inline-block h-5 w-5 transform translate-x-0 rounded-full bg-white shadow ring-0 transition duration-200 ease-in-out\"></span>\n </button>\n </div>\n\n <div class=\"mt-2 border-t border-gray-200 pt-2\">\n <button id=\"agent-panel-console-toggle\" class=\"flex w-full items-center justify-between text-left text-sm font-medium text-gray-600 hover:text-black\">\n <span>Console</span>\n <i id=\"agent-panel-console-chevron\" data-lucide=\"chevron-down\" class=\"h-4 w-4 transition-transform\"></i>\n </button>\n\n <div id=\"agent-panel-console-container\" class=\"mt-2 hidden\">\n <pre id=\"agent-panel-console\" class=\"max-h-56 overflow-y-auto rounded-md bg-gray-900 p-2 text-xs font-mono leading-relaxed text-white\"></pre>\n <div class=\"mt-2 grid grid-cols-3 gap-2\">\n <button id=\"agent-panel-clear-logs\" class=\"flex items-center justify-center gap-2 rounded-md bg-gray-100 px-3 py-1.5 text-xs text-gray-600 hover:bg-gray-200\">\n <i data-lucide=\"trash-2\" class=\"h-3 w-3\"></i> Clear Logs\n </button>\n <button id=\"agent-panel-clear-exec\" class=\"flex items-center justify-center gap-2 rounded-md bg-gray-100 px-3 py-1.5 text-xs text-gray-600 hover:bg-gray-200\">\n <i data-lucide=\"rotate-ccw\" class=\"h-3 w-3\"></i> Forget Executions\n </button>\n <button id=\"agent-panel-clear-canvas\" class=\"flex items-center justify-center gap-2 rounded-md bg-gray-100 px-3 py-1.5 text-xs text-gray-600 hover:bg-gray-200\">\n <i data-lucide=\"eraser\" class=\"h-3 w-3\"></i> Clear Canvas\n </button>\n </div>\n </div>\n </div>\n </div>\n\n <script>\n (function() {\n const AgentPanel = {\n // --- State ---\n isActive: false, isConsoleOpen: true, version: 'v0.6.2',\n logs: [], processed: new Set(), currentSuneId: '', idPoll: null,\n \n // --- Config ---\n ns: '', mountId: 'agent-runtime',\n storageKey: '', logStorageKey: '', processedKey: '',\n \n // --- DOM Cache ---\n $: {},\n \n // --- Methods ---\n init(suneId) {\n const rootEl = $(`#agent-panel-root`);\n rootEl.attr('id', `agent-panel-root-${suneId}`);\n this.cacheDom(suneId);\n \n this.updateKeysFor(suneId);\n this.restorePerSune();\n this.bindComposer();\n this.startSuneWatcher();\n this.bindUIEvents();\n \n this.updateUI();\n this.log('SYS', `Agent ready on ${this.currentSuneId}`);\n try{window.lucide?.createIcons?.()}catch(e){}\n },\n \n cacheDom(suneId) {\n const root = `#agent-panel-root-${suneId}`;\n this.$ = {\n root: $(root),\n version: $(root).find('#agent-panel-version'),\n status: $(root).find('#agent-panel-status'),\n toggle: $(root).find('#agent-panel-toggle'),\n toggleHandle: $(root).find('#agent-panel-toggle-handle'),\n consoleToggle: $(root).find('#agent-panel-console-toggle'),\n consoleChevron: $(root).find('#agent-panel-console-chevron'),\n consoleContainer: $(root).find('#agent-panel-console-container'),\n console: $(root).find('#agent-panel-console'),\n clearLogsBtn: $(root).find('#agent-panel-clear-logs'),\n clearExecBtn: $(root).find('#agent-panel-clear-exec'),\n clearCanvasBtn: $(root).find('#agent-panel-clear-canvas')\n };\n },\n\n bindUIEvents() {\n this.$.toggle.on('click', () => {\n this.isActive = !this.isActive;\n localStorage.setItem(this.storageKey, this.isActive);\n this.log('SYS', this.isActive ? 'Agent Activated' : 'Agent Deactivated');\n this.updateUI();\n });\n this.$.consoleToggle.on('click', () => {\n this.isConsoleOpen = !this.isConsoleOpen;\n this.updateUI();\n });\n this.$.clearLogsBtn.on('click', this.clearLogs.bind(this));\n this.$.clearExecBtn.on('click', this.clearExecHistory.bind(this));\n this.$.clearCanvasBtn.on('click', this.clearCanvas.bind(this));\n },\n \n updateUI() {\n this.$.version.text(this.version);\n this.$.status.text(this.isActive ? 'Status: Active' : 'Status: Inactive');\n this.$.toggle.attr('aria-checked', this.isActive);\n this.$.toggle.toggleClass('bg-black', this.isActive).toggleClass('bg-gray-300', !this.isActive);\n this.$.toggleHandle.toggleClass('translate-x-5', this.isActive).toggleClass('translate-x-0', !this.isActive);\n\n this.$.consoleContainer.toggleClass('hidden', !this.isConsoleOpen);\n this.$.consoleChevron.toggleClass('rotate-180', this.isConsoleOpen);\n \n this.$.console.text(this.logs.join('\\n') || 'Console is empty.');\n this.scrollConsole();\n },\n\n restorePerSune(){\n try{this.isActive=localStorage.getItem(this.storageKey)==='true'}catch{};\n try{this.logs=JSON.parse(localStorage.getItem(this.logStorageKey)||'[]')}catch{this.logs=[]};\n try{this.processed=new Set(JSON.parse(localStorage.getItem(this.processedKey)||'[]'))}catch{this.processed=new Set()};\n },\n\n updateKeysFor(id){\n this.currentSuneId=id; this.ns=`.sag-${this.currentSuneId}`;\n this.storageKey=`sune_agent_panel_active_${id}`; this.logStorageKey=`sune_agent_panel_logs_${id}`; this.processedKey=`sune_agent_processed_${id}`;\n },\n \n bindComposer(){\n const $c=$('#composer'); $c.off(`sune:newSuneResponse${this.ns}`).on(`sune:newSuneResponse${this.ns}`,e=>this.onSuneResponse(e));\n $c.off(`submit${this.ns}`).on(`submit${this.ns}`,()=>this.injectContext());\n $c.off(`sune:changed${this.ns}`).on(`sune:changed${this.ns}`,e=>this.onSuneChanged(e?.detail?.id||window.SUNE?.id));\n },\n\n startSuneWatcher(){\n try{clearInterval(this.idPoll)}catch{};\n this.idPoll=setInterval(()=>{const id=window.SUNE?.id; if(id&&id!==this.currentSuneId)this.onSuneChanged(id)},333);\n },\n\n onSuneChanged(newId){\n if(!newId||newId===this.currentSuneId)return;\n try{$('#composer').off(`.sag-${this.currentSuneId}`)}catch{};\n const prev=this.currentSuneId; this.updateKeysFor(newId); this.restorePerSune(); this.clearCanvas(); this.bindComposer();\n this.log('SYS',`Switched Sune ${prev||'-'} -> ${newId}`); this.updateUI();\n },\n \n saveLogs(){try{localStorage.setItem(this.logStorageKey,JSON.stringify(this.logs.slice(-200)))}catch{}},\n saveProcessed(){try{localStorage.setItem(this.processedKey,JSON.stringify([...this.processed].slice(-200)))}catch{}},\n markProcessed(k){this.processed.add(k);this.saveProcessed()},\n hasProcessed(k){return this.processed.has(k)},\n hash(s){let h=0;for(let i=0;i<s.length;i++)h=((h<<5)-h)+s.charCodeAt(i)|0;return'h'+(h>>>0).toString(16)},\n uid(){return Date.now().toString(36)+Math.random().toString(36).slice(2,6)},\n \n log(t, m){\n const ts=new Date().toLocaleTimeString(); this.logs.push(`[${ts} ${t}] ${m}`); if(this.logs.length>200)this.logs.shift();\n this.saveLogs(); this.updateUI();\n },\n\n scrollConsole(){\n const el=this.$.console[0]; if(el) el.scrollTop=el.scrollHeight\n },\n \n ensureMount(){\n const host=document.getElementById('suneHtml'); if(!host){this.log('WARN','#suneHtml not found');return null}\n host.classList.remove('hidden'); let mount=host.querySelector('#'+this.mountId);\n if(!mount){mount=document.createElement('div');mount.id=this.mountId;mount.className='agent-runtime space-y-2 p-2';host.appendChild(mount)} return mount;\n },\n\n injectHTMLBlock(html){\n const mount=this.ensureMount();if(!mount)return;const wrap=document.createElement('div');wrap.setAttribute('data-agent-block',this.uid());wrap.className='rounded-lg border border-gray-200 bg-white/50 p-2';wrap.innerHTML=html;mount.appendChild(wrap);\n [...wrap.querySelectorAll('script')].forEach(old=>{const s=document.createElement('script');[...old.attributes].forEach(a=>s.setAttribute(a.name,a.value));s.textContent=old.textContent||'';old.parentNode?.replaceChild(s,old)});\n try{window.lucide?.createIcons?.()}catch{}; this.log('HTML','Injected HTML block');\n },\n \n executeJSBlock(code){\n this.log('EXEC','Running JS block'); try{(new Function(code)).call(window);this.log('DONE','JS executed')}catch(err){console.error('[Sune Agent] JS Error:',err);this.log('ERROR',err?.message||'Exec fail');window.SUNE?.log?.('Agent Error: '+(err?.message||'Exec fail'))}\n },\n \n async autoAskNext(msgId){\n const key=`auto:${msgId||this.uid()}`;if(this.hasProcessed(key))return this.log('SKIP','Auto-send processed for '+(msgId||'-'));\n const setVal=v=>{try{$('#input').val(v);this.log('AUTO','Input set')}catch(e){this.log('ERROR','Set input fail: '+(e?.message||e))}};\n const sendNow=()=>{try{const f=$('#composer')[0],b=$('#sendBtn');if(b.attr('type')==='submit'){b.trigger('click')}else{f?.requestSubmit?f.requestSubmit():$c.trigger('submit')};this.log('AUTO','Sent')}catch(e){this.log('ERROR','Dispatch fail: '+(e?.message||e))}};\n const waitIdle=ms=>new Promise(r=>{let t=Date.now(),i=setInterval(()=>{if(!window.state?.busy||Date.now()-t>ms){clearInterval(i);r()}},50)});\n this.log('AUTO','Triggering: Whats next?'); setVal('Whats next?'); await waitIdle(4000);\n if(window.state?.busy){this.log('BLOCK','Still busy, retrying');await waitIdle(4000)}\n if(window.state?.busy){this.log('FAIL','Aborted: still busy');return}\n sendNow();this.markProcessed(key);this.log('AUTO','Dispatched');\n },\n \n onSuneResponse(e){\n if(!this.isActive)return this.log('SKIP','Inactive; ignoring response');\n const msg=e?.detail?.message;if(!msg||msg.role!=='assistant'||!Array.isArray(msg.content))return;\n const txt=(window.partsToText(msg.content)||'')+'\\n'+(window.partsToText(window.SUNE?.lastReply?.content||'')||'');\n let any=false,m;const reJS=/```(?:javascript|js)\\n([\\s\\S]*?)\\n```/gi,reHTML=/```(?:html|htm)\\n([\\s\\S]*?)\\n```/gi,msgKey=msg.id||this.hash(txt);\n if(!this.hasProcessed(msgKey)){while((m=reJS.exec(txt))!==null){if((m[1]||'').trim()){any=true;this.executeJSBlock((m[1]||'').trim())}}while((m=reHTML.exec(txt))!==null){if((m[1]||'').trim()){any=true;this.injectHTMLBlock((m[1]||'').trim())}}if(any){this.markProcessed(msgKey);this.log('PROC','Executed blocks for '+msgKey)}}\n if(/<3/.test(txt)){this.log('MATCH','Found <3, auto-sending');this.autoAskNext(msg.id||msgKey)}else this.log('MISS','No <3 in reply');\n },\n \n injectContext(){\n if(!this.isActive||!this.logs.length)return;const input=$('#input'),cur=String(input.val()||'');if(!cur.trim())return;input.val(`[AGENT LOGS]\\n${this.logs.join('\\n')}\\n[/AGENT LOGS]\\n\\n`+cur);\n this.log('INFO','Injected logs into context');\n },\n \n clearLogs(){this.logs=[];this.saveLogs();this.log('SYS','Console cleared');this.updateUI()},\n clearExecHistory(){this.processed.clear();this.saveProcessed();this.log('SYS','Execution history cleared')},\n clearCanvas(){const m=document.querySelector('#'+this.mountId);if(m){m.innerHTML='';this.log('SYS','Cleared injected HTML canvas')}}\n };\n\n // Ensure a unique agent per sune\n if (!window.SuneAgents) window.SuneAgents = {};\n const suneId = window.SUNE?.id;\n if (suneId && !window.SuneAgents[suneId]) {\n window.SuneAgents[suneId] = Object.create(AgentPanel);\n window.SuneAgents[suneId].init(suneId);\n }\n })();\n </script>\n</div>\n","extension_html":"<sune src='https://raw.githubusercontent.com/sune-org/store/refs/heads/main/sync.sune' private />","hide_composer":false},"storage":{}}]