Files
.sune/inline-commit.sune

1 line
12 KiB
Plaintext

[{"id":"jbcwown","name":"1 Click GitHub Commit","pinned":false,"avatar":"","url":"gh://multipleof4/.sune/inline-commit.sune","updatedAt":1769549049573,"settings":{"model":"google/gemini-2.5-pro","temperature":"","top_p":"","top_k":"","frequency_penalty":"","repetition_penalty":"","min_p":"","top_a":"","verbosity":"","reasoning_effort":"default","system_prompt":"You include this sune in the extension.html of any sune you use to code with.\nThen use this system prompt and you will get 1 click button to commit your code inside code blocks:\nIn the preceding line to code blocks, apply this format: \n<newline>\n<newline>\n[org/repo@branch/file](https://github.com/org/repo/blob/branch/file \"Fix/Feat/Refactor/etc: Summarize in less than 50chars your following edit\")\nYou can delete files like so:\n<newline>\n<newline>\n[org/repo@branch/file](https://github.com/org/repo/blob/branch/file \"Delete: Summarize in less than 50chars\")\n<newline>\n<newline>","html":"<!--\nSune: GitHub Commit & Delete Helper\nVersion: 3.1\n-->\n<div id=\"sune_github_helper\" x-data=\"{v:'3.1'}\" class=\"hidden\"></div>\n<script>\n(()=>{\nconst suneEl=document.getElementById('sune_github_helper');\nif(!suneEl)return;\nconst SUNE_NAME='[Sune: GitHub Helper]';\nconsole.log(`${SUNE_NAME} v3.1`);\n\nconst defC=['bg-slate-100','text-slate-700','hover:bg-slate-200'],\nokC=['bg-green-100','text-green-800'],errC=['bg-red-100','text-red-800'];\n\nconst safeJson=async r=>{const t=await r.text();try{return JSON.parse(t)}catch{return{message:t.slice(0,200)}}};\n\nconst deleteFile=async(btn,owner,repo,branch,path,commitMsg)=>{\nconst token=window.USER?.githubToken;\nif(!token){alert('GitHub token not set.');return}\nbtn.disabled=true;\nbtn.innerHTML='<span class=\"animate-pulse\">Checking...</span>';\nconst H={'Authorization':`Bearer ${token}`,'Accept':'application/vnd.github.v3+json'};\ntry{\nconst U=`https://api.github.com/repos/${owner}/${repo}/contents/${path}`;\nconst r1=await fetch(`${U}?ref=${branch}`,{headers:H});\nif(r1.status===404)throw new Error('File not found.');\nif(!r1.ok)throw new Error(`Get failed: ${(await safeJson(r1)).message}`);\nconst{sha}=await r1.json();\nif(!sha)throw new Error('No SHA.');\nbtn.innerHTML='<span class=\"animate-pulse\">Deleting...</span>';\nconst r2=await fetch(U,{method:'DELETE',headers:H,body:JSON.stringify({message:commitMsg?.trim()||`Delete ${path.split('/').pop()}`,sha,branch})});\nif(!r2.ok)throw new Error(`Delete failed: ${(await safeJson(r2)).message}`);\nbtn.classList.remove(...defC);btn.classList.add(...okC);\nbtn.innerHTML='<i data-lucide=\"check\" class=\"h-3.5 w-3.5\"></i> Deleted';\n}catch(e){\nalert(`${SUNE_NAME} Delete Failed:\\n\\n${e.message}`);\nbtn.classList.remove(...defC);btn.classList.add(...errC);\nbtn.innerHTML='<i data-lucide=\"x\" class=\"h-3.5 w-3.5\"></i> Failed';\nsetTimeout(()=>{btn.disabled=false;btn.classList.remove(...errC);btn.classList.add(...defC);btn.innerHTML='<i data-lucide=\"trash-2\" class=\"h-3.5 w-3.5\"></i> Delete';window.lucide?.createIcons()},4e3);\n}finally{window.lucide?.createIcons()}\n};\n\nconst commitFile=async(btn,owner,repo,branch,path,content,commitMsg)=>{\nconst token=window.USER?.githubToken;\nif(!token){alert('GitHub token not set.');return}\nbtn.disabled=true;\nbtn.innerHTML='<span class=\"animate-pulse\">Checking...</span>';\nconst H={'Authorization':`Bearer ${token}`,'Accept':'application/vnd.github.v3+json'};\ntry{\nconst r0=await fetch('https://api.github.com/user',{headers:H});\nif(!r0.ok)throw new Error('Token invalid.');\nconst uname=(await r0.json()).login;\nbtn.innerHTML='<span class=\"animate-pulse\">Committing...</span>';\nconst rRepo=await fetch(`https://api.github.com/repos/${owner}/${repo}`,{headers:H});\nif(rRepo.status===404){\nconst isOrg=owner.toLowerCase()!==uname.toLowerCase(),\nurl=isOrg?`https://api.github.com/orgs/${owner}/repos`:'https://api.github.com/user/repos',\nrc=await fetch(url,{method:'POST',headers:H,body:JSON.stringify({name:repo,private:false})});\nif(!rc.ok)throw new Error(`Create repo failed: ${(await safeJson(rc)).message}`);\n}else if(!rRepo.ok)throw new Error(`Repo check failed: ${(await safeJson(rRepo)).message}`);\nconst U=`https://api.github.com/repos/${owner}/${repo}/contents/${path}`;\nlet sha;\nconst rf=await fetch(`${U}?ref=${branch}`,{headers:H});\nif(rf.ok)sha=(await rf.json()).sha;\nelse if(rf.status!==404)throw new Error(`Get file failed: ${(await safeJson(rf)).message}`);\nconst msg=commitMsg?.trim()||(sha?`Update ${path.split('/').pop()}`:`Add ${path.split('/').pop()}`),\nb64=btoa(unescape(encodeURIComponent(content)));\nconst rp=await fetch(U,{method:'PUT',headers:H,body:JSON.stringify({message:msg,content:b64,sha,branch})});\nif(!rp.ok)throw new Error(`Commit failed: ${(await safeJson(rp)).message}`);\nbtn.classList.remove(...defC);btn.classList.add(...okC);\nbtn.innerHTML='<i data-lucide=\"check\" class=\"h-3.5 w-3.5\"></i> Success';\n}catch(e){\nalert(`${SUNE_NAME} Commit Failed:\\n\\n${e.message}`);\nbtn.classList.remove(...defC);btn.classList.add(...errC);\nbtn.innerHTML='<i data-lucide=\"x\" class=\"h-3.5 w-3.5\"></i> Failed';\nsetTimeout(()=>{btn.disabled=false;btn.classList.remove(...errC);btn.classList.add(...defC);btn.innerHTML='<i data-lucide=\"github\" class=\"h-3.5 w-3.5\"></i> Commit';window.lucide?.createIcons()},4e3);\n}finally{window.lucide?.createIcons()}\n};\n\nconst processBubble=b=>{\nif(!b||b.dataset.suneGchProcessed)return;\nlet added=false;\nb.querySelectorAll('p:not(:has(button.commit-btn))').forEach(p=>{\nconst a=p.querySelector('a');if(!a)return;\nconst m=a.textContent.trim().match(/^([^\\/]+)\\/([^\\/]+)@([^\\/]+)\\/(.+)$/);if(!m)return;\nconst[_,owner,repo,branch,path]=m,msg=a.title,isDel=(msg||'').toLowerCase().startsWith('delete:');\nconst btn=document.createElement('button');\nbtn.className='commit-btn ml-2 inline-flex items-center gap-1.5 rounded-md bg-slate-100 px-2 py-1 text-xs font-medium text-slate-700 hover:bg-slate-200 transition-colors';\nif(isDel){\nbtn.innerHTML='<i data-lucide=\"trash-2\" class=\"h-3.5 w-3.5\"></i> Delete';\nbtn.onclick=()=>deleteFile(btn,owner,repo,branch,path.trim(),msg);\np.append(btn);added=true;\n}else{\nconst pre=p.nextElementSibling;if(pre?.tagName!=='PRE')return;\nconst code=pre.querySelector('code')?.innerText;if(code==null)return;\nbtn.innerHTML='<i data-lucide=\"github\" class=\"h-3.5 w-3.5\"></i> Commit';\nbtn.onclick=()=>commitFile(btn,owner,repo,branch,path.trim(),code,msg);\np.append(btn);added=true;\n}\n});\nif(added){b.dataset.suneGchProcessed='true';window.lucide?.createIcons()}\nreturn added;\n};\n\nconst scan=()=>document.querySelectorAll('#messages .msg-bubble').forEach(processBubble);\nconst obs=new MutationObserver(ms=>{for(const m of ms)for(const n of m.addedNodes)if(n.nodeType===1){(n.matches?.('.msg-bubble')?[n]:n.querySelectorAll?.('.msg-bubble'))?.forEach(processBubble)}});\nconst onResp=e=>{const id=e?.detail?.message?.id;if(!id)return;let a=0;const t=()=>{a++;const b=window.getBubbleById(id);if(b&&processBubble(b))return;if(a<8)setTimeout(t,300)};setTimeout(t,150)};\nconst c=window.el?.composer,m=window.el?.messages;\nif(c&&m){obs.observe(m,{childList:true,subtree:true});c.addEventListener('sune:newSuneResponse',onResp);scan()}\nsuneEl.addEventListener('sune:unmount',()=>{obs.disconnect();c?.removeEventListener('sune:newSuneResponse',onResp)});\n})();\n</script>\n","extension_html":"<sune src='https://raw.githubusercontent.com/sune-org/store/refs/heads/main/sync.sune' private></sune>","hide_composer":true,"include_thoughts":false,"json_output":false,"img_output":false,"aspect_ratio":"1:1","image_size":"1K","ignore_master_prompt":false,"json_schema":""},"storage":{}}]