tinybase
Version:
A reactive data store and sync engine.
2 lines (1 loc) • 8.57 kB
JavaScript
const a=a=>typeof a,t="tinybase",e="",n=",",s=a(e),i="true",o=(a,t,e)=>a.replace(t,e),r=Promise,c=a=>(t,e,n)=>a(t)?n?.():e(t),l=clearInterval,w=a=>null==a,u=a=>void 0===a,y=a=>null===a,d=c(w),p=c(u),g=t=>a(t)==s,v=a=>Array.isArray(a),E=(a,t,e)=>a.slice(t,e),A=a=>a.length,m=()=>{},N=async a=>r.all(a),$=a=>{throw Error(a)},h=async(a,t,e)=>{try{return await a()}catch(a){t?.(a)}},f=(a,t)=>a.forEach(t),b=(a,t="")=>a.join(t),C=(a,t)=>a.map(t),T=a=>0==A(a),O=(a,t)=>a.filter(t),D=(a,...t)=>a.push(...t),I=a=>a.shift(),S=(a,t)=>a?.has(t)??!1,L=a=>u(a)||0==(a=>a?.size??0)(a),R=a=>[...a?.values()??[]],_=(a,t)=>a?.forEach(t),M=(a,t)=>a?.delete(t),P="_",F="_id",j="SELECT",k="WHERE",U="TABLE",x="ALTER "+U,B="FROM",J="DELETE "+B,Y=j+"*"+B,q="pragma_",z="data_version",G="schema_version",H="pragma_table_",V="CREATE "+U,K=a=>b(C(((a,t="",e)=>a.split(t,e))(a,"."),a=>`"${o(a,/"/g,'""')}"`),"."),Q=(a,t=[1])=>b(C(a,()=>"$"+t[0]++),n),W=(a,t=i)=>k+`(${o(t,"$tableName",K(a))})`,X=Object,Z=a=>X.getPrototypeOf(a),aa=X.entries,ta=a=>!w(a)&&d(Z(a),a=>a==X.prototype||w(Z(a)),()=>!0),ea=X.keys,na=X.freeze,sa=(a=[])=>X.fromEntries(a),ia=(...a)=>X.assign({},...a),oa=(a,t)=>(delete a[t],a),ra=(a,t)=>C(aa(a),([a,e])=>t(e,a)),ca=(a,t)=>sa(ra(a,(a,e)=>[e,t(a,e)])),la=a=>X.values(a),wa=a=>A(ea(a)),ua=a=>ta(a)&&0==wa(a),ya=Map,da=a=>new ya(a),pa=(a,t)=>a?.get(t),ga=(a,t)=>C([...a?.entries()??[]],([a,e])=>t(e,a)),va=(a,t,e)=>u(e)?(M(a,t),a):a?.set(t,e),Ea=(a,t,e,n)=>(S(a,t)?n?.(pa(a,t)):va(a,t,e()),pa(a,t)),Aa=(a,t,e,n,s=0)=>p((e?Ea:pa)(a,t[s],s>A(t)-2?e:da),i=>{if(s>A(t)-2)return n?.(i)&&va(a,t[s]),i;const o=Aa(i,t,e,n,s+1);return L(i)&&va(a,t[s]),o}),ma=a=>new Set(v(a)||u(a)?a:[a]),Na=(a,t)=>a?.add(t),$a="ColumnName",ha="store",fa="json",ba=ha+"TableName",Ca=ha+"Id"+$a,Ta=ha+$a,Oa="autoLoadIntervalSeconds",Da="rowId"+$a,Ia="tableId",Sa="tableName",La="deleteEmptyColumns",Ra="deleteEmptyTable",_a="condition",Ma={mode:fa,[Oa]:1},Pa={load:0,save:0,[Sa]:t+"_values"},Fa=(a,t,e,n,s)=>{const i=da();return ca(a,(a,o)=>{const r=E(la(ia(t,g(a)?{[e]:a}:a)),0,wa(t));y(r[0])||n(o,r[0])||(s(o,r[0]),va(i,o,r))}),i},ja=JSON.stringify,ka=JSON.parse,Ua=a=>""===a?void 0:v(a)?C(a,Ua):ta(a)?ca(a,Ua):a,xa=/^\d+$/,Ba=da(),Ja=da(),Ya=(a,t,n,s,i,o,r,c={},l=0,w=[])=>{let d,g,E,m=0,N=0,b=0;Ea(Ba,w,()=>0),Ea(Ja,w,()=>[]);const C=da(),[T,O,S,R,P]=((a=1,t,e)=>1!=a&&t.isMergeable()?[1,t.getMergeableContent,()=>t.getTransactionMergeableChanges(!e),([[a],[t]])=>!ua(a)||!ua(t),t.setDefaultContent]:2!=a?[0,t.getContent,t.getTransactionChanges,([a,t])=>!ua(a)||!ua(t),t.setContent]:$("Store type not supported by this Persister"))(r,a,l),[F,j,k]=(()=>{let a;const[t,n]=(()=>{const a=[];let t=0;return[n=>(n?I(a):null)??e+t++,t=>{xa.test(t)&&A(a)<1e3&&D(a,t)}]})(),s=da();return[(n,i,o,r=[],c=()=>[])=>{a??=W;const l=t(1);return va(s,l,[n,i,o,r,c]),Na(Aa(i,o??[e],ma),l),l},(t,n,...i)=>f(((a,t=[e])=>{const n=[],s=(a,e)=>e==A(t)?D(n,a):y(t[e])?_(a,a=>s(a,e+1)):f([t[e],null],t=>s(pa(a,t),e+1));return s(a,0),n})(t,n),t=>_(t,t=>pa(s,t)[0](a,...n??[],...i))),a=>p(pa(s,a),([,t,i])=>(Aa(t,i??[e],void 0,t=>(M(t,a),L(t)?1:0)),va(s,a),n(a),i)),t=>p(pa(s,t),([t,,e=[],n,s])=>{const i=(...o)=>{const r=A(o);r==A(e)?t(a,...o,...s(o)):y(e[r])?f(n[r]?.(...o)??[],a=>i(...o,a)):i(...o,e[r])};i()})]})(),U=a=>{a!=m&&(m=a,j(C,void 0,m))},x=t=>{(T&&v(t?.[0])?1===t?.[2]?a.applyMergeableChanges:a.setMergeableContent:1===t?.[2]?a.applyChanges:a.setContent)(t)},B=async()=>{V()&&a.hadMutated?.()&&await z()},J=async a=>(2!=m&&(U(1),N++,await Q(async()=>{await h(async()=>{const e=await t();v(e)?x(e):a?P(a):$("Content is not an array: "+e)},()=>{a&&P(a)}),U(0),await B()})),W),Y=async a=>(q(),await J(a),await h(async()=>g=await s(async(a,t)=>{t||a?2!=m&&(U(1),N++,x(t??a),U(0),await B()):await J()}),o),W),q=async()=>(g&&(await h(()=>i(g),o),g=void 0),W),z=async a=>(1!=m&&(U(2),b++,await Q(async()=>{await h(()=>n(O,a),o),U(0)})),W),G=async()=>(H(),await z(),E=a.addDidFinishTransactionListener(()=>{const a=S();R(a)&&z(a)}),W),H=async()=>(E&&(a.delListener(E),E=void 0),W),V=()=>!u(E),K=async(a=!1)=>{const[t,e]=a?[H,q]:[q,H];return await t(),await e(),W},Q=async(...a)=>(D(pa(Ja,w),...a),await(async()=>{if(!pa(Ba,w)){for(va(Ba,w,1);!u(d=I(pa(Ja,w)));)await h(d,o);va(Ba,w,0)}})(),W),W={load:J,startAutoLoad:Y,stopAutoLoad:q,isAutoLoading:()=>!u(g),save:z,startAutoSave:G,stopAutoSave:H,isAutoSaving:V,startAutoPersisting:async(a,t=!1)=>{const[e,n]=t?[G,Y]:[Y,G];return await e(a),await n(a),W},stopAutoPersisting:K,getStatus:()=>m,addStatusListener:a=>F(a,C),delListener:t=>(k(t),a),schedule:Q,getStore:()=>a,destroy:()=>(pa(Ja,w).splice(0,void 0),K()),getStats:()=>({loads:N,saves:b}),...c};return na(W)},qa=(a,t,e,s,o,r=za,c,l)=>{const w=da();return[async()=>{w.clear(),C(await e(a,t),({tn:a,cn:t})=>Na(Ea(w,a,ma),t))},async(t,e,n)=>((a,t)=>S(pa(w,a),t))(t,e)?sa(O(C(await a(Y+K(t)+W(t,n)),a=>[a[e],l?ca(oa(a,e),l):oa(a,e)]),([a,t])=>!u(a)&&!ua(t))):{},async(t,e,s,l,y,d=!1,p=i)=>{const g=ma();ca(s??{},a=>C(ea(a??{}),a=>Na(g,a)));const v=R(g);if(!d&&y&&p==i&&T(v)&&S(w,t))return await a("DROP "+U+K(t)),void va(w,t);const E=pa(w,t),A=ma(R(E));if(T(v)||(S(w,t)?await N(C([e,...v],async(n,s)=>{M(A,n)||(await a(x+K(t)+"ADD"+K(n)+o),0==s&&await a("CREATE UNIQUE INDEX pk ON "+K(t)+`(${K(e)})`),Na(E,n))})):(await a(V+K(t)+`(${K(e)}${o} PRIMARY KEY${b(C(v,a=>n+K(a)+o))});`),va(w,t,ma([e,...v])))),await N([...!d&&l?C(R(A),async n=>{n!=e&&(await a(x+K(t)+"DROP"+K(n)),M(E,n))}):[]]),d)u(s)?await a(J+K(t)+W(t,p)):await N(ra(s,async(n,s)=>{u(n)?await a(J+K(t)+W(t,p)+`AND(${K(e)}=$1)`,[s]):T(v)||await r(a,t,e,ea(n),{[s]:c?C(la(n),c):la(n)},E)}));else if(T(v))S(w,t)&&await a(J+K(t)+W(t,p));else{const n=O(R(pa(w,t)),a=>a!=e),i={},o=[];ca(s??{},(a,t)=>{i[t]=C(n,t=>c?c(a?.[t]):a?.[t]),D(o,t)}),await r(a,t,e,n,i),await a(J+K(t)+W(t,p)+`AND${K(e)}NOT IN(${Q(o)})`,o)}},async t=>{let e;return await a("BEGIN"),await h(async()=>e=await t(),s),await a("END"),e}]},za=async(a,t,e,s,i)=>{const o=[1];await a("INSERT INTO"+K(t)+"("+((...a)=>b(C(a,K),n))(e,...s)+")VALUES"+b(ra(i,a=>"($"+o[0]+++","+Q(a,o)+")"),n)+"ON CONFLICT("+K(e)+")DO UPDATE SET"+b(C(s,a=>K(a)+"=excluded."+K(a)),n),ra(i,(a,t)=>[t,...C(a,a=>a??null)]).flat())},Ga=(a,t,e,n,s,i,o,[r,c,l],w,y,d,p,g,v)=>{const[E,A,m,N]=qa(t,w,y,s,g,v),$=Ya(a,()=>N(async()=>{return await E(),a=(await A(r,c))[P]?.[l]??"null",Ua(ka(a));var a}),a=>N(async()=>{var t;await E(),await m(r,c,{[P]:{[l]:(t=a()??null,ja(t,(a,t)=>u(t)?"":t))}},!0,!0)}),e,n,s,o,{[p]:()=>d,destroy:async()=>(await $.stopAutoPersisting(),i(),$)},0,d);return $},Ha=(a,t,e,n,s,i,o,[r,c,[l,w,y]],d,p,g,v,E,A,m,$)=>{const[h,f,b,C]=qa(t,d,p,s,E,A,m,$),T=(a,t)=>N(ga(c,async([e,n,s,i,o],r)=>{t&&!(r in a)||await b(e,n,a[r],s,i,t,o)})),D=async(a,t)=>w?await b(y,F,{[P]:a},!0,!0,t):null,I=Ya(a,()=>C(async()=>{await h();const a=await(async()=>sa(O(await N(ga(r,async([a,t,e],n)=>[a,await f(n,t,e)])),a=>!ua(a[1]))))(),t=await(async()=>l?(await f(y,F))[P]:{})();return ua(a)&&u(t)?void 0:[a,t]}),(a,t)=>C(async()=>{if(await h(),u(t)){const[t,e]=a();await T(t),await D(e)}else await T(t[0],!0),await D(t[1],!0)}),e,n,s,o,{[v]:()=>g,destroy:async()=>(await I.stopAutoPersisting(),i(),I)},0,g);return I},Va=(a,n,s,o,r,c,u,y,d,p,v="getDb",A)=>{let m,N,$;const f=((a,t)=>t?async(e,n)=>(t(e,n),await a(e,n)):a)(s,c),[b,C,T,O]=(a=>{const e=(a=>ia(Ma,g(a)?{[ba]:a}:a??{}))(a),n=e[Oa];if(e.mode==fa){const a=e[ba]??t;return[1,n,[a,e[Ca]??F,e[Ta]??ha],ma(a)]}const{tables:{load:s={},save:o={}}={},values:r={}}=e,c=E(la(ia(Pa,r)),0,wa(Pa)),l=c[2],w=ma(l),u=ma(l),y=Fa(s,{[Ia]:null,[Da]:F,[_a]:i},Ia,a=>S(u,a),a=>Na(w,a)),d=Fa(o,{[Sa]:null,[Da]:F,[La]:0,[Ra]:0,[_a]:null},Sa,(a,t)=>S(u,t),(a,t)=>Na(w,t));var p;return p=(a,t)=>t[4]??=pa(y,t[0])?.[2]??i,_(d,(a,t)=>p(0,a)),[0,n,[y,d,c],w]})(n);return(b?Ga:Ha)(a,f,a=>{let t;const e=()=>t=setInterval(()=>h(async()=>{const[{d:t,s:e,c:n}]=await f(j+` ${z} d,${G} s,TOTAL_CHANGES() c FROM ${q}${z} JOIN ${q}${G}`);t==m&&e==N&&n==$||(w(m)||a(),m=t,N=e,$=n)}),1e3*C),n=()=>{m=N=$=null,l(t)},s=o(t=>{O.has(t)&&(n(),a(),e())});return e(),()=>{n(),r(s)}},a=>a(),u,y,d,T,R(O),async(a,t)=>await a(j+` t.name tn,c.name cn FROM ${H}list()t,${H}info(t.name)c ${k} t.schema='main'AND t.type IN('table','view')AND t.name IN(${Q(t)})ORDER BY t.name,c.name`,t),p,v,e,A,a=>(a=>!0===a)(a)?1:(a=>!1===a)(a)?0:a,void 0)},Ka=(a,t,e,n,s,i)=>Va(a,n,async(a,t=[])=>e.exec(a,{bind:t,rowMode:"object",returnValue:"resultRows"}).map(a=>({...a})),a=>t.capi.sqlite3_update_hook(e,(t,e,n,s)=>a(s),0),()=>t.capi.sqlite3_update_hook(e,m,0),s,i,m,3,e);export{Ka as createSqliteWasmPersister};