Files
devsune/public/sw.js
2025-08-22 20:42:57 -07:00

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})
}