mirror of
https://github.com/sune-org/store.git
synced 2026-01-13 16:17:58 +00:00
1 line
17 KiB
JSON
1 line
17 KiB
JSON
[{"id":"cjnh9pp","name":"Agent","pinned":false,"avatar":"","url":"gh://sune-org/store/agent.sune","updatedAt":1757437685690,"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\n x-data=\"{\n // UI\n isActive:false,isConsoleOpen:true,version:'v0.6.2',\n\n // Logs + keys\n logs:[],storageKey:`sune_agent_panel_active_${window.SUNE.id}`,logStorageKey:`sune_agent_panel_logs_${window.SUNE.id}`,\n\n // De-dupe\n ns:'',processed:new Set(),processedKey:`sune_agent_processed_${window.SUNE.id}`,\n\n // Mount info\n mountId:'agent-runtime',\n\n // Sune switch handling\n currentSuneId:window.SUNE?.id||'',idPoll:null,\n\n // -------- Init --------\n init(){\n this.updateKeysFor(this.currentSuneId); this.restorePerSune(); this.bindComposer(); this.startSuneWatcher();\n $watch('isActive',v=>localStorage.setItem(this.storageKey,v)); $watch('logs',()=>this.saveLogs());\n this.$nextTick(()=>{this.scrollConsole(); try{window.lucide?.createIcons?.()}catch{}}); this.log('SYS',`Agent ready on ${this.currentSuneId}`);\n },\n\n // -------- Per-sune state --------\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 updateKeysFor(id){\n this.currentSuneId=id||window.SUNE?.id||''; this.ns=`.sag-${this.currentSuneId}`;\n this.storageKey=`sune_agent_panel_active_${this.currentSuneId}`; \n this.logStorageKey=`sune_agent_panel_logs_${this.currentSuneId}`;\n this.processedKey=`sune_agent_processed_${this.currentSuneId}`;\n },\n\n // -------- Composer binding --------\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}`);\n },\n\n // -------- Utils --------\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 log(t,m){ const ts=new Date().toLocaleTimeString(); this.logs.push(`[${ts} ${t}] ${m}`); if(this.logs.length>200)this.logs.shift(); this.$nextTick(()=>this.scrollConsole()) },\n scrollConsole(){ const el=this.$refs.console; if(el) el.scrollTop=el.scrollHeight },\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) }\n return mount;\n },\n\n // -------- HTML / JS exec --------\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.Alpine?.initTree?.(wrap) }catch(e){ this.log('WARN','Alpine init failed: '+(e?.message||'unknown')) }\n try{ window.lucide?.createIcons?.() }catch{}\n 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 Execution Error:',err); this.log('ERROR',err?.message||'Execution failed'); window.SUNE?.log?.('Agent Error: '+(err?.message||'Execution failed')) }\n },\n\n // -------- Auto Next: core --------\n async autoAskNext(msgId){\n const key=`auto:${msgId||this.uid()}`; if(this.hasProcessed(key)) return this.log('SKIP','Auto-send already processed for '+(msgId||'-'));\n const setVal=v=>{ try{$('#input').val(v); this.log('AUTO','Input set')}catch(e){ this.log('ERROR','Set input failed: '+(e?.message||e)) } };\n const sendNow=()=>{ try{\n const form=$('#composer')[0], btn=document.getElementById('sendBtn');\n if(btn && btn.getAttribute('type')==='submit'){ btn.click(); this.log('AUTO','Clicked #sendBtn (DOM)') }\n else if($('#sendBtn').length){ $('#sendBtn').trigger('click'); this.log('AUTO','Triggered #sendBtn click') }\n else if(form?.requestSubmit){ form.requestSubmit(); this.log('AUTO','Form.requestSubmit()') }\n else { $('#composer').trigger('submit'); this.log('AUTO','Triggered form submit') }\n window.haptic?.();\n }catch(e){ this.log('ERROR','Dispatch failed: '+(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 auto-send: What’s next?'); setVal('What’s 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 after wait'); return }\n sendNow(); this.markProcessed(key); this.log('AUTO','Auto-send dispatched');\n },\n\n // -------- Event handling --------\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 // Fenced blocks\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 // Auto next if '<3' anywhere in reply text\n if(/<3/.test(txt)){ this.log('MATCH','Found <3 in reply; scheduling auto-send'); this.autoAskNext(msg.id||msgKey) } else this.log('MISS','No <3 in reply');\n },\n\n // -------- Actions --------\n injectContext(){\n if(!this.isActive||!this.logs.length) return;\n 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 clearLogs(){ this.logs=[]; this.saveLogs(); this.log('SYS','Console cleared') },\n clearExecHistory(){ this.processed.clear(); this.saveProcessed(); this.log('SYS','Execution history cleared') },\n clearCanvas(){ const mount=document.querySelector('#'+this.mountId); if(mount){ mount.innerHTML=''; this.log('SYS','Cleared injected HTML canvas') } }\n }\"\n x-init=\"init()\"\n class=\"mx-auto w-full max-w-3xl rounded-xl border border-gray-200 bg-white p-3 shadow-sm\"\n >\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 class=\"text-xs text-gray-400\" x-text=\"version\"></span>\n </div>\n\n <div class=\"mt-3 flex items-center justify-between rounded-lg bg-gray-100 p-2\">\n <label class=\"cursor-pointer text-sm text-gray-700\" x-text=\"isActive ? 'Status: Active' : 'Status: Inactive'\"></label>\n <button\n type=\"button\" role=\"switch\" :aria-checked=\"isActive\"\n @click=\"isActive=!isActive; log('SYS', isActive ? 'Agent Activated' : 'Agent Deactivated')\"\n :class=\"isActive ? 'bg-black' : 'bg-gray-300'\"\n class=\"relative inline-flex h-6 w-11 flex-shrink-0 cursor-pointer rounded-full border-2 border-transparent transition-colors duration-200 ease-in-out focus:outline-none\"\n >\n <span :class=\"isActive ? 'translate-x-5' : 'translate-x-0'\" class=\"pointer-events-none inline-block h-5 w-5 transform 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 @click=\"isConsoleOpen=!isConsoleOpen\" 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 data-lucide=\"chevron-down\" class=\"h-4 w-4 transition-transform\" :class=\"isConsoleOpen && 'rotate-180'\"></i>\n </button>\n\n <div x-show=\"isConsoleOpen\" x-transition class=\"mt-2\">\n <pre x-ref=\"console\" class=\"max-h-56 overflow-y-auto rounded-md bg-gray-900 p-2 text-xs font-mono leading-relaxed text-white\" x-text=\"logs.join('\\n') || 'Console is empty.'\"></pre>\n <div class=\"mt-2 grid grid-cols-3 gap-2\">\n <button @click=\"clearLogs()\" 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 @click=\"clearExecHistory()\" 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 @click=\"clearCanvas()\" 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</div>\n","extension_html":"<sune src='https://raw.githubusercontent.com/sune-org/store/refs/heads/main/sync.sune' private />","hide_composer":false},"storage":{}}] |