mirror of
https://github.com/multipleof4/devsune.git
synced 2026-01-13 16:07:55 +00:00
112 lines
5.7 KiB
JavaScript
112 lines
5.7 KiB
JavaScript
self.addEventListener("install",()=>{})
|
|
self.addEventListener("activate",()=>{})
|
|
|
|
const DB="sw-streams",STORE="sessions"
|
|
let dbp=null
|
|
function db(){if(dbp)return dbp;dbp=new Promise((res,rej)=>{const r=indexedDB.open(DB,1);r.onupgradeneeded=()=>{r.result.createObjectStore(STORE,{keyPath:"sid"})};r.onsuccess=()=>res(r.result);r.onerror=()=>rej(r.error)});return dbp}
|
|
function idbGet(sid){return db().then(d=>new Promise((res,rej)=>{const t=d.transaction(STORE,"readonly").objectStore(STORE).get(sid);t.onsuccess=()=>res(t.result||null);t.onerror=()=>rej(t.error)}))}
|
|
function idbPut(rec){return db().then(d=>new Promise((res,rej)=>{const t=d.transaction(STORE,"readwrite").objectStore(STORE).put(rec);t.onsuccess=()=>res();t.onerror=()=>rej(t.error)}))}
|
|
function idbMerge(sid,patch){return idbGet(sid).then(cur=>{const rec=Object.assign({sid,text:"",done:false,updatedAt:Date.now()},cur||{},patch||{});rec.updatedAt=Date.now();return idbPut(rec)})}
|
|
function idbAppend(sid,add){return idbGet(sid).then(cur=>{const rec=Object.assign({sid,text:"",done:false,updatedAt:Date.now()},cur||{});rec.text+=add;rec.updatedAt=Date.now();return idbPut(rec)})}
|
|
function idbDelete(sid){return db().then(d=>new Promise((res,rej)=>{const t=d.transaction(STORE,"readwrite").objectStore(STORE).delete(sid);t.onsuccess=()=>res();t.onerror=()=>rej(t.error)}))}
|
|
function idbAll(){return db().then(d=>new Promise((res,rej)=>{const t=d.transaction(STORE,"readonly").objectStore(STORE).getAll();t.onsuccess=()=>res(t.result||[]);t.onerror=()=>rej(t.error)}))}
|
|
|
|
const ctrls=new Map()
|
|
|
|
self.addEventListener("fetch",event=>{
|
|
const u=new URL(event.request.url)
|
|
const isOR=u.origin==="https://openrouter.ai"&&u.pathname==="/api/v1/chat/completions"&&event.request.method==="POST"
|
|
if(isOR){event.respondWith(proxyOpenRouter(event));return}
|
|
if(u.pathname==="/swproxy/history"&&event.request.method==="GET"){event.respondWith(handleHistory(u));return}
|
|
if(u.pathname==="/swproxy/clear"&&event.request.method==="POST"){event.respondWith(handleClear(event.request));return}
|
|
if(u.pathname==="/swproxy/abort"&&event.request.method==="POST"){event.respondWith(handleAbort(event.request));return}
|
|
if(u.pathname==="/swproxy/sessions"&&event.request.method==="GET"){event.respondWith(handleSessions());return}
|
|
})
|
|
|
|
async function proxyOpenRouter(event){
|
|
const req=event.request
|
|
let bodyText=""
|
|
try{bodyText=await req.clone().text()}catch(_){}
|
|
const isStream=/\"stream\"\s*:\s*true/.test(bodyText)
|
|
const sid="s_"+Math.random().toString(36).slice(2)+Date.now().toString(36)
|
|
await idbMerge(sid,{sid,text:"",done:false})
|
|
console.log("[SW] intercept",sid,"stream=",isStream)
|
|
const init={method:"POST",headers:new Headers(req.headers),body:bodyText||req.body,mode:"cors",cache:"no-store",credentials:"omit",signal:req.signal}
|
|
try{
|
|
const upstream=await fetch("https://openrouter.ai/api/v1/chat/completions",init)
|
|
const hs=new Headers(upstream.headers)
|
|
hs.set("Cache-Control","no-store")
|
|
hs.set("X-SW-Proxied","1")
|
|
hs.set("X-SW-Stream-Sid",sid)
|
|
if(!isStream||!upstream.body){await idbMerge(sid,{done:true});console.log("[SW]",sid,"non-stream or no body");return new Response(upstream.body,{status:upstream.status,statusText:upstream.statusText,headers:hs})}
|
|
const [branchClient,branchStore]=upstream.body.tee()
|
|
pumpToStore(branchStore,sid)
|
|
return new Response(branchClient,{status:upstream.status,statusText:upstream.statusText,headers:hs})
|
|
}catch(e){
|
|
await idbMerge(sid,{done:true})
|
|
console.log("[SW]",sid,"error",String(e&&e.message||e))
|
|
return new Response("upstream_error",{status:502})
|
|
}
|
|
}
|
|
|
|
function pumpToStore(stream,sid){
|
|
const reader=stream.getReader()
|
|
const dec=new TextDecoder()
|
|
let buf="",acc=0,last=0
|
|
function parsePush(txt){
|
|
buf+=txt
|
|
let i
|
|
while((i=buf.indexOf("\n\n"))!==-1){
|
|
const chunk=buf.slice(0,i).trim()
|
|
buf=buf.slice(i+2)
|
|
if(!chunk)continue
|
|
if(chunk.startsWith("data:")){
|
|
const data=chunk.slice(5).trim()
|
|
if(data==="[DONE]"){idbMerge(sid,{done:true});ctrls.delete(sid);console.log("[SW]",sid,"done totalChars=",acc);continue}
|
|
try{
|
|
const j=JSON.parse(data)
|
|
const d=j&&j.choices&&j.choices[0]&&j.choices[0].delta&&j.choices[0].delta.content||""
|
|
if(d){acc+=d.length;idbAppend(sid,d);if(acc-last>=500){last=acc;console.log("[SW]",sid,"bufferedChars=",acc)}}
|
|
}catch(_){}
|
|
}
|
|
}
|
|
}
|
|
function step(){return reader.read().then(({value,done})=>{
|
|
if(done){idbMerge(sid,{done:true});ctrls.delete(sid);console.log("[SW]",sid,"eof totalChars=",acc);return}
|
|
parsePush(dec.decode(value,{stream:true}))
|
|
return step()
|
|
}).catch(()=>{idbMerge(sid,{done:true});ctrls.delete(sid);console.log("[SW]",sid,"pump error")})}
|
|
step()
|
|
}
|
|
|
|
async function handleHistory(u){
|
|
const sid=u.searchParams.get("sid")||""
|
|
const rec=await idbGet(sid)
|
|
const body=JSON.stringify({sid,text:rec?.text||"",done:!!rec?.done,updatedAt:rec?.updatedAt||0})
|
|
return new Response(body,{status:200,headers:{"Content-Type":"application/json","Cache-Control":"no-store"}})
|
|
}
|
|
|
|
async function handleSessions(){
|
|
const all=await idbAll()
|
|
const list=all.map(x=>({sid:x.sid,textLen:(x.text||"").length,done:!!x.done,updatedAt:x.updatedAt||0}))
|
|
return new Response(JSON.stringify(list),{status:200,headers:{"Content-Type":"application/json","Cache-Control":"no-store"}})
|
|
}
|
|
|
|
async function handleClear(req){
|
|
const {sid}=await req.json()
|
|
await idbDelete(sid)
|
|
ctrls.delete(sid)
|
|
console.log("[SW]",sid,"cleared")
|
|
return new Response("ok",{status:200})
|
|
}
|
|
|
|
async function handleAbort(req){
|
|
const {sid}=await req.json()
|
|
const c=ctrls.get(sid)
|
|
if(c)c.abort()
|
|
ctrls.delete(sid)
|
|
await idbMerge(sid,{done:true})
|
|
console.log("[SW]",sid,"aborted")
|
|
return new Response("ok",{status:200})
|
|
}
|