@lpdjs/firestore-repo-service 2.2.5 → 2.2.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1,4 +1,4 @@
1
- import {Timestamp}from'firebase-admin/firestore';import {z}from'zod';import {jsx,jsxs,Fragment}from'hono/jsx/jsx-runtime';import {renderToString}from'hono/jsx/dom/server';function kn(e,t){let n=[];for(let r=0;r<e.length;r+=t)n.push(e.slice(r,r+t));return n}function we(e,t){return t.orderBy&&t.orderBy.forEach(n=>{e=e.orderBy(String(n.field),n.direction??"asc");}),t.limit&&(e=e.limit(t.limit)),t.offset&&(e=e.offset(t.offset)),t.select&&t.select.length>0&&(e=e.select(...t.select.map(n=>String(n)))),t.startAt&&(e=Array.isArray(t.startAt)?e.startAt(...t.startAt):e.startAt(t.startAt)),t.startAfter&&(e=Array.isArray(t.startAfter)?e.startAfter(...t.startAfter):e.startAfter(t.startAfter)),t.endAt&&(e=Array.isArray(t.endAt)?e.endAt(...t.endAt):e.endAt(t.endAt)),t.endBefore&&(e=Array.isArray(t.endBefore)?e.endBefore(...t.endBefore):e.endBefore(t.endBefore)),e}function qe(e){let[,t,n]=e;return (t==="in"||t==="array-contains-any")&&Array.isArray(n)&&n.length>30}function wt(e){let[t,n,r]=e;return qe(e)?kn(r,30).map(s=>[t,n,s]):[e]}function Ie(e,t){let n=e;for(let[r,o,s]of t)n=n.where(String(r),o,s);return n}async function St(e){let t=await Promise.all(e.map(o=>o.get())),n=new Map;t.forEach(o=>{o.docs.forEach(s=>{n.has(s.id)||n.set(s.id,s);});});let r=t[0];if(!r)throw new Error("No snapshots returned");return {...r,docs:Array.from(n.values()),size:n.size,empty:n.size===0}}async function ue(e,t){let n=t.orWhere&&t.orWhere.length>0,r=t.orWhereGroups&&t.orWhereGroups.length>0;if(!n&&!r){if(!t.where||t.where.length===0)return we(e,t).get();if(!t.where.some(qe)){let h=Ie(e,t.where);return h=we(h,t),h.get()}let i=t.where.map(wt),y=Qe(i).map(h=>{let c=Ie(e,h);return c=we(c,t),c});return St(y)}let o=t.where??[],s=[...t.orWhere?.map(u=>[u])??[],...t.orWhereGroups??[]],a=[];for(let u of s){let i=[...o,...u];if(i.some(qe)){let y=i.map(wt),c=Qe(y).map(d=>{let p=Ie(e,d);return p=we(p,t),p});a.push(...c);}else {let y=Ie(e,i);y=we(y,t),a.push(y);}}return St(a)}function Qe(e){if(e.length===0)return [[]];let t=e[0];if(e.length===1&&t)return t.map(s=>[s]);if(!t)return [[]];let n=e.slice(1),r=Qe(n),o=[];for(let s of t)for(let a of r)o.push([s,...a]);return o}var Ue="preserve";function An(e){Ue=e;}function Le(){return Ue}function Pn(e){return typeof e=="object"&&e!==null&&typeof e._seconds=="number"&&typeof e._nanoseconds=="number"}function Ge(e){if(e==null)return null;if(e instanceof Date)return Number.isNaN(e.getTime())?null:e;if(e instanceof Timestamp)return e.toDate();if(Pn(e))return new Date(e._seconds*1e3+Math.floor(e._nanoseconds/1e6));if(typeof e=="string"){let t=new Date(e);return Number.isNaN(t.getTime())?null:t}if(typeof e=="number"){let t=new Date(e);return Number.isNaN(t.getTime())?null:t}return null}function En(e){if(typeof e!="object"||e===null)return false;let t=Object.getPrototypeOf(e);return t===Object.prototype||t===null}function Ne(e){if(e instanceof Timestamp)return e.toDate();if(Array.isArray(e))return e.map(t=>Ne(t));if(En(e)){let t={};for(let[n,r]of Object.entries(e))t[n]=Ne(r);return t}return e}function F(e){return Ue==="normalize"?Ne(e):e}function Ct(e,t){let n=[];for(let r=0;r<e.length;r+=t)n.push(e.slice(r,r+t));return n}function ee(e){return e.charAt(0).toUpperCase()+e.slice(1)}function ne(e,t){return t.where&&t.where.forEach(([n,r,o])=>{e=e.where(String(n),r,o);}),t.orderBy&&t.orderBy.forEach(n=>{e=e.orderBy(String(n.field),n.direction??"asc");}),t.limit&&(e=e.limit(t.limit)),t.offset&&(e=e.offset(t.offset)),t.select&&t.select.length>0&&(e=e.select(...t.select.map(n=>String(n)))),t.startAt&&(e=Array.isArray(t.startAt)?e.startAt(...t.startAt):e.startAt(t.startAt)),t.startAfter&&(e=Array.isArray(t.startAfter)?e.startAfter(...t.startAfter):e.startAfter(t.startAfter)),t.endAt&&(e=Array.isArray(t.endAt)?e.endAt(...t.endAt):e.endAt(t.endAt)),t.endBefore&&(e=Array.isArray(t.endBefore)?e.endBefore(...t.endBefore):e.endBefore(t.endBefore)),e}async function _e(e,t){let n={...t,limit:t.pageSize+1};t.cursor&&(t.direction==="prev"?n.endBefore=t.cursor:n.startAfter=t.cursor);let o=(await ue(e,n)).docs,s=o.length>t.pageSize,a=s?o.slice(0,t.pageSize):o,u=a.map(l=>F({...l.data(),docId:l.id})),i=t.direction==="prev";return {data:u,nextCursor:i?a.length>0?a[a.length-1]:void 0:s?a[a.length-1]:void 0,prevCursor:i?s?a[0]:void 0:a[0],hasNextPage:i?!!t.cursor:s,hasPrevPage:i?s:!!t.cursor,pageSize:u.length}}async function*He(e,t){let n,r=true;for(;r;){let o=await _e(e,{...t,cursor:n,direction:"next"});yield o,r=o.hasNextPage,n=o.nextCursor;}}function $t(e){return {count:async(t={})=>{let n=e;return n=ne(n,t),(await n.count().get()).data().count},sum:async(t,n={})=>{let r=e;r=ne(r,n);let o=await r.get(),s=0;return o.forEach(a=>{let u=a.data()[t];typeof u=="number"&&(s+=u);}),s},average:async(t,n={})=>{let r=e;r=ne(r,n);let o=await r.get();if(o.empty)return null;let s=0,a=0;return o.forEach(u=>{let i=u.data()[t];typeof i=="number"&&(s+=i,a++);}),a>0?s/a:null}}}function Ot(e,t,n,r,o,s){let a=()=>new Date;return {create:()=>{let u=e.batch();return {batch:u,set:(...i)=>{let l=i[i.length-1],y=typeof l=="object"&&l!==null&&"merge"in l,h=y?i[i.length-2]:i[i.length-1],c=y?i.slice(0,-2):i.slice(0,-1),d=y?l:{merge:true},p=t(...c),m={...h},f=c[c.length-1];n&&f&&(m[n]=f),r&&(m[r]=p.path),o&&(m[o]=a()),s&&(m[s]=a()),u.set(p,m,d);},update:(...i)=>{let l=i.pop(),h=t(...i),c={...l};s&&(c[s]=a()),u.update(h,c);},delete:(...i)=>{let l=t(...i);u.delete(l);},commit:async()=>{await u.commit();}}}}}function kt(e,t,n){let r=()=>new Date;return {set:async o=>{let s=e.bulkWriter(),a=0;for(let u of o){if(!u)continue;let{docRef:i,data:l,merge:y=true}=u,h={...l};t&&(h[t]=r()),n&&(h[n]=r()),s.set(i,h,{merge:y}),a++,a>=500&&(await s.flush(),a=0);}await s.close();},update:async o=>{let s=e.bulkWriter(),a=0;for(let u of o){if(!u)continue;let{docRef:i,data:l}=u,y={...l};n&&(y[n]=r()),s.update(i,y),a++,a>=500&&(await s.flush(),a=0);}await s.close();},delete:async o=>{let s=e.bulkWriter(),a=0;for(let u of o)u&&(s.delete(u),a++,a>=500&&(await s.flush(),a=0));await s.close();}}}function At(e,t,n,r,o,s){let a=()=>new Date;return {create:async h=>{if(!e)throw new Error("Cannot use create() on collection groups. Use set() with a specific document ID instead.");let c,d,p={...h};if(o&&(p[o]=a()),s&&(p[s]=a()),h[n]){d=h[n],c=e.doc(d);let f=r?{...p,[r]:c.path}:p;await c.set(f);}else {c=await e.add(p),d=c.id;let f={[n]:d};r&&(f[r]=c.path),await c.update(f);}let m=await c.get();return F(m.data())},set:async(...h)=>{let c=h[h.length-1],d=typeof c=="object"&&c!==null&&"merge"in c,p=d?h[h.length-2]:h[h.length-1],m=d?h.slice(0,-2):h.slice(0,-1),f=d?c:{merge:true},v={...p};s&&(v[s]=a());let g=t(...m),b=m[m.length-1];n&&b!=null&&(v[n]=b),r&&(v[r]=g.path),await g.set(v,f);let R=await g.get();return F(R.data())},update:async(...h)=>{let c=h.pop(),d=h,p={...c};s&&(p[s]=a());let m=t(...d);await m.update(p);let f=await m.get();return F(f.data())},delete:async(...h)=>{await t(...h).delete();}}}function Pt(e,t,n,r,o){let s={};return s.byList=async(a,u,i="in",l={})=>{if(u.length===0)return [];let y=[],h=Ct(u,30);for(let c of h){let d=e;d=d.where(a,i,c),l.select&&l.select.length>0&&(d=d.select(...l.select.map(m=>String(m)))),(await d.get()).forEach(m=>{let f=F(m.data());y.push(l.returnDoc?{data:f,doc:m}:f);});}return y},t.forEach(a=>{let u=`by${ee(String(a))}`;s[u]=async(i,l={})=>{let y=typeof l=="boolean"?{returnDoc:l}:l;if(String(a)===o){let f=await r(i).get();if(!f.exists)return null;let v=F(f.data());return y.returnDoc?{data:v,doc:f}:v}let h=e;h=h.where(String(a),"==",i).limit(1),y.select&&y.select.length>0&&(h=h.select(...y.select.map(m=>String(m))));let c=await h.get();if(c.empty)return null;let d=c.docs[0];if(!d)return null;let p=F(d.data());return y.returnDoc?{data:p,doc:d}:p};}),s}function Et(e,t,n,r){let o={},s=async(a,u)=>{if(!n||!r||u.length===0)return a;let i=u.map(l=>typeof l=="string"?{key:l}:{key:l.relation,select:l.select});return Promise.all(a.map(async l=>{let y=await Promise.all(i.map(async({key:c,select:d})=>{let p=n[c];if(!p)return [c,void 0];let m=r[p.repo];if(!m)return [c,void 0];let f=l[c];if(f==null)return [c,p.type==="one"?null:[]];let v=d?{select:d}:void 0;try{if(p.type==="one"){let g=`by${ee(p.key)}`,b=typeof m.get?.[g]=="function"?await m.get[g](f,v):null;return [c,b]}else {let g=`by${ee(p.key)}`,b=typeof m.query?.[g]=="function"?await m.query[g](f,v):[];return [c,b]}}catch(g){return console.error(`[include] Error populating "${c}":`,g),[c,p.type==="one"?null:[]]}})),h={};for(let[c,d]of y)c!==void 0&&(h[c]=d);return {...l,populated:h}}))};return t.forEach(a=>{let u=`by${ee(a)}`;o[u]=async(i,l={})=>{let y={...l,where:[[a,"==",i],...l.where??[]]};return (await ue(e,y)).docs.map(c=>F(c.data()))};}),o.by=async a=>(await ue(e,a)).docs.map(i=>F(i.data())),o.getAll=async(a={})=>(await ue(e,a)).docs.map(i=>F(i.data())),o.onSnapshot=(a,u,i)=>ne(e,a).onSnapshot(y=>{u(y.docs.map(h=>F(h.data())));},i),o.paginate=async a=>{let{include:u,...i}=a,l=await _e(e,i);if(u&&u.length>0){let y=await s(l.data,u);return {...l,data:y}}return l},o.paginateAll=async function*(a){let{include:u,...i}=a;for await(let l of He(e,i))if(u&&u.length>0){let y=await s(l.data,u);yield {...l,data:y};}else yield l;},o}function Dt(e,t){return {populate:async(n,r)=>{if(!e.relationalKeys)return {...n,populated:{}};let o,s={};if(typeof r=="object"&&!Array.isArray(r))if("relation"in r){let i=r;o=[i.relation],i.select&&(s[i.relation]=i.select);}else if("relations"in r){let i=r;o=Array.isArray(i.relations)?i.relations:[i.relations],s=i.select??{};}else o=[];else o=Array.isArray(r)?r:[r];let a=await Promise.all(o.map(async i=>{let l=e.relationalKeys?.[i];if(!l)return console.warn(`[populate] Relation "${i}" not found in config`),[i,void 0];let y=t[l.repo];if(!y)return console.warn(`[populate] Repository "${l.repo}" not found in mapping`),[i,void 0];let h=n[i];if(h==null)return [i,l.type==="one"?null:[]];let c=s[i],d=c?{select:c}:void 0;try{if(l.type==="one"){let p=`by${ee(l.key)}`,m=typeof y.get?.[p]=="function"?await y.get[p](h,d):null;return [i,m]}else {let p=`by${ee(l.key)}`,m=typeof y.query?.[p]=="function"?await y.query[p](h,d):[];return [i,m]}}catch(p){return console.error(`[populate] Error populating "${i}":`,p),[i,l.type==="one"?null:[]]}})),u={};for(let[i,l]of a)l!==void 0&&(u[i]=l);return {...n,populated:u}}}}function It(e,t){return {run:async n=>e.runTransaction(async r=>n({get:async(...s)=>{let a=t(...s),u=await r.get(a);return u.exists?F({...u.data(),docId:u.id}):null},set:(...s)=>{let a=s[s.length-1],u=typeof a=="object"&&a!==null&&"merge"in a,i=u?s[s.length-2]:s[s.length-1],l=u?s.slice(0,-2):s.slice(0,-1),y=u?a:{merge:true},h=t(...l);r.set(h,i,y);},update:(...s)=>{let a=s[s.length-1],u=s.slice(0,-1),i=t(...u);r.update(i,a);},delete:(...s)=>{let a=t(...s);r.delete(a);},raw:r}))}}function Dn(e){if(typeof e!="function")return [];let n=e.toString().match(/^\s*(?:function\s*\w*\s*)?\(([^)]*)\)/);if(!n?.[1])return [];let r=n[1].split(",").map(o=>o.trim().replace(/\s*[:=].*$/,"").trim()).filter(Boolean);return r.length<=2?[]:r.slice(1,-1)}function We(e,t,n={}){let r=t.isGroup?e.collectionGroup(t.path):e.collection(t.path),o=t.isGroup?null:e.collection(t.path),s=(...p)=>t.refCb(e,...p),a=Pt(r,t.foreignKeys,o,s,t.documentKey),u=Et(r,t.queryKeys,t.relationalKeys,n),i=$t(r),l=At(o,s,t.documentKey,t.pathKey,t.createdKey,t.updatedKey),y=Ot(e,s,t.documentKey,t.pathKey,t.createdKey,t.updatedKey),h=It(e,s),c=kt(e,t.createdKey,t.updatedKey),d=Dt(t,n);return {ref:r,documentRef:s,get:a,query:u,aggregate:i,...l,batch:y,transaction:h,bulk:c,...d,schema:t.schema,relationalKeys:t.relationalKeys,_systemKeys:[t.documentKey,t.pathKey,t.createdKey,t.updatedKey].filter(p=>typeof p=="string"),_pathKey:t.pathKey??null,_isGroup:!!t.isGroup,_parentKeys:t.isGroup?Dn(t.refCb):[],_createdKey:t.createdKey??null}}function In(e){let t=[],n=e.replace(/[.*+?^${}()|[\]\\]/g,r=>r===":"?r:`\\${r}`).replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g,(r,o)=>(t.push(o),"([^/]+)"));return {pattern:new RegExp(`^${n}$`),paramNames:t}}function Nn(e){let t=e.path??e.url??"/",n=t.indexOf("?");return n===-1?t:t.slice(0,n)}var G=class{constructor(){this.routes=[];this.middlewares=[];this.notFoundHandler=(t,n)=>{n.status(404).send("Not Found");};this.errorHandler=(t,n,r)=>{console.error("[MiniRouter]",t),r.status(500).send("Internal Server Error");};}use(t){return this.middlewares.push(t),this}get(t,n){return this.addRoute("GET",t,n)}post(t,n){return this.addRoute("POST",t,n)}put(t,n){return this.addRoute("PUT",t,n)}patch(t,n){return this.addRoute("PATCH",t,n)}delete(t,n){return this.addRoute("DELETE",t,n)}onNotFound(t){return this.notFoundHandler=t,this}onError(t){return this.errorHandler=t,this}addRoute(t,n,r){let{pattern:o,paramNames:s}=In(n);return this.routes.push({method:t.toUpperCase(),pattern:o,paramNames:s,handler:r}),this}async handle(t,n){let r=(t.method??"GET").toUpperCase(),o=Nn(t),s=null,a={};for(let l of this.routes){if(l.method!==r)continue;let y=o.match(l.pattern);if(y){s=l,a={},l.paramNames.forEach((h,c)=>{a[h]=decodeURIComponent(y[c+1]??"");});break}}let u=Object.assign(t,{params:a}),i=s?s.handler:this.notFoundHandler;try{await this.runMiddlewareChain(u,n,i);}catch(l){this.errorHandler(l,t,n);}}async runMiddlewareChain(t,n,r){let o=0,s=async()=>{if(o<this.middlewares.length){let a=this.middlewares[o++];await a(t,n,s);}else await r(t,n);};await s();}};var _n={string:"ZodString",number:"ZodNumber",bigint:"ZodBigInt",boolean:"ZodBoolean",date:"ZodDate",enum:"ZodEnum",nativeEnum:"ZodNativeEnum",literal:"ZodLiteral",object:"ZodObject",array:"ZodArray",optional:"ZodOptional",nullable:"ZodNullable",default:"ZodDefault",coerce:"ZodCoerce",union:"ZodUnion",undefined:"ZodUndefined",unknown:"ZodUnknown",any:"ZodAny",record:"ZodRecord"};function j(e){let t=e,n=t._zod?.def?.type;if(n)return _n[n]??`Zod${n.charAt(0).toUpperCase()}${n.slice(1)}`;let r=t._def?.typeName;return r||""}function q(e){let t=e;if(t._zod?.def?.innerType)return t._zod.def.innerType;if(t._def?.innerType)return t._def.innerType}function Nt(e){let t=e;if(t._zod?.def?.element)return t._zod.def.element;if(t._def?.type)return t._def.type}function _t(e){let t=e;if(t._zod?.def?.defaultValue!==void 0)return t._zod.def.defaultValue;let n=t._def?.defaultValue;return typeof n=="function"?n():n}function re(e){let t=e;return t.shape&&typeof t.shape=="object"?t.shape:t._zod?.def?.shape&&typeof t._zod.def.shape=="object"?t._zod.def.shape:t._def?.shape?typeof t._def.shape=="function"?t._def.shape():t._def.shape:{}}function Se(e){let t=e;return Array.isArray(t.options)?t.options:t._zod?.def?.entries?Object.values(t._zod.def.entries):Array.isArray(t._def?.values)?t._def.values:[]}function Te(e){let t=e;return t._zod?.def?.entries?t._zod.def.entries:t.enum&&typeof t.enum=="object"?t.enum:t._def?.values&&typeof t._def.values=="object"?t._def.values:{}}function je(e){let t=e;return Array.isArray(t._zod?.def?.values)?t._zod.def.values[0]:t._def?.value}function jt(e){let t=e,n=[],r=t._zod?.def?.checks;if(Array.isArray(r)){for(let s of r)s.format&&n.push(s.format);if(n.length>0)return n}let o=t._def?.checks;if(Array.isArray(o))for(let s of o)s.kind&&n.push(s.kind);return n}var oe="__sync_version";var jn=new Set(["ZodOptional","ZodNullable","ZodDefault"]);function Fn(e){let t=e,n=false;for(;;){let r=j(t);if(!jn.has(r))break;(r==="ZodOptional"||r==="ZodNullable")&&(n=true);let o=q(t);if(!o)break;t=o;}return {inner:t,nullable:n}}var zn={ZodString:"string",ZodNumber:"number",ZodBigInt:"bigint",ZodBoolean:"boolean",ZodDate:"timestamp",ZodEnum:"string",ZodNativeEnum:"string",ZodLiteral:"string"};function Ft(e,t,n,r,o,s,a,u){for(let[i,l]of Object.entries(e)){let y=n?`${n}__${i}`:i;if(o.has(i)||o.has(y))continue;let{inner:h,nullable:c}=Fn(l),d=j(h),p=r||c;if(d==="ZodObject"){let g=re(h);Ft(g,t,y,p,o,s,a,u);continue}let m=zn[d]??"json",f=y===a||i===a,v=s[y]??s[i]??y;u.push({name:v,sqlType:t.mapType(m),nullable:f?false:p,isPrimaryKey:f});}}function Fe(e,t,n={}){let{primaryKey:r,exclude:o=[],columnMap:s={}}=n,a=new Set(o),u=re(e),i=[];return Ft(u,t,"",false,a,s,r,i),i.some(l=>l.name===oe)||i.push({name:oe,sqlType:t.mapType("bigint"),nullable:true,isPrimaryKey:false,description:"Monotonic publish version (Date.now() ms). Internal."}),i}function zt(e){if(e==null)return null;if(typeof e=="object"&&typeof e.toDate=="function")return e.toDate().toISOString();if(e instanceof Date)return e.toISOString();if(Buffer.isBuffer(e))return e.toString("base64");if(e instanceof Uint8Array)return Buffer.from(e).toString("base64");if(typeof e=="object"&&"latitude"in e&&"longitude"in e){let t=e;return JSON.stringify({lat:t.latitude,lng:t.longitude})}return Array.isArray(e)?JSON.stringify(e.map(zt)):e}function Zt(e,t,n){for(let[r,o]of Object.entries(e)){let s=t?`${t}__${r}`:r;o!=null&&typeof o=="object"&&!Array.isArray(o)&&!(o instanceof Date)&&!Buffer.isBuffer(o)&&!(o instanceof Uint8Array)&&typeof o.toDate!="function"&&!("latitude"in o&&"longitude"in o)?Zt(o,s,n):n[s]=zt(o);}}function Ce(e,t){let n=new Set(t?.exclude),r=t?.columnMap??{},o={};Zt(e,"",o);let s={};for(let[a,u]of Object.entries(o)){if(n.has(a))continue;let i=a.split("__")[0];if(i!==a&&n.has(i))continue;let l=r[a]??(a.includes("__")?r[a.split("__").pop()]:void 0)??a;s[l]=u;}return s}function pe(e,t){if(process.env.FUNCTIONS_EMULATOR==="true"){let o=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??"demo-project",s=process.env.FUNCTION_REGION??"us-central1",a=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${o}/${s}/${a}${t}`}let n=process.env.K_SERVICE,r=e.hostname??e.headers?.host??"";return n&&r.includes("cloudfunctions.net")?`/${n.toLowerCase()}${t}`:t}function W(e,t,n){return `<!DOCTYPE html>
1
+ import {jsx,jsxs,Fragment}from'hono/jsx/jsx-runtime';import {Timestamp}from'firebase-admin/firestore';import {z as z$1}from'zod';import {renderToString}from'hono/jsx/dom/server';var Xn=Object.defineProperty;var He=(e,t)=>()=>(e&&(t=e(e=0)),t);var Yn=(e,t)=>{for(var n in t)Xn(e,n,{get:t[n],enumerable:true});};function M(e){let t=e,n=t._zod?.def?.type;if(n)return ir[n]??`Zod${n.charAt(0).toUpperCase()}${n.slice(1)}`;let r=t._def?.typeName;return r||""}function U(e){let t=e;if(t._zod?.def?.innerType)return t._zod.def.innerType;if(t._def?.innerType)return t._def.innerType}function en(e){let t=e;if(t._zod?.def?.element)return t._zod.def.element;if(t._def?.type)return t._def.type}function tn(e){let t=e;if(t._zod?.def?.defaultValue!==void 0)return t._zod.def.defaultValue;let n=t._def?.defaultValue;return typeof n=="function"?n():n}function J(e){let t=e;return t.shape&&typeof t.shape=="object"?t.shape:t._zod?.def?.shape&&typeof t._zod.def.shape=="object"?t._zod.def.shape:t._def?.shape?typeof t._def.shape=="function"?t._def.shape():t._def.shape:{}}function Ne(e){let t=e;return Array.isArray(t.options)?t.options:t._zod?.def?.entries?Object.values(t._zod.def.entries):Array.isArray(t._def?.values)?t._def.values:[]}function _e(e){let t=e;return t._zod?.def?.entries?t._zod.def.entries:t.enum&&typeof t.enum=="object"?t.enum:t._def?.values&&typeof t._def.values=="object"?t._def.values:{}}function Je(e){let t=e;return Array.isArray(t._zod?.def?.values)?t._zod.def.values[0]:t._def?.value}function nn(e){let t=e,n=[],r=t._zod?.def?.checks;if(Array.isArray(r)){for(let o of r)o.format&&n.push(o.format);if(n.length>0)return n}let s=t._def?.checks;if(Array.isArray(s))for(let o of s)o.kind&&n.push(o.kind);return n}var ir,je=He(()=>{ir={string:"ZodString",number:"ZodNumber",bigint:"ZodBigInt",boolean:"ZodBoolean",date:"ZodDate",enum:"ZodEnum",nativeEnum:"ZodNativeEnum",literal:"ZodLiteral",object:"ZodObject",array:"ZodArray",optional:"ZodOptional",nullable:"ZodNullable",default:"ZodDefault",coerce:"ZodCoerce",union:"ZodUnion",undefined:"ZodUndefined",unknown:"ZodUnknown",any:"ZodAny",record:"ZodRecord"};});function hn({val:e}){return jsx("span",{class:"text-sm text-base-content/80 font-mono tabular-nums whitespace-nowrap",children:e.toLocaleString()})}function hr({message:e}){return jsx("span",{class:"tooltip tooltip-warning tooltip-right inline-flex align-middle ml-1 text-warning","data-tip":e,role:"img","aria-label":e,children:jsx("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24",fill:"currentColor",class:"size-3.5",children:jsx("path",{d:"M12 2 1 22h22L12 2zm0 6 7.5 13h-15L12 8zm-1 4v4h2v-4h-2zm0 5v2h2v-2h-2z"})})})}function $e({val:e,mismatch:t}){let n=br(e);return t?jsxs("span",{class:"inline-flex items-start gap-0.5",children:[n,jsx(hr,{message:t})]}):n}function br(e){if(e==null)return jsx("span",{class:"opacity-30 italic text-xs",children:"\u2014"});if(typeof e=="boolean")return e?jsx("span",{class:"badge badge-success badge-sm",children:"true"}):jsx("span",{class:"badge badge-error badge-sm",children:"false"});if(e instanceof Date)return jsx(hn,{val:e});if(typeof e=="object"&&e!==null&&typeof e.toDate=="function")return jsx(hn,{val:e.toDate()});if(typeof e=="number")return jsx("span",{class:"text-sm font-mono tabular-nums",children:String(e)});if(Array.isArray(e))return e.length===0?jsx("span",{class:"text-xs text-base-content/30",children:"[]"}):jsxs("ul",{class:"list-none p-0 m-0 space-y-0.5 text-xs",children:[e.slice(0,8).map((n,r)=>jsx("li",{class:"break-all",children:typeof n=="object"?JSON.stringify(n):String(n)},r)),e.length>8&&jsxs("li",{class:"text-base-content/40 italic",children:["+",e.length-8," more\u2026"]})]});if(typeof e=="object"&&e!==null){let n=Object.entries(e);return n.length===0?jsx("span",{class:"text-xs text-base-content/30",children:"{}"}):jsxs("dl",{class:"grid grid-cols-[auto_1fr] gap-x-2 gap-y-0.5 text-xs m-0",children:[n.slice(0,8).map(([r,s])=>jsxs(Fragment,{children:[jsx("dt",{class:"text-base-content/50 font-semibold whitespace-nowrap",children:r}),jsx("dd",{class:"break-all",children:String(s??"")})]})),n.length>8&&jsxs("dt",{class:"col-span-2 text-base-content/40 italic",children:["+",n.length-8," more\u2026"]})]})}let t=String(e);return jsx("span",{class:"text-sm break-all",children:t})}var et=He(()=>{});function Ae(e){if(!e)return "unknown";let t=ft(e);switch(M(t)){case "ZodString":return "string";case "ZodNumber":return "number";case "ZodBigInt":return "bigint";case "ZodBoolean":return "boolean";case "ZodDate":return "date";case "ZodArray":return "array";case "ZodObject":case "ZodRecord":return "object";case "ZodEnum":case "ZodNativeEnum":return "enum";case "ZodLiteral":return "literal";default:return "unknown"}}function ft(e){let t=M(e);if(t==="ZodOptional"||t==="ZodNullable"||t==="ZodDefault"){let n=U(e);return n?ft(n):e}return e}function Oe(e,t){if(!e)return;let n=t.split("."),r=e;for(let s of n){if(!r)return;let o=ft(r);if(M(o)!=="ZodObject")return;r=J(o)[s];}return r}function wr(e){return e==null?"null":e instanceof Date?"date":Array.isArray(e)?"array":typeof e=="string"?"string":typeof e=="boolean"?"boolean":typeof e=="bigint"?"bigint":typeof e=="number"?"number":typeof e=="object"?typeof e.toDate=="function"?"date":"object":"unknown"}function Me(e,t){if(e==="unknown")return null;let n=wr(t);return n==="null"?null:e==="enum"||e==="literal"?n==="string"||n==="number"?null:`Expected ${e} (string/number), got ${n}`:e===n||e==="number"&&n==="bigint"||e==="bigint"&&n==="number"?null:`Expected ${e}, got ${n}`}var mt=He(()=>{je();});var xn={};Yn(xn,{PanelMany:()=>Tr,PanelOne:()=>Sr,RightPanel:()=>yt});function Sr({doc:e,repoName:t,basePath:n,schema:r,columns:s}){if(!e)return jsx("div",{class:"text-center py-12 text-base-content/50",children:"Document not found."});let o=String(e.docId??e.id??""),a=`${n}/${t}/${encodeURIComponent(o)}/edit`,d=vn(r,s);return jsxs("div",{class:"space-y-4",children:[jsxs("div",{class:"flex items-center justify-between gap-2",children:[jsxs("div",{class:"text-xs text-base-content/60",children:[jsx("span",{class:"font-mono",children:t}),jsx("span",{class:"opacity-50",children:" \xB7 "}),jsx("span",{class:"font-mono break-all",children:o})]}),jsx("a",{href:a,class:"btn btn-sm btn-primary",children:"Edit \u2192"})]}),jsx("dl",{class:"grid grid-cols-[max-content_1fr] gap-x-4 gap-y-2 border border-base-300 rounded-box p-4 bg-base-100",children:s.map(i=>{let f=e[i],g=d[i],x=g?Me(g,f):null;return jsxs(Fragment,{children:[jsx("dt",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide pt-0.5",children:i}),jsx("dd",{class:"min-w-0",children:jsx($e,{val:f,mismatch:x})})]})})})]})}function Tr({docs:e,repoName:t,basePath:n,fk:r,fv:s,columns:o,schema:a,pagination:d}){let i=`${n}/${t}?fv_${r}=${encodeURIComponent(s)}`,f=vn(a,o);return jsxs("div",{class:"space-y-4",children:[jsxs("div",{class:"flex items-center justify-between gap-2",children:[jsxs("div",{class:"text-xs text-base-content/60",children:[jsx("span",{class:"font-mono",children:t}),jsx("span",{class:"opacity-50",children:" where "}),jsx("span",{class:"font-mono",children:r}),jsx("span",{class:"opacity-50",children:" = "}),jsx("span",{class:"font-mono break-all",children:s}),jsx("span",{class:"opacity-50",children:" \xB7 "}),jsxs("span",{children:[e.length," doc",e.length!==1?"s":""]})]}),jsx("a",{href:i,class:"btn btn-sm btn-outline",children:"Full view \u2192"})]}),jsx("div",{class:"overflow-x-auto rounded-box border border-base-300 bg-base-100",children:jsxs("table",{class:"table table-xs w-full",children:[jsx("thead",{children:jsxs("tr",{class:"bg-base-200/50",children:[o.map((g,x)=>jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:g},x)),jsx("th",{})]})}),jsx("tbody",{children:e.length===0?jsx("tr",{children:jsx("td",{colspan:o.length+1,class:"text-center py-10 text-base-content/40",children:"No related documents"})}):e.map((g,x)=>{let h=String(g.docId??g.id??""),b=`${n}/${t}/${encodeURIComponent(h)}/edit`;return jsxs("tr",{class:"hover",children:[o.map((m,y)=>{let c=g[m],p=f[m],u=p?Me(p,c):null;return jsx("td",{class:"align-top py-1.5",children:jsx($e,{val:c,mismatch:u})},y)}),jsx("td",{class:"text-right py-1.5",children:jsx("a",{href:b,class:"btn btn-xs btn-ghost",children:"Edit"})})]},x)})})]})}),(d.hasPrev||d.hasNext)&&jsxs("div",{class:"flex justify-center items-center gap-2",children:[d.hasPrev?jsx("button",{type:"button",class:"btn btn-xs btn-outline","data-frs-panel-page":"prev","data-cursor":d.prevCursor,children:"\u2190 Previous"}):jsx("button",{class:"btn btn-xs btn-outline",disabled:true,children:"\u2190 Previous"}),d.hasNext?jsx("button",{type:"button",class:"btn btn-xs btn-outline","data-frs-panel-page":"next","data-cursor":d.nextCursor,children:"Next \u2192"}):jsx("button",{class:"btn btn-xs btn-outline",disabled:true,children:"Next \u2192"})]})]})}function vn(e,t){if(!e)return {};let n={};for(let r of t)n[r]=Ae(Oe(e,r));return n}var yt,gt=He(()=>{et();mt();yt=()=>jsxs("div",{class:"fixed inset-0 z-[100] hidden pointer-events-none","data-frs-panel-root":true,"aria-hidden":"true",children:[jsx("div",{class:"absolute inset-0 bg-black/30 opacity-0 transition-opacity duration-200 pointer-events-auto","data-frs-panel-backdrop":true}),jsxs("aside",{class:"absolute top-0 right-0 h-full w-full md:w-1/2 bg-base-100 shadow-2xl border-l border-base-300 translate-x-full transition-transform duration-200 pointer-events-auto flex flex-col","data-frs-panel":true,role:"dialog","aria-label":"Relation preview",children:[jsxs("header",{class:"flex items-center justify-between px-5 py-3 border-b border-base-300 bg-base-200/40 shrink-0",children:[jsx("h2",{class:"font-semibold text-base truncate","data-frs-panel-title":true,children:"Relation"}),jsx("button",{type:"button",class:"btn btn-sm btn-ghost btn-circle","data-frs-panel-close":true,"aria-label":"Close panel",children:"\u2715"})]}),jsx("div",{class:"flex-1 overflow-auto p-5 text-sm","data-frs-panel-body":true,children:jsx("div",{class:"flex items-center justify-center py-12 text-base-content/40",children:jsx("span",{class:"loading loading-spinner loading-md"})})})]})]});});function er(e,t){let n=[];for(let r=0;r<e.length;r+=t)n.push(e.slice(r,r+t));return n}function De(e,t){return t.orderBy&&t.orderBy.forEach(n=>{e=e.orderBy(String(n.field),n.direction??"asc");}),t.limit&&(e=e.limit(t.limit)),t.offset&&(e=e.offset(t.offset)),t.select&&t.select.length>0&&(e=e.select(...t.select.map(n=>String(n)))),t.startAt&&(e=Array.isArray(t.startAt)?e.startAt(...t.startAt):e.startAt(t.startAt)),t.startAfter&&(e=Array.isArray(t.startAfter)?e.startAfter(...t.startAfter):e.startAfter(t.startAfter)),t.endAt&&(e=Array.isArray(t.endAt)?e.endAt(...t.endAt):e.endAt(t.endAt)),t.endBefore&&(e=Array.isArray(t.endBefore)?e.endBefore(...t.endBefore):e.endBefore(t.endBefore)),e}function ot(e){let[,t,n]=e;return (t==="in"||t==="array-contains-any")&&Array.isArray(n)&&n.length>30}function qt(e){let[t,n,r]=e;return ot(e)?er(r,30).map(o=>[t,n,o]):[e]}function Ge(e,t){let n=e;for(let[r,s,o]of t)n=n.where(String(r),s,o);return n}async function Kt(e){let t=await Promise.all(e.map(s=>s.get())),n=new Map;t.forEach(s=>{s.docs.forEach(o=>{n.has(o.id)||n.set(o.id,o);});});let r=t[0];if(!r)throw new Error("No snapshots returned");return {...r,docs:Array.from(n.values()),size:n.size,empty:n.size===0}}async function be(e,t){let n=t.orWhere&&t.orWhere.length>0,r=t.orWhereGroups&&t.orWhereGroups.length>0;if(!n&&!r){if(!t.where||t.where.length===0)return De(e,t).get();if(!t.where.some(ot)){let x=Ge(e,t.where);return x=De(x,t),x.get()}let i=t.where.map(qt),g=at(i).map(x=>{let h=Ge(e,x);return h=De(h,t),h});return Kt(g)}let s=t.where??[],o=[...t.orWhere?.map(d=>[d])??[],...t.orWhereGroups??[]],a=[];for(let d of o){let i=[...s,...d];if(i.some(ot)){let g=i.map(qt),h=at(g).map(b=>{let m=Ge(e,b);return m=De(m,t),m});a.push(...h);}else {let g=Ge(e,i);g=De(g,t),a.push(g);}}return Kt(a)}function at(e){if(e.length===0)return [[]];let t=e[0];if(e.length===1&&t)return t.map(o=>[o]);if(!t)return [[]];let n=e.slice(1),r=at(n),s=[];for(let o of t)for(let a of r)s.push([o,...a]);return s}var it="preserve";function tr(e){it=e;}function lt(){return it}function nr(e){return typeof e=="object"&&e!==null&&typeof e._seconds=="number"&&typeof e._nanoseconds=="number"}function ct(e){if(e==null)return null;if(e instanceof Date)return Number.isNaN(e.getTime())?null:e;if(e instanceof Timestamp)return e.toDate();if(nr(e))return new Date(e._seconds*1e3+Math.floor(e._nanoseconds/1e6));if(typeof e=="string"){let t=new Date(e);return Number.isNaN(t.getTime())?null:t}if(typeof e=="number"){let t=new Date(e);return Number.isNaN(t.getTime())?null:t}return null}function rr(e){if(typeof e!="object"||e===null)return false;let t=Object.getPrototypeOf(e);return t===Object.prototype||t===null}function We(e){if(e instanceof Timestamp)return e.toDate();if(Array.isArray(e))return e.map(t=>We(t));if(rr(e)){let t={};for(let[n,r]of Object.entries(e))t[n]=We(r);return t}return e}function q(e){return it==="normalize"?We(e):e}function Lt(e,t){let n=[];for(let r=0;r<e.length;r+=t)n.push(e.slice(r,r+t));return n}function ie(e){return e.charAt(0).toUpperCase()+e.slice(1)}function le(e,t){return t.where&&t.where.forEach(([n,r,s])=>{e=e.where(String(n),r,s);}),t.orderBy&&t.orderBy.forEach(n=>{e=e.orderBy(String(n.field),n.direction??"asc");}),t.limit&&(e=e.limit(t.limit)),t.offset&&(e=e.offset(t.offset)),t.select&&t.select.length>0&&(e=e.select(...t.select.map(n=>String(n)))),t.startAt&&(e=Array.isArray(t.startAt)?e.startAt(...t.startAt):e.startAt(t.startAt)),t.startAfter&&(e=Array.isArray(t.startAfter)?e.startAfter(...t.startAfter):e.startAfter(t.startAfter)),t.endAt&&(e=Array.isArray(t.endAt)?e.endAt(...t.endAt):e.endAt(t.endAt)),t.endBefore&&(e=Array.isArray(t.endBefore)?e.endBefore(...t.endBefore):e.endBefore(t.endBefore)),e}async function Ve(e,t){let n={...t,limit:t.pageSize+1};t.cursor&&(t.direction==="prev"?n.endBefore=t.cursor:n.startAfter=t.cursor);let s=(await be(e,n)).docs,o=s.length>t.pageSize,a=o?s.slice(0,t.pageSize):s,d=a.map(f=>q({...f.data(),docId:f.id})),i=t.direction==="prev";return {data:d,nextCursor:i?a.length>0?a[a.length-1]:void 0:o?a[a.length-1]:void 0,prevCursor:i?o?a[0]:void 0:a[0],hasNextPage:i?!!t.cursor:o,hasPrevPage:i?o:!!t.cursor,pageSize:d.length}}async function*dt(e,t){let n,r=true;for(;r;){let s=await Ve(e,{...t,cursor:n,direction:"next"});yield s,r=s.hasNextPage,n=s.nextCursor;}}function Qt(e){return {count:async(t={})=>{let n=e;return n=le(n,t),(await n.count().get()).data().count},sum:async(t,n={})=>{let r=e;r=le(r,n);let s=await r.get(),o=0;return s.forEach(a=>{let d=a.data()[t];typeof d=="number"&&(o+=d);}),o},average:async(t,n={})=>{let r=e;r=le(r,n);let s=await r.get();if(s.empty)return null;let o=0,a=0;return s.forEach(d=>{let i=d.data()[t];typeof i=="number"&&(o+=i,a++);}),a>0?o/a:null}}}function Ht(e,t,n,r,s,o){let a=()=>new Date;return {create:()=>{let d=e.batch();return {batch:d,set:(...i)=>{let f=i[i.length-1],g=typeof f=="object"&&f!==null&&"merge"in f,x=g?i[i.length-2]:i[i.length-1],h=g?i.slice(0,-2):i.slice(0,-1),b=g?f:{merge:true},m=t(...h),y={...x},c=h[h.length-1];n&&c&&(y[n]=c),r&&(y[r]=m.path),s&&(y[s]=a()),o&&(y[o]=a()),d.set(m,y,b);},update:(...i)=>{let f=i.pop(),x=t(...i),h={...f};o&&(h[o]=a()),d.update(x,h);},delete:(...i)=>{let f=t(...i);d.delete(f);},commit:async()=>{await d.commit();}}}}}function Gt(e,t,n){let r=()=>new Date;return {set:async s=>{let o=e.bulkWriter(),a=0;for(let d of s){if(!d)continue;let{docRef:i,data:f,merge:g=true}=d,x={...f};t&&(x[t]=r()),n&&(x[n]=r()),o.set(i,x,{merge:g}),a++,a>=500&&(await o.flush(),a=0);}await o.close();},update:async s=>{let o=e.bulkWriter(),a=0;for(let d of s){if(!d)continue;let{docRef:i,data:f}=d,g={...f};n&&(g[n]=r()),o.update(i,g),a++,a>=500&&(await o.flush(),a=0);}await o.close();},delete:async s=>{let o=e.bulkWriter(),a=0;for(let d of s)d&&(o.delete(d),a++,a>=500&&(await o.flush(),a=0));await o.close();}}}function Wt(e,t,n,r,s,o){let a=()=>new Date;return {create:async x=>{if(!e)throw new Error("Cannot use create() on collection groups. Use set() with a specific document ID instead.");let h,b,m={...x};if(s&&(m[s]=a()),o&&(m[o]=a()),x[n]){b=x[n],h=e.doc(b);let c=r?{...m,[r]:h.path}:m;await h.set(c);}else {h=await e.add(m),b=h.id;let c={[n]:b};r&&(c[r]=h.path),await h.update(c);}let y=await h.get();return q(y.data())},set:async(...x)=>{let h=x[x.length-1],b=typeof h=="object"&&h!==null&&"merge"in h,m=b?x[x.length-2]:x[x.length-1],y=b?x.slice(0,-2):x.slice(0,-1),c=b?h:{merge:true},p={...m};o&&(p[o]=a());let u=t(...y),l=y[y.length-1];n&&l!=null&&(p[n]=l),r&&(p[r]=u.path),await u.set(p,c);let v=await u.get();return q(v.data())},update:async(...x)=>{let h=x.pop(),b=x,m={...h};o&&(m[o]=a());let y=t(...b);await y.update(m);let c=await y.get();return q(c.data())},delete:async(...x)=>{await t(...x).delete();}}}function Vt(e,t,n,r,s){let o={};return o.byList=async(a,d,i="in",f={})=>{if(d.length===0)return [];let g=[],x=Lt(d,30);for(let h of x){let b=e;b=b.where(a,i,h),f.select&&f.select.length>0&&(b=b.select(...f.select.map(y=>String(y)))),(await b.get()).forEach(y=>{let c=q(y.data());g.push(f.returnDoc?{data:c,doc:y}:c);});}return g},t.forEach(a=>{let d=`by${ie(String(a))}`;o[d]=async(i,f={})=>{let g=typeof f=="boolean"?{returnDoc:f}:f;if(String(a)===s){let c=await r(i).get();if(!c.exists)return null;let p=q(c.data());return g.returnDoc?{data:p,doc:c}:p}let x=e;x=x.where(String(a),"==",i).limit(1),g.select&&g.select.length>0&&(x=x.select(...g.select.map(y=>String(y))));let h=await x.get();if(h.empty)return null;let b=h.docs[0];if(!b)return null;let m=q(b.data());return g.returnDoc?{data:m,doc:b}:m};}),o}function Jt(e,t,n,r){let s={},o=async(a,d)=>{if(!n||!r||d.length===0)return a;let i=d.map(f=>typeof f=="string"?{key:f}:{key:f.relation,select:f.select});return Promise.all(a.map(async f=>{let g=await Promise.all(i.map(async({key:h,select:b})=>{let m=n[h];if(!m)return [h,void 0];let y=r[m.repo];if(!y)return [h,void 0];let c=f[h];if(c==null)return [h,m.type==="one"?null:[]];let p=b?{select:b}:void 0;try{if(m.type==="one"){let u=`by${ie(m.key)}`,l=typeof y.get?.[u]=="function"?await y.get[u](c,p):null;return [h,l]}else {let u=`by${ie(m.key)}`,l=typeof y.query?.[u]=="function"?await y.query[u](c,p):[];return [h,l]}}catch(u){return console.error(`[include] Error populating "${h}":`,u),[h,m.type==="one"?null:[]]}})),x={};for(let[h,b]of g)h!==void 0&&(x[h]=b);return {...f,populated:x}}))};return t.forEach(a=>{let d=`by${ie(a)}`;s[d]=async(i,f={})=>{let g={...f,where:[[a,"==",i],...f.where??[]]};return (await be(e,g)).docs.map(h=>q(h.data()))};}),s.by=async a=>(await be(e,a)).docs.map(i=>q(i.data())),s.getAll=async(a={})=>(await be(e,a)).docs.map(i=>q(i.data())),s.onSnapshot=(a,d,i)=>le(e,a).onSnapshot(g=>{d(g.docs.map(x=>q(x.data())));},i),s.paginate=async a=>{let{include:d,...i}=a,f=await Ve(e,i);if(d&&d.length>0){let g=await o(f.data,d);return {...f,data:g}}return f},s.paginateAll=async function*(a){let{include:d,...i}=a;for await(let f of dt(e,i))if(d&&d.length>0){let g=await o(f.data,d);yield {...f,data:g};}else yield f;},s}function Xt(e,t){return {populate:async(n,r)=>{if(!e.relationalKeys)return {...n,populated:{}};let s,o={};if(typeof r=="object"&&!Array.isArray(r))if("relation"in r){let i=r;s=[i.relation],i.select&&(o[i.relation]=i.select);}else if("relations"in r){let i=r;s=Array.isArray(i.relations)?i.relations:[i.relations],o=i.select??{};}else s=[];else s=Array.isArray(r)?r:[r];let a=await Promise.all(s.map(async i=>{let f=e.relationalKeys?.[i];if(!f)return console.warn(`[populate] Relation "${i}" not found in config`),[i,void 0];let g=t[f.repo];if(!g)return console.warn(`[populate] Repository "${f.repo}" not found in mapping`),[i,void 0];let x=n[i];if(x==null)return [i,f.type==="one"?null:[]];let h=o[i],b=h?{select:h}:void 0;try{if(f.type==="one"){let m=`by${ie(f.key)}`,y=typeof g.get?.[m]=="function"?await g.get[m](x,b):null;return [i,y]}else {let m=`by${ie(f.key)}`,y=typeof g.query?.[m]=="function"?await g.query[m](x,b):[];return [i,y]}}catch(m){return console.error(`[populate] Error populating "${i}":`,m),[i,f.type==="one"?null:[]]}})),d={};for(let[i,f]of a)f!==void 0&&(d[i]=f);return {...n,populated:d}}}}function Yt(e,t){return {run:async n=>e.runTransaction(async r=>n({get:async(...o)=>{let a=t(...o),d=await r.get(a);return d.exists?q({...d.data(),docId:d.id}):null},set:(...o)=>{let a=o[o.length-1],d=typeof a=="object"&&a!==null&&"merge"in a,i=d?o[o.length-2]:o[o.length-1],f=d?o.slice(0,-2):o.slice(0,-1),g=d?a:{merge:true},x=t(...f);r.set(x,i,g);},update:(...o)=>{let a=o[o.length-1],d=o.slice(0,-1),i=t(...d);r.update(i,a);},delete:(...o)=>{let a=t(...o);r.delete(a);},raw:r}))}}function sr(e){if(typeof e!="function")return [];let n=e.toString().match(/^\s*(?:function\s*\w*\s*)?\(([^)]*)\)/);if(!n?.[1])return [];let r=n[1].split(",").map(s=>s.trim().replace(/\s*[:=].*$/,"").trim()).filter(Boolean);return r.length<=2?[]:r.slice(1,-1)}function ut(e,t,n={}){let r=t.isGroup?e.collectionGroup(t.path):e.collection(t.path),s=t.isGroup?null:e.collection(t.path),o=(...m)=>t.refCb(e,...m),a=Vt(r,t.foreignKeys,s,o,t.documentKey),d=Jt(r,t.queryKeys,t.relationalKeys,n),i=Qt(r),f=Wt(s,o,t.documentKey,t.pathKey,t.createdKey,t.updatedKey),g=Ht(e,o,t.documentKey,t.pathKey,t.createdKey,t.updatedKey),x=Yt(e,o),h=Gt(e,t.createdKey,t.updatedKey),b=Xt(t,n);return {ref:r,documentRef:o,get:a,query:d,aggregate:i,...f,batch:g,transaction:x,bulk:h,...b,schema:t.schema,relationalKeys:t.relationalKeys,_systemKeys:[t.documentKey,t.pathKey,t.createdKey,t.updatedKey].filter(m=>typeof m=="string"),_pathKey:t.pathKey??null,_isGroup:!!t.isGroup,_parentKeys:t.isGroup?sr(t.refCb):[],_createdKey:t.createdKey??null}}function or(e){let t=[],n=e.replace(/[.*+?^${}()|[\]\\]/g,r=>r===":"?r:`\\${r}`).replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g,(r,s)=>(t.push(s),"([^/]+)"));return {pattern:new RegExp(`^${n}$`),paramNames:t}}function ar(e){let t=e.path??e.url??"/",n=t.indexOf("?");return n===-1?t:t.slice(0,n)}var X=class{constructor(){this.routes=[];this.middlewares=[];this.notFoundHandler=(t,n)=>{n.status(404).send("Not Found");};this.errorHandler=(t,n,r)=>{console.error("[MiniRouter]",t),r.status(500).send("Internal Server Error");};}use(t){return this.middlewares.push(t),this}get(t,n){return this.addRoute("GET",t,n)}post(t,n){return this.addRoute("POST",t,n)}put(t,n){return this.addRoute("PUT",t,n)}patch(t,n){return this.addRoute("PATCH",t,n)}delete(t,n){return this.addRoute("DELETE",t,n)}onNotFound(t){return this.notFoundHandler=t,this}onError(t){return this.errorHandler=t,this}addRoute(t,n,r){let{pattern:s,paramNames:o}=or(n);return this.routes.push({method:t.toUpperCase(),pattern:s,paramNames:o,handler:r}),this}async handle(t,n){let r=(t.method??"GET").toUpperCase(),s=ar(t),o=null,a={};for(let f of this.routes){if(f.method!==r)continue;let g=s.match(f.pattern);if(g){o=f,a={},f.paramNames.forEach((x,h)=>{a[x]=decodeURIComponent(g[h+1]??"");});break}}let d=Object.assign(t,{params:a}),i=o?o.handler:this.notFoundHandler;try{await this.runMiddlewareChain(d,n,i);}catch(f){this.errorHandler(f,t,n);}}async runMiddlewareChain(t,n,r){let s=0,o=async()=>{if(s<this.middlewares.length){let a=this.middlewares[s++];await a(t,n,o);}else await r(t,n);};await o();}};je();var ce="__sync_version";var lr=new Set(["ZodOptional","ZodNullable","ZodDefault"]);function cr(e){let t=e,n=false;for(;;){let r=M(t);if(!lr.has(r))break;(r==="ZodOptional"||r==="ZodNullable")&&(n=true);let s=U(t);if(!s)break;t=s;}return {inner:t,nullable:n}}var dr={ZodString:"string",ZodNumber:"number",ZodBigInt:"bigint",ZodBoolean:"boolean",ZodDate:"timestamp",ZodEnum:"string",ZodNativeEnum:"string",ZodLiteral:"string"};function rn(e,t,n,r,s,o,a,d){for(let[i,f]of Object.entries(e)){let g=n?`${n}__${i}`:i;if(s.has(i)||s.has(g))continue;let{inner:x,nullable:h}=cr(f),b=M(x),m=r||h;if(b==="ZodObject"){let u=J(x);rn(u,t,g,m,s,o,a,d);continue}let y=dr[b]??"json",c=g===a||i===a,p=o[g]??o[i]??g;d.push({name:p,sqlType:t.mapType(y),nullable:c?false:m,isPrimaryKey:c});}}function Xe(e,t,n={}){let{primaryKey:r,exclude:s=[],columnMap:o={}}=n,a=new Set(s),d=J(e),i=[];return rn(d,t,"",false,a,o,r,i),i.some(f=>f.name===ce)||i.push({name:ce,sqlType:t.mapType("bigint"),nullable:true,isPrimaryKey:false,description:"Monotonic publish version (Date.now() ms). Internal."}),i}function sn(e){if(e==null)return null;if(typeof e=="object"&&typeof e.toDate=="function")return e.toDate().toISOString();if(e instanceof Date)return e.toISOString();if(Buffer.isBuffer(e))return e.toString("base64");if(e instanceof Uint8Array)return Buffer.from(e).toString("base64");if(typeof e=="object"&&"latitude"in e&&"longitude"in e){let t=e;return JSON.stringify({lat:t.latitude,lng:t.longitude})}return Array.isArray(e)?JSON.stringify(e.map(sn)):e}function on(e,t,n){for(let[r,s]of Object.entries(e)){let o=t?`${t}__${r}`:r;s!=null&&typeof s=="object"&&!Array.isArray(s)&&!(s instanceof Date)&&!Buffer.isBuffer(s)&&!(s instanceof Uint8Array)&&typeof s.toDate!="function"&&!("latitude"in s&&"longitude"in s)?on(s,o,n):n[o]=sn(s);}}function ze(e,t){let n=new Set(t?.exclude),r=t?.columnMap??{},s={};on(e,"",s);let o={};for(let[a,d]of Object.entries(s)){if(n.has(a))continue;let i=a.split("__")[0];if(i!==a&&n.has(i))continue;let f=r[a]??(a.includes("__")?r[a.split("__").pop()]:void 0)??a;o[f]=d;}return o}function ve(e,t){if(process.env.FUNCTIONS_EMULATOR==="true"){let s=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??"demo-project",o=process.env.FUNCTION_REGION??"us-central1",a=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${s}/${o}/${a}${t}`}let n=process.env.K_SERVICE,r=e.hostname??e.headers?.host??"";return n&&r.includes("cloudfunctions.net")?`/${n.toLowerCase()}${t}`:t}function te(e,t,n){return `<!DOCTYPE html>
2
2
  <html lang="en"><head>
3
3
  <meta charset="utf-8"><meta name="viewport" content="width=device-width,initial-scale=1">
4
4
  <title>${e} \u2014 Sync Admin</title>
@@ -28,68 +28,68 @@ import {Timestamp}from'firebase-admin/firestore';import {z}from'zod';import {jsx
28
28
  <nav><a href="${t}/">\u2190 Dashboard</a></nav>
29
29
  <h1>${e}</h1>
30
30
  ${n}
31
- </body></html>`}function V(e,t,n=200){e.status(n).set("Content-Type","text/html; charset=utf-8").send(t);}function fe(e,t,n=200){e.status(n).set("Content-Type","application/json").send(JSON.stringify(t,null,2));}function $e(e){return (e.headers?.accept??"").includes("application/json")}function Mt(e,t,n,r,o,s,a,u){let i=(o.basePath??"/").replace(/\/$/,"")||"",l=o.featuresFlag??{},y=[];for(let[c,d]of Object.entries(e)){let p=s[c];y.push({name:c,schema:d.schema??null,documentKey:d._systemKeys?.[0]??d.documentKey??"docId",tableName:p?.tableName??c,isGroup:!!d._isGroup,repoCfg:p,repo:d});}let h=new G;if(o.auth)if(typeof o.auth=="function")h.use(o.auth);else {let c=o.auth.realm??"Sync Admin",d="Basic "+Buffer.from(`${o.auth.username}:${o.auth.password}`).toString("base64");h.use((p,m,f)=>{if((p.headers?.authorization??"")!==d){m.status(401).set("WWW-Authenticate",`Basic realm="${c}"`).set("Content-Type","text/plain").send("Unauthorized");return}f();});}return h.get(`${i}/`,(c,d)=>{let p=pe(c,i),m=y.map(b=>{let R=[];return l.healthCheck&&R.push(`<a class="btn" href="${p}/${b.name}/health">Health</a>`),l.manualSync&&R.push(`<a class="btn btn-primary" href="${p}/${b.name}/force-sync">Force Sync</a>`),`<tr>
32
- <td><strong>${b.name}</strong></td>
33
- <td>${b.tableName}</td>
34
- <td>${b.isGroup?'<span class="badge badge-warn">group</span>':'<span class="badge badge-ok">collection</span>'}</td>
35
- <td>${b.schema?"\u2713":"\u2717"}</td>
36
- <td>${R.join(" ")}</td>
31
+ </body></html>`}function ne(e,t,n=200){e.status(n).set("Content-Type","text/html; charset=utf-8").send(t);}function xe(e,t,n=200){e.status(n).set("Content-Type","application/json").send(JSON.stringify(t,null,2));}function Fe(e){return (e.headers?.accept??"").includes("application/json")}function an(e,t,n,r,s,o,a,d){let i=(s.basePath??"/").replace(/\/$/,"")||"",f=s.featuresFlag??{},g=[];for(let[h,b]of Object.entries(e)){let m=o[h];g.push({name:h,schema:b.schema??null,documentKey:b._systemKeys?.[0]??b.documentKey??"docId",tableName:m?.tableName??h,isGroup:!!b._isGroup,repoCfg:m,repo:b});}let x=new X;if(s.auth)if(typeof s.auth=="function")x.use(s.auth);else {let h=s.auth.realm??"Sync Admin",b="Basic "+Buffer.from(`${s.auth.username}:${s.auth.password}`).toString("base64");x.use((m,y,c)=>{if((m.headers?.authorization??"")!==b){y.status(401).set("WWW-Authenticate",`Basic realm="${h}"`).set("Content-Type","text/plain").send("Unauthorized");return}c();});}return x.get(`${i}/`,(h,b)=>{let m=ve(h,i),y=g.map(l=>{let v=[];return f.healthCheck&&v.push(`<a class="btn" href="${m}/${l.name}/health">Health</a>`),f.manualSync&&v.push(`<a class="btn btn-primary" href="${m}/${l.name}/force-sync">Force Sync</a>`),`<tr>
32
+ <td><strong>${l.name}</strong></td>
33
+ <td>${l.tableName}</td>
34
+ <td>${l.isGroup?'<span class="badge badge-warn">group</span>':'<span class="badge badge-ok">collection</span>'}</td>
35
+ <td>${l.schema?"\u2713":"\u2717"}</td>
36
+ <td>${v.join(" ")}</td>
37
37
  </tr>`}).join(`
38
- `),f=l.viewQueue?`<p><a class="btn" href="${p}/queues">View Queues</a></p>`:"",v=l.configCheck?`<p style="margin-top:.5rem"><a class="btn" href="${p}/config-check">\u2699 Config Check</a></p>`:"",g=W("Sync Dashboard",p,`<div class="card">
38
+ `),c=f.viewQueue?`<p><a class="btn" href="${m}/queues">View Queues</a></p>`:"",p=f.configCheck?`<p style="margin-top:.5rem"><a class="btn" href="${m}/config-check">\u2699 Config Check</a></p>`:"",u=te("Sync Dashboard",m,`<div class="card">
39
39
  <table>
40
40
  <thead><tr><th>Repository</th><th>Table</th><th>Type</th><th>Schema</th><th>Actions</th></tr></thead>
41
- <tbody>${m}</tbody>
41
+ <tbody>${y}</tbody>
42
42
  </table>
43
- ${f}
44
- ${v}
45
- </div>`);V(d,g);}),h.get(`${i}`,(c,d)=>{let p=pe(c,i);d.status(302).set("Location",`${p}/`).send("");}),l.healthCheck&&h.get(`${i}/:repoName/health`,async(c,d)=>{let p=pe(c,i),m=y.find($=>$.name===c.params.repoName);if(!m){V(d,W("Not Found",p,`<p>Unknown repo: ${c.params.repoName}</p>`),404);return}if(!m.schema){V(d,W("Health Check",p,`<p class="badge badge-warn">No Zod schema attached to "${m.name}"</p>`));return}let f=Fe(m.schema,t.dialect,{primaryKey:m.documentKey,exclude:m.repoCfg?.exclude,columnMap:m.repoCfg?.columnMap}),v=[],g=false,b=null;try{g=await t.tableExists(m.tableName),g&&(v=await t.getTableColumns(m.tableName));}catch($){b=$?.message??String($);}let R=new Set(v),x=new Set(f.map($=>$.name)),S=f.filter($=>!R.has($.name)),C=v.filter($=>!x.has($)),P=f.filter($=>R.has($.name)),w=g&&S.length===0&&!b;if($e(c)){fe(d,{repo:m.name,table:m.tableName,tableExists:g,healthy:w,error:b,columns:{expected:f.map($=>({name:$.name,type:$.sqlType,nullable:$.nullable,isPrimaryKey:$.isPrimaryKey})),actual:v,matched:P.map($=>$.name),missing:S.map($=>({name:$.name,type:$.sqlType})),extra:C}});return}let T=w?'<span class="badge badge-ok">Healthy</span>':'<span class="badge badge-err">Unhealthy</span>',O=f.map($=>{let K=R.has($.name)?'<span class="badge badge-ok">OK</span>':'<span class="badge badge-err">MISSING</span>';return `<tr><td>${$.name}</td><td>${$.sqlType}</td><td>${$.nullable?"Yes":"No"}</td><td>${$.isPrimaryKey?"\u2713":""}</td><td>${K}</td></tr>`}).join(`
46
- `),k=C.map($=>`<tr><td>${$}</td><td colspan="3" class="muted">not in schema</td><td><span class="badge badge-warn">EXTRA</span></td></tr>`).join(`
47
- `),I=W(`Health: ${m.name}`,p,`<div class="card">
48
- <p>Table: <code>${m.tableName}</code> ${g?T:'<span class="badge badge-err">NOT FOUND</span>'}</p>
49
- ${b?`<p class="badge badge-err">Error: ${b}</p>`:""}
43
+ ${c}
44
+ ${p}
45
+ </div>`);ne(b,u);}),x.get(`${i}`,(h,b)=>{let m=ve(h,i);b.status(302).set("Location",`${m}/`).send("");}),f.healthCheck&&x.get(`${i}/:repoName/health`,async(h,b)=>{let m=ve(h,i),y=g.find(C=>C.name===h.params.repoName);if(!y){ne(b,te("Not Found",m,`<p>Unknown repo: ${h.params.repoName}</p>`),404);return}if(!y.schema){ne(b,te("Health Check",m,`<p class="badge badge-warn">No Zod schema attached to "${y.name}"</p>`));return}let c=Xe(y.schema,t.dialect,{primaryKey:y.documentKey,exclude:y.repoCfg?.exclude,columnMap:y.repoCfg?.columnMap}),p=[],u=false,l=null;try{u=await t.tableExists(y.tableName),u&&(p=await t.getTableColumns(y.tableName));}catch(C){l=C?.message??String(C);}let v=new Set(p),R=new Set(c.map(C=>C.name)),T=c.filter(C=>!v.has(C.name)),$=p.filter(C=>!R.has(C)),O=c.filter(C=>v.has(C.name)),w=u&&T.length===0&&!l;if(Fe(h)){xe(b,{repo:y.name,table:y.tableName,tableExists:u,healthy:w,error:l,columns:{expected:c.map(C=>({name:C.name,type:C.sqlType,nullable:C.nullable,isPrimaryKey:C.isPrimaryKey})),actual:p,matched:O.map(C=>C.name),missing:T.map(C=>({name:C.name,type:C.sqlType})),extra:$}});return}let S=w?'<span class="badge badge-ok">Healthy</span>':'<span class="badge badge-err">Unhealthy</span>',k=c.map(C=>{let B=v.has(C.name)?'<span class="badge badge-ok">OK</span>':'<span class="badge badge-err">MISSING</span>';return `<tr><td>${C.name}</td><td>${C.sqlType}</td><td>${C.nullable?"Yes":"No"}</td><td>${C.isPrimaryKey?"\u2713":""}</td><td>${B}</td></tr>`}).join(`
46
+ `),A=$.map(C=>`<tr><td>${C}</td><td colspan="3" class="muted">not in schema</td><td><span class="badge badge-warn">EXTRA</span></td></tr>`).join(`
47
+ `),E=te(`Health: ${y.name}`,m,`<div class="card">
48
+ <p>Table: <code>${y.tableName}</code> ${u?S:'<span class="badge badge-err">NOT FOUND</span>'}</p>
49
+ ${l?`<p class="badge badge-err">Error: ${l}</p>`:""}
50
50
  <h2>Columns</h2>
51
51
  <table>
52
52
  <thead><tr><th>Column</th><th>SQL Type</th><th>Nullable</th><th>PK</th><th>Status</th></tr></thead>
53
- <tbody>${O}${k}</tbody>
53
+ <tbody>${k}${A}</tbody>
54
54
  </table>
55
- </div>`);V(d,I);}),l.manualSync&&(h.get(`${i}/:repoName/force-sync`,(c,d)=>{let p=pe(c,i),m=y.find(v=>v.name===c.params.repoName);if(!m){V(d,W("Not Found",p,`<p>Unknown repo: ${c.params.repoName}</p>`),404);return}let f=W(`Force Sync: ${m.name}`,p,`<div class="card">
56
- <p>This will read <strong>all</strong> documents from the <code>${m.name}</code> Firestore collection
57
- and upsert them into the <code>${m.tableName}</code> SQL table.</p>
55
+ </div>`);ne(b,E);}),f.manualSync&&(x.get(`${i}/:repoName/force-sync`,(h,b)=>{let m=ve(h,i),y=g.find(p=>p.name===h.params.repoName);if(!y){ne(b,te("Not Found",m,`<p>Unknown repo: ${h.params.repoName}</p>`),404);return}let c=te(`Force Sync: ${y.name}`,m,`<div class="card">
56
+ <p>This will read <strong>all</strong> documents from the <code>${y.name}</code> Firestore collection
57
+ and upsert them into the <code>${y.tableName}</code> SQL table.</p>
58
58
  <p class="muted" style="margin:.75rem 0">This may take a while for large collections.</p>
59
- <form method="POST" action="${p}/${m.name}/force-sync">
59
+ <form method="POST" action="${m}/${y.name}/force-sync">
60
60
  <button type="submit" class="btn btn-primary">Start Force Sync</button>
61
61
  </form>
62
- </div>`);V(d,f);}),h.post(`${i}/:repoName/force-sync`,async(c,d)=>{let p=pe(c,i),m=y.find(w=>w.name===c.params.repoName);if(!m){fe(d,{error:`Unknown repo: ${c.params.repoName}`},404);return}let f=m.repo.ref;if(!f){fe(d,{error:`No collection reference for "${m.name}"`},400);return}let v=0,g=0,b=[],R=500,x=f.limit(R),S=null;try{for(;;){let O=await(S?x.startAfter(S):x).get();if(O.empty)break;for(let k of O.docs){let I=k.data(),$=String(I[m.documentKey]??k.id),K=Ce(I,{exclude:m.repoCfg?.exclude,columnMap:m.repoCfg?.columnMap});try{await r({operation:"UPSERT",repoName:m.name,docId:$,data:K,timestamp:new Date().toISOString()}),v++;}catch(_){g++;let te=_?.message??String(_);console.error(`[ForceSync:${m.name}] doc=${$} failed:`,_),b.length<5&&b.push(`${$}: ${te}`);}}if(S=O.docs[O.docs.length-1],O.docs.length<R)break}let w=n.get(m.name);w&&await w.flush();}catch(w){if($e(c)){fe(d,{error:w?.message??String(w),synced:v,errors:g},500);return}V(d,W(`Force Sync: ${m.name}`,p,`<div class="card">
62
+ </div>`);ne(b,c);}),x.post(`${i}/:repoName/force-sync`,async(h,b)=>{let m=ve(h,i),y=g.find(w=>w.name===h.params.repoName);if(!y){xe(b,{error:`Unknown repo: ${h.params.repoName}`},404);return}let c=y.repo.ref;if(!c){xe(b,{error:`No collection reference for "${y.name}"`},400);return}let p=0,u=0,l=[],v=500,R=c.limit(v),T=null;try{for(;;){let k=await(T?R.startAfter(T):R).get();if(k.empty)break;for(let A of k.docs){let E=A.data(),C=String(E[y.documentKey]??A.id),B=ze(E,{exclude:y.repoCfg?.exclude,columnMap:y.repoCfg?.columnMap});try{await r({operation:"UPSERT",repoName:y.name,docId:C,data:B,timestamp:new Date().toISOString()}),p++;}catch(D){u++;let _=D?.message??String(D);console.error(`[ForceSync:${y.name}] doc=${C} failed:`,D),l.length<5&&l.push(`${C}: ${_}`);}}if(T=k.docs[k.docs.length-1],k.docs.length<v)break}let w=n.get(y.name);w&&await w.flush();}catch(w){if(Fe(h)){xe(b,{error:w?.message??String(w),synced:p,errors:u},500);return}ne(b,te(`Force Sync: ${y.name}`,m,`<div class="card">
63
63
  <p class="badge badge-err">Error: ${w?.message??String(w)}</p>
64
- <p>Synced ${v} docs before failure (${g} errors).</p>
65
- </div>`),500);return}if($e(c)){fe(d,{repo:m.name,table:m.tableName,synced:v,errors:g,...b.length>0&&{errorSamples:b}});return}let C=b.length>0?`<details style="margin-top:1rem"><summary>First ${b.length} error(s)</summary>
66
- <pre style="white-space:pre-wrap">${b.map(w=>w.replace(/[<>&]/g,T=>`&#${T.charCodeAt(0)};`)).join(`
67
-
68
- `)}</pre></details>`:"",P=W(`Force Sync: ${m.name}`,p,`<div class="card">
69
- <p class="badge ${g>0?"badge-warn":"badge-ok"}">${g>0?"Completed with errors":"Complete"}</p>
70
- <p>Synced <strong>${v}</strong> documents to <code>${m.tableName}</code>.</p>
71
- ${g>0?`<p class="badge badge-warn">${g} error(s)</p>`:""}
72
- ${C}
73
- </div>`);V(d,P);})),l.viewQueue&&h.get(`${i}/queues`,(c,d)=>{let p=pe(c,i),m=[];for(let g of y){let b=n.get(g.name);m.push({repo:g.name,table:g.tableName,pending:b?b.size:0});}if($e(c)){fe(d,{queues:m});return}let f=m.map(g=>`<tr><td>${g.repo}</td><td>${g.table}</td><td>${g.pending===0?'<span class="badge badge-ok">0</span>':`<span class="badge badge-warn">${g.pending}</span>`}</td></tr>`).join(`
74
- `),v=W("Sync Queues",p,`<div class="card">
64
+ <p>Synced ${p} docs before failure (${u} errors).</p>
65
+ </div>`),500);return}if(Fe(h)){xe(b,{repo:y.name,table:y.tableName,synced:p,errors:u,...l.length>0&&{errorSamples:l}});return}let $=l.length>0?`<details style="margin-top:1rem"><summary>First ${l.length} error(s)</summary>
66
+ <pre style="white-space:pre-wrap">${l.map(w=>w.replace(/[<>&]/g,S=>`&#${S.charCodeAt(0)};`)).join(`
67
+
68
+ `)}</pre></details>`:"",O=te(`Force Sync: ${y.name}`,m,`<div class="card">
69
+ <p class="badge ${u>0?"badge-warn":"badge-ok"}">${u>0?"Completed with errors":"Complete"}</p>
70
+ <p>Synced <strong>${p}</strong> documents to <code>${y.tableName}</code>.</p>
71
+ ${u>0?`<p class="badge badge-warn">${u} error(s)</p>`:""}
72
+ ${$}
73
+ </div>`);ne(b,O);})),f.viewQueue&&x.get(`${i}/queues`,(h,b)=>{let m=ve(h,i),y=[];for(let u of g){let l=n.get(u.name);y.push({repo:u.name,table:u.tableName,pending:l?l.size:0});}if(Fe(h)){xe(b,{queues:y});return}let c=y.map(u=>`<tr><td>${u.repo}</td><td>${u.table}</td><td>${u.pending===0?'<span class="badge badge-ok">0</span>':`<span class="badge badge-warn">${u.pending}</span>`}</td></tr>`).join(`
74
+ `),p=te("Sync Queues",m,`<div class="card">
75
75
  <table>
76
76
  <thead><tr><th>Repository</th><th>Table</th><th>Pending</th></tr></thead>
77
- <tbody>${f}</tbody>
77
+ <tbody>${c}</tbody>
78
78
  </table>
79
- </div>`);V(d,v);}),l.configCheck&&h.get(`${i}/config-check`,async(c,d)=>{let p=pe(c,i),m=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??process.env.GCP_PROJECT??"unknown",f="https://console.cloud.google.com",v=u??"firestore-sync",g=[];try{await t.tableExists("__nonexistent_health_check__"),g.push({name:"BigQuery API",category:"bigquery",status:"ok",message:"BigQuery API is reachable"});}catch(w){let T=w?.message??String(w),O=T.toLowerCase(),k=O.includes("disabled")||O.includes("has not been used")||O.includes("accessnotconfigured"),I=O.includes("permission")||T.includes("403")||O.includes("access denied"),$=O.includes("project")&&O.includes("not found"),K=O.includes("not found")||T.includes("404");k?g.push({name:"BigQuery API",category:"bigquery",status:"error",message:"BigQuery API is not enabled",fix:{gcloud:`gcloud services enable bigquery.googleapis.com --project=${m}`,console:`${f}/apis/library/bigquery.googleapis.com?project=${m}`}}):$?g.push({name:"BigQuery Project",category:"bigquery",status:"error",message:T,fix:{hint:"The GCP project does not exist or the credentials don't have access to it. In the Firebase emulator, GCLOUD_PROJECT may override the configured projectId. Ensure you pass the correct projectId to the BigQuery constructor and have valid credentials.",console:`${f}/home/dashboard`}}):I?g.push({name:"BigQuery API",category:"bigquery",status:"error",message:`Permission denied: ${T}`,fix:{hint:"Grant the service account BigQuery Data Editor + BigQuery Job User roles",gcloud:[`SA=$(gcloud run services describe YOUR_SERVICE --region=YOUR_REGION --format="value(spec.template.spec.serviceAccountName)" --project=${m})`,`gcloud projects add-iam-policy-binding ${m} --member="serviceAccount:$SA" --role="roles/bigquery.dataEditor"`,`gcloud projects add-iam-policy-binding ${m} --member="serviceAccount:$SA" --role="roles/bigquery.jobUser"`].join(`
80
- `),console:`${f}/iam-admin/iam?project=${m}`}}):K?g.push({name:"BigQuery Dataset",category:"bigquery",status:"error",message:`Dataset not found: ${T}`,fix:{hint:"Create the dataset first",gcloud:`bq mk --dataset ${m}:YOUR_DATASET_ID`,console:`${f}/bigquery?project=${m}`}}):g.push({name:"BigQuery API",category:"bigquery",status:"ok",message:"BigQuery API is reachable (table lookup returned expected error)"});}for(let w of y)try{let T=await t.tableExists(w.tableName);g.push({name:`Table: ${w.tableName}`,category:"bigquery",status:T?"ok":"warn",message:T?`Table \`${w.tableName}\` exists`:`Table \`${w.tableName}\` does not exist yet`,...!T&&{fix:{hint:"Table will be auto-created on first sync if autoMigrate is enabled. Or create it manually."}}});}catch(T){g.push({name:`Table: ${w.tableName}`,category:"bigquery",status:"error",message:T?.message??String(T)});}if(a)for(let w of y){let T=`${v}-${w.name}`;try{let O=a.topic(T);if(typeof O.exists=="function"){let[k]=await O.exists();g.push({name:`Topic: ${T}`,category:"pubsub",status:k?"ok":"error",message:k?`Topic \`${T}\` exists`:`Topic \`${T}\` does not exist`,...!k&&{fix:{gcloud:`gcloud pubsub topics create ${T} --project=${m}`,console:`${f}/cloudpubsub/topic/list?project=${m}`}}});}else g.push({name:`Topic: ${T}`,category:"pubsub",status:"warn",message:"Cannot verify topic existence (PubSub client doesn't expose .exists())",fix:{gcloud:`gcloud pubsub topics create ${T} --project=${m}`,console:`${f}/cloudpubsub/topic/list?project=${m}`,hint:"Ensure the topic exists. It is auto-created by the Firebase emulator but must exist in production."}});}catch(O){let k=O?.message??String(O),I=k.includes("disabled")||k.includes("has not been used");if(g.push({name:I?"Pub/Sub API":`Topic: ${T}`,category:"pubsub",status:"error",message:I?"Pub/Sub API is not enabled":k,fix:I?{gcloud:`gcloud services enable pubsub.googleapis.com --project=${m}`,console:`${f}/apis/library/pubsub.googleapis.com?project=${m}`}:{gcloud:`gcloud pubsub topics create ${T} --project=${m}`,console:`${f}/cloudpubsub/topic/list?project=${m}`}}),I)break}}else g.push({name:"Pub/Sub Client",category:"pubsub",status:"warn",message:"PubSub client not available for config check"});if($e(c)){let w=g.every(T=>T.status==="ok");fe(d,{project:m,healthy:w,checks:g});return}let b=w=>w==="ok"?'<span class="badge badge-ok">OK</span>':w==="warn"?'<span class="badge badge-warn">WARN</span>':'<span class="badge badge-err">ERROR</span>',R={bigquery:g.filter(w=>w.category==="bigquery"),pubsub:g.filter(w=>w.category==="pubsub"),firestore:g.filter(w=>w.category==="firestore")},x=(w,T)=>{if(T.length===0)return "";let O=T.map(k=>{let I="";if(k.fix){let $=[];k.fix.hint&&$.push(`<p class="muted">${k.fix.hint}</p>`),k.fix.gcloud&&$.push(`<pre>$ ${k.fix.gcloud}</pre>`),k.fix.console&&$.push(`<p><a href="${k.fix.console}" target="_blank">Open GCP Console \u2192</a></p>`),I=`<div style="margin-top:.5rem">${$.join("")}</div>`;}return `<tr>
81
- <td>${b(k.status)}</td>
82
- <td><strong>${k.name}</strong><br><span class="muted">${k.message}</span>${I}</td>
79
+ </div>`);ne(b,p);}),f.configCheck&&x.get(`${i}/config-check`,async(h,b)=>{let m=ve(h,i),y=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??process.env.GCP_PROJECT??"unknown",c="https://console.cloud.google.com",p=d??"firestore-sync",u=[];try{await t.tableExists("__nonexistent_health_check__"),u.push({name:"BigQuery API",category:"bigquery",status:"ok",message:"BigQuery API is reachable"});}catch(w){let S=w?.message??String(w),k=S.toLowerCase(),A=k.includes("disabled")||k.includes("has not been used")||k.includes("accessnotconfigured"),E=k.includes("permission")||S.includes("403")||k.includes("access denied"),C=k.includes("project")&&k.includes("not found"),B=k.includes("not found")||S.includes("404");A?u.push({name:"BigQuery API",category:"bigquery",status:"error",message:"BigQuery API is not enabled",fix:{gcloud:`gcloud services enable bigquery.googleapis.com --project=${y}`,console:`${c}/apis/library/bigquery.googleapis.com?project=${y}`}}):C?u.push({name:"BigQuery Project",category:"bigquery",status:"error",message:S,fix:{hint:"The GCP project does not exist or the credentials don't have access to it. In the Firebase emulator, GCLOUD_PROJECT may override the configured projectId. Ensure you pass the correct projectId to the BigQuery constructor and have valid credentials.",console:`${c}/home/dashboard`}}):E?u.push({name:"BigQuery API",category:"bigquery",status:"error",message:`Permission denied: ${S}`,fix:{hint:"Grant the service account BigQuery Data Editor + BigQuery Job User roles",gcloud:[`SA=$(gcloud run services describe YOUR_SERVICE --region=YOUR_REGION --format="value(spec.template.spec.serviceAccountName)" --project=${y})`,`gcloud projects add-iam-policy-binding ${y} --member="serviceAccount:$SA" --role="roles/bigquery.dataEditor"`,`gcloud projects add-iam-policy-binding ${y} --member="serviceAccount:$SA" --role="roles/bigquery.jobUser"`].join(`
80
+ `),console:`${c}/iam-admin/iam?project=${y}`}}):B?u.push({name:"BigQuery Dataset",category:"bigquery",status:"error",message:`Dataset not found: ${S}`,fix:{hint:"Create the dataset first",gcloud:`bq mk --dataset ${y}:YOUR_DATASET_ID`,console:`${c}/bigquery?project=${y}`}}):u.push({name:"BigQuery API",category:"bigquery",status:"ok",message:"BigQuery API is reachable (table lookup returned expected error)"});}for(let w of g)try{let S=await t.tableExists(w.tableName);u.push({name:`Table: ${w.tableName}`,category:"bigquery",status:S?"ok":"warn",message:S?`Table \`${w.tableName}\` exists`:`Table \`${w.tableName}\` does not exist yet`,...!S&&{fix:{hint:"Table will be auto-created on first sync if autoMigrate is enabled. Or create it manually."}}});}catch(S){u.push({name:`Table: ${w.tableName}`,category:"bigquery",status:"error",message:S?.message??String(S)});}if(a)for(let w of g){let S=`${p}-${w.name}`;try{let k=a.topic(S);if(typeof k.exists=="function"){let[A]=await k.exists();u.push({name:`Topic: ${S}`,category:"pubsub",status:A?"ok":"error",message:A?`Topic \`${S}\` exists`:`Topic \`${S}\` does not exist`,...!A&&{fix:{gcloud:`gcloud pubsub topics create ${S} --project=${y}`,console:`${c}/cloudpubsub/topic/list?project=${y}`}}});}else u.push({name:`Topic: ${S}`,category:"pubsub",status:"warn",message:"Cannot verify topic existence (PubSub client doesn't expose .exists())",fix:{gcloud:`gcloud pubsub topics create ${S} --project=${y}`,console:`${c}/cloudpubsub/topic/list?project=${y}`,hint:"Ensure the topic exists. It is auto-created by the Firebase emulator but must exist in production."}});}catch(k){let A=k?.message??String(k),E=A.includes("disabled")||A.includes("has not been used");if(u.push({name:E?"Pub/Sub API":`Topic: ${S}`,category:"pubsub",status:"error",message:E?"Pub/Sub API is not enabled":A,fix:E?{gcloud:`gcloud services enable pubsub.googleapis.com --project=${y}`,console:`${c}/apis/library/pubsub.googleapis.com?project=${y}`}:{gcloud:`gcloud pubsub topics create ${S} --project=${y}`,console:`${c}/cloudpubsub/topic/list?project=${y}`}}),E)break}}else u.push({name:"Pub/Sub Client",category:"pubsub",status:"warn",message:"PubSub client not available for config check"});if(Fe(h)){let w=u.every(S=>S.status==="ok");xe(b,{project:y,healthy:w,checks:u});return}let l=w=>w==="ok"?'<span class="badge badge-ok">OK</span>':w==="warn"?'<span class="badge badge-warn">WARN</span>':'<span class="badge badge-err">ERROR</span>',v={bigquery:u.filter(w=>w.category==="bigquery"),pubsub:u.filter(w=>w.category==="pubsub"),firestore:u.filter(w=>w.category==="firestore")},R=(w,S)=>{if(S.length===0)return "";let k=S.map(A=>{let E="";if(A.fix){let C=[];A.fix.hint&&C.push(`<p class="muted">${A.fix.hint}</p>`),A.fix.gcloud&&C.push(`<pre>$ ${A.fix.gcloud}</pre>`),A.fix.console&&C.push(`<p><a href="${A.fix.console}" target="_blank">Open GCP Console \u2192</a></p>`),E=`<div style="margin-top:.5rem">${C.join("")}</div>`;}return `<tr>
81
+ <td>${l(A.status)}</td>
82
+ <td><strong>${A.name}</strong><br><span class="muted">${A.message}</span>${E}</td>
83
83
  </tr>`}).join(`
84
84
  `);return `<h2>${w}</h2>
85
85
  <table><thead><tr><th style="width:80px">Status</th><th>Check</th></tr></thead>
86
- <tbody>${O}</tbody></table>`},C=g.every(w=>w.status==="ok")?'<span class="badge badge-ok">All checks passed</span>':'<span class="badge badge-warn">Some issues found</span>',P=W("Config Check",p,`<div class="card">
87
- <p>Project: <code>${m}</code> ${C}</p>
88
- ${x("BigQuery",R.bigquery)}
89
- ${x("Pub/Sub",R.pubsub)}
90
- ${x("Firestore",R.firestore)}
91
- </div>`);V(d,P);}),async(c,d)=>{await h.handle(c,d);}}var Zn="firestore-sync";function Mn(e,t){let n=t.ref?.path??void 0;return n?`${n}/{docId}`:(console.warn(`[SyncTriggers] Cannot determine collection path for "${e}". Skipping.`),null)}function Bt(e,t){let{onDocumentCreated:n,onDocumentUpdated:r,onDocumentDeleted:o}=t.deps.firestoreTriggers,s=t.deps.pubsub,a=t?.topicPrefix??Zn,u={},i=new Map;function l(h){let c=i.get(h);return c||(c=s.topic(h),i.set(h,c),c)}async function y(h,c){await l(h).publishMessage({json:c});}for(let[h,c]of Object.entries(e)){let d=t?.repos?.[h],p;if(c._isGroup){if(!d?.triggerPath){console.warn(`[SyncTriggers] Skipping collection-group repo "${h}". Provide a triggerPath in the sync repos config for group collections.`);continue}p=d.triggerPath;}else p=d?.triggerPath??Mn(h,c);if(!p)continue;let m=c._systemKeys?.[0]??"docId",f=`${a}-${h}`;u[`${h}_onCreate`]=n(p,async v=>{let g=v.data;if(!g)return;let b=g.data();if(!b)return;let R=String(b[m]??g.id),x=Ce(b,{exclude:d?.exclude,columnMap:d?.columnMap}),S={operation:"INSERT",repoName:h,docId:R,data:x,timestamp:new Date().toISOString(),version:Date.now()};await y(f,S);}),u[`${h}_onUpdate`]=r(p,async v=>{let g=v.data?.after;if(!g)return;let b=g.data();if(!b)return;let R=String(b[m]??g.id),x=Ce(b,{exclude:d?.exclude,columnMap:d?.columnMap}),S={operation:"UPSERT",repoName:h,docId:R,data:x,timestamp:new Date().toISOString(),version:Date.now()};await y(f,S);}),u[`${h}_onDelete`]=o(p,async v=>{let g=v.data;if(!g)return;let b=g.data(),R=String(b?.[m]??g.id),x={operation:"DELETE",repoName:h,docId:R,data:null,timestamp:new Date().toISOString(),version:Date.now()};await y(f,x);});}return u}var ze=class{constructor(t){this.buffer=[];this.flushing=false;this.timer=null;this.adapter=t.adapter,this.tableName=t.tableName,this.primaryKey=t.primaryKey,this.batchSize=t.batchSize??100,this.onFlushError=t.onFlushError;let n=t.flushIntervalMs??5e3;n>0&&(this.timer=setInterval(()=>{this.flush();},n),typeof this.timer=="object"&&"unref"in this.timer&&this.timer.unref());}get size(){return this.buffer.length}enqueue(...t){this.buffer.push(...t),this.buffer.length>=this.batchSize&&this.flush();}async flush(){if(this.flushing||this.buffer.length===0)return;this.flushing=true;let t=this.buffer.splice(0,this.batchSize);try{let n=new Map,r=[];for(let s of t)if(s.operation==="DELETE")r.push(s.docId),n.delete(s.docId);else if(s.data){let a=n.get(s.docId);if(!a)n.set(s.docId,s.data);else {let u=Number(a[oe]??0);Number(s.data[oe]??0)>=u&&n.set(s.docId,s.data);}}let o=Array.from(n.values());o.length>0&&await this.adapter.upsertRows(this.tableName,o,this.primaryKey),r.length>0&&await this.adapter.deleteRows(this.tableName,this.primaryKey,r);}catch(n){this.onFlushError?await this.onFlushError(t,n).catch(r=>{console.error(`[SyncQueue] Flush error for ${this.tableName}:`,n),console.error("[SyncQueue] Error handler also failed:",r);}):(this.buffer.unshift(...t),console.error(`[SyncQueue] Flush failed for ${this.tableName}:`,n));}finally{this.flushing=false;}}async shutdown(){this.timer&&(clearInterval(this.timer),this.timer=null),await this.flush();}};var Kt=new Set;async function Bn(e,t,n,r,o,s,a){if(Kt.has(e))return;let u=Fe(n,t.dialect,{primaryKey:o,exclude:s,columnMap:a});if(!await t.tableExists(r))await t.createTable({tableName:r,columns:u});else {let l=new Set(await t.getTableColumns(r)),y=u.filter(h=>!l.has(h.name));y.length>0&&await t.addColumns(r,y);}Kt.add(e);}function qt(e,t){let{deps:n,adapter:r,batchSize:o=100,flushIntervalMs:s=5e3,autoMigrate:a=false,topicPrefix:u="firestore-sync",workerOptions:i,repos:l={}}=t,y=new Map;function h(p,m){let f=y.get(p);if(f)return f;let g=l[p]?.tableName??p,b=async(R,x)=>{console.error(`[SyncWorker] Flush failed for "${p}" (${R.length} events):`,x);try{let S=`${u}-${p}-dlq`,C=n.pubsub.topic(S),[P]=await C.exists();P||(await C.create(),console.info(`[SyncWorker] Created DLQ topic "${S}"`));for(let w of R)await C.publishMessage({json:w});}catch(S){console.error(`[SyncWorker] Dead-letter publish also failed for ${p}:`,S);}};return f=new ze({adapter:r,tableName:g,primaryKey:m,batchSize:o,flushIntervalMs:s,onFlushError:b}),y.set(p,f),f}async function c(p){let{repoName:m}=p,f=e[m];if(!f){console.warn(`[SyncWorker] Unknown repo "${m}", skipping event`);return}let v=f._systemKeys?.[0]??f.documentKey??"docId",g=l[m],b=g?.columnMap,R=b?.[v]??v;if(a){let S=f.schema??void 0;if(S){let C=g?.tableName??m;await Bn(m,r,S,C,v,g?.exclude,b);}}let x=h(m,R);p.data&&(p.data[oe]=p.version??Date.now()),x.enqueue(p);}function d(p){let m=async f=>{let v=f.data?.message?.json??f.data?.json;if(!v){console.warn("[SyncWorker] Received empty PubSub message");return}await c(v);let g=y.get(v.repoName);g&&await g.flush();};return i?n.pubsubHandler.onMessagePublished({topic:p,...i},m):n.pubsubHandler.onMessagePublished(p,m)}return {handleMessage:c,createHandler:d,queues:y,async shutdown(){let p=[];for(let m of y.values())p.push(m.shutdown());await Promise.all(p);}}}var Kn="firestore-sync";function Qt(e){if(typeof e!="function")return e;let t=e,n;return new Proxy({},{get(r,o){return n||(n=t()),n[o]},has(r,o){return n||(n=t()),o in n}})}function Ut(e,t){let{deps:n,adapter:r,topicPrefix:o=Kn,batchSize:s,flushIntervalMs:a,autoMigrate:u,admin:i,workerOptions:l,repos:y}=t,h=Qt(n.pubsub),c=Qt(r),d=Bt(e,{deps:{firestoreTriggers:n.firestoreTriggers,pubsub:h},topicPrefix:o,repos:y}),p=qt(e,{deps:{pubsubHandler:n.pubsubHandler,pubsub:h},adapter:c,batchSize:s,flushIntervalMs:a,autoMigrate:u,topicPrefix:o,workerOptions:l,repos:y}),m={};for(let g of Object.keys(e))m[`sync_${g}`]=p.createHandler(`${o}-${g}`);let f=null;i&&(f=Mt(e,c,p.queues,p.handleMessage,i,y??{},h,o),m.adminsync=i.onRequest?i.httpsOptions?i.onRequest(i.httpsOptions,f):i.onRequest(f):f);let v={functions:{...d,...m},adminHandler:f,handleMessage:p.handleMessage,queues:p.queues,shutdown:p.shutdown};for(let g of ["adminHandler","handleMessage","queues","shutdown"])Object.defineProperty(v,g,{enumerable:false});return v}function qn(e){return e.replace(/([A-Z])/g," $1").replace(/_/g," ").replace(/^\s/,"").replace(/^./,t=>t.toUpperCase())}function Lt(e){let t=e,n=true,r=false,o;for(;;){let s=j(t);if(s==="ZodOptional")n=false,t=q(t);else if(s==="ZodNullable")n=false,r=true,t=q(t);else if(s==="ZodDefault")n=false,o=_t(t),t=q(t);else break}return {inner:t,required:n,nullable:r,defaultValue:o}}function J(e,t=""){if(j(e)==="ZodObject"){let r=re(e);return Object.entries(r).map(([o,s])=>Gt(t?`${t}.${o}`:o,o,s))}return [Gt(t||"value",t||"value",e)]}function Gt(e,t,n){let{inner:r,required:o,nullable:s,defaultValue:a}=Lt(n),u=j(r),i=qn(t.split(".").pop()??t);switch(u){case "ZodString":{let l=jt(r),y=l.includes("email"),h=l.includes("url");return {name:e,label:i,type:"text",required:o,nullable:s,defaultValue:a,hint:y?"email":h?"url":void 0}}case "ZodNumber":case "ZodBigInt":return {name:e,label:i,type:"number",required:o,nullable:s,defaultValue:a};case "ZodBoolean":return {name:e,label:i,type:"checkbox",required:o,nullable:s,defaultValue:a};case "ZodDate":case "ZodCoerce":return {name:e,label:i,type:"datetime-local",required:o,nullable:s,defaultValue:a};case "ZodEnum":{let l=Se(r);return {name:e,label:i,type:"select",required:o,nullable:s,defaultValue:a,options:l}}case "ZodNativeEnum":{let l=Te(r),y=Object.values(l).filter(h=>typeof h=="string");return {name:e,label:i,type:"select",required:o,nullable:s,defaultValue:a,options:y}}case "ZodLiteral":{let l=String(je(r)??"");return {name:e,label:i,type:"select",required:o,nullable:s,defaultValue:a,options:[l]}}case "ZodObject":{let l=J(r,e);return {name:e,label:i,type:"textarea",required:o,nullable:s,defaultValue:a,nested:l,hint:"JSON object"}}case "ZodArray":{let l=Nt(r);if(!l)return {name:e,label:i,type:"textarea",required:o,nullable:s,defaultValue:a,hint:"JSON array"};let{inner:y}=Lt(l),h=j(y),c,d,p;switch(h){case "ZodString":c="text";break;case "ZodNumber":case "ZodBigInt":c="number";break;case "ZodBoolean":c="checkbox";break;case "ZodDate":c="datetime-local";break;case "ZodEnum":c="select",d=Se(y);break;case "ZodNativeEnum":c="select",d=Object.values(Te(y)).filter(m=>typeof m=="string");break;case "ZodObject":c="object",p=J(y);break;default:return {name:e,label:i,type:"textarea",required:o,nullable:s,defaultValue:a,hint:"JSON array"}}return {name:e,label:i,type:"textarea",required:o,nullable:s,defaultValue:a,arrayElementType:c,arrayElementOptions:d,arrayElementFields:p}}default:return {name:e,label:i,type:"textarea",required:o,nullable:s,defaultValue:a,hint:"JSON"}}}function Ve(e,t=0){let n=t>0?`ml-${t*4}`:"",r=`field_${e.name.replace(/\./g,"__")}`,o=e.name,s=e.required?" required":"",a=e.defaultValue==="__null__",u=!a&&e.defaultValue!=null?String(e.defaultValue):"",i=e.nullable&&e.type!=="checkbox"?`<span class="flex items-center gap-1 shrink-0">
92
- <input type="hidden" id="${r}__isnull" name="${o}__isnull" value="${a?"1":""}">
86
+ <tbody>${k}</tbody></table>`},$=u.every(w=>w.status==="ok")?'<span class="badge badge-ok">All checks passed</span>':'<span class="badge badge-warn">Some issues found</span>',O=te("Config Check",m,`<div class="card">
87
+ <p>Project: <code>${y}</code> ${$}</p>
88
+ ${R("BigQuery",v.bigquery)}
89
+ ${R("Pub/Sub",v.pubsub)}
90
+ ${R("Firestore",v.firestore)}
91
+ </div>`);ne(b,O);}),async(h,b)=>{await x.handle(h,b);}}var ur="firestore-sync";function pr(e,t){let n=t.ref?.path??void 0;return n?`${n}/{docId}`:(console.warn(`[SyncTriggers] Cannot determine collection path for "${e}". Skipping.`),null)}function ln(e,t){let{onDocumentCreated:n,onDocumentUpdated:r,onDocumentDeleted:s}=t.deps.firestoreTriggers,o=t.deps.pubsub,a=t?.topicPrefix??ur,d={},i=new Map;function f(x){let h=i.get(x);return h||(h=o.topic(x),i.set(x,h),h)}async function g(x,h){await f(x).publishMessage({json:h});}for(let[x,h]of Object.entries(e)){let b=t?.repos?.[x],m;if(h._isGroup){if(!b?.triggerPath){console.warn(`[SyncTriggers] Skipping collection-group repo "${x}". Provide a triggerPath in the sync repos config for group collections.`);continue}m=b.triggerPath;}else m=b?.triggerPath??pr(x,h);if(!m)continue;let y=h._systemKeys?.[0]??"docId",c=`${a}-${x}`;d[`${x}_onCreate`]=n(m,async p=>{let u=p.data;if(!u)return;let l=u.data();if(!l)return;let v=String(l[y]??u.id),R=ze(l,{exclude:b?.exclude,columnMap:b?.columnMap}),T={operation:"INSERT",repoName:x,docId:v,data:R,timestamp:new Date().toISOString(),version:Date.now()};await g(c,T);}),d[`${x}_onUpdate`]=r(m,async p=>{let u=p.data?.after;if(!u)return;let l=u.data();if(!l)return;let v=String(l[y]??u.id),R=ze(l,{exclude:b?.exclude,columnMap:b?.columnMap}),T={operation:"UPSERT",repoName:x,docId:v,data:R,timestamp:new Date().toISOString(),version:Date.now()};await g(c,T);}),d[`${x}_onDelete`]=s(m,async p=>{let u=p.data;if(!u)return;let l=u.data(),v=String(l?.[y]??u.id),R={operation:"DELETE",repoName:x,docId:v,data:null,timestamp:new Date().toISOString(),version:Date.now()};await g(c,R);});}return d}var Ye=class{constructor(t){this.buffer=[];this.flushing=false;this.timer=null;this.adapter=t.adapter,this.tableName=t.tableName,this.primaryKey=t.primaryKey,this.batchSize=t.batchSize??100,this.onFlushError=t.onFlushError;let n=t.flushIntervalMs??5e3;n>0&&(this.timer=setInterval(()=>{this.flush();},n),typeof this.timer=="object"&&"unref"in this.timer&&this.timer.unref());}get size(){return this.buffer.length}enqueue(...t){this.buffer.push(...t),this.buffer.length>=this.batchSize&&this.flush();}async flush(){if(this.flushing||this.buffer.length===0)return;this.flushing=true;let t=this.buffer.splice(0,this.batchSize);try{let n=new Map,r=[];for(let o of t)if(o.operation==="DELETE")r.push(o.docId),n.delete(o.docId);else if(o.data){let a=n.get(o.docId);if(!a)n.set(o.docId,o.data);else {let d=Number(a[ce]??0);Number(o.data[ce]??0)>=d&&n.set(o.docId,o.data);}}let s=Array.from(n.values());s.length>0&&await this.adapter.upsertRows(this.tableName,s,this.primaryKey),r.length>0&&await this.adapter.deleteRows(this.tableName,this.primaryKey,r);}catch(n){this.onFlushError?await this.onFlushError(t,n).catch(r=>{console.error(`[SyncQueue] Flush error for ${this.tableName}:`,n),console.error("[SyncQueue] Error handler also failed:",r);}):(this.buffer.unshift(...t),console.error(`[SyncQueue] Flush failed for ${this.tableName}:`,n));}finally{this.flushing=false;}}async shutdown(){this.timer&&(clearInterval(this.timer),this.timer=null),await this.flush();}};var cn=new Set;async function fr(e,t,n,r,s,o,a){if(cn.has(e))return;let d=Xe(n,t.dialect,{primaryKey:s,exclude:o,columnMap:a});if(!await t.tableExists(r))await t.createTable({tableName:r,columns:d});else {let f=new Set(await t.getTableColumns(r)),g=d.filter(x=>!f.has(x.name));g.length>0&&await t.addColumns(r,g);}cn.add(e);}function dn(e,t){let{deps:n,adapter:r,batchSize:s=100,flushIntervalMs:o=5e3,autoMigrate:a=false,topicPrefix:d="firestore-sync",workerOptions:i,repos:f={}}=t,g=new Map;function x(m,y){let c=g.get(m);if(c)return c;let u=f[m]?.tableName??m,l=async(v,R)=>{console.error(`[SyncWorker] Flush failed for "${m}" (${v.length} events):`,R);try{let T=`${d}-${m}-dlq`,$=n.pubsub.topic(T),[O]=await $.exists();O||(await $.create(),console.info(`[SyncWorker] Created DLQ topic "${T}"`));for(let w of v)await $.publishMessage({json:w});}catch(T){console.error(`[SyncWorker] Dead-letter publish also failed for ${m}:`,T);}};return c=new Ye({adapter:r,tableName:u,primaryKey:y,batchSize:s,flushIntervalMs:o,onFlushError:l}),g.set(m,c),c}async function h(m){let{repoName:y}=m,c=e[y];if(!c){console.warn(`[SyncWorker] Unknown repo "${y}", skipping event`);return}let p=c._systemKeys?.[0]??c.documentKey??"docId",u=f[y],l=u?.columnMap,v=l?.[p]??p;if(a){let T=c.schema??void 0;if(T){let $=u?.tableName??y;await fr(y,r,T,$,p,u?.exclude,l);}}let R=x(y,v);m.data&&(m.data[ce]=m.version??Date.now()),R.enqueue(m);}function b(m){let y=async c=>{let p=c.data?.message?.json??c.data?.json;if(!p){console.warn("[SyncWorker] Received empty PubSub message");return}await h(p);let u=g.get(p.repoName);u&&await u.flush();};return i?n.pubsubHandler.onMessagePublished({topic:m,...i},y):n.pubsubHandler.onMessagePublished(m,y)}return {handleMessage:h,createHandler:b,queues:g,async shutdown(){let m=[];for(let y of g.values())m.push(y.shutdown());await Promise.all(m);}}}var mr="firestore-sync";function un(e){if(typeof e!="function")return e;let t=e,n;return new Proxy({},{get(r,s){return n||(n=t()),n[s]},has(r,s){return n||(n=t()),s in n}})}function pn(e,t){let{deps:n,adapter:r,topicPrefix:s=mr,batchSize:o,flushIntervalMs:a,autoMigrate:d,admin:i,workerOptions:f,repos:g}=t,x=un(n.pubsub),h=un(r),b=ln(e,{deps:{firestoreTriggers:n.firestoreTriggers,pubsub:x},topicPrefix:s,repos:g}),m=dn(e,{deps:{pubsubHandler:n.pubsubHandler,pubsub:x},adapter:h,batchSize:o,flushIntervalMs:a,autoMigrate:d,topicPrefix:s,workerOptions:f,repos:g}),y={};for(let u of Object.keys(e))y[`sync_${u}`]=m.createHandler(`${s}-${u}`);let c=null;i&&(c=an(e,h,m.queues,m.handleMessage,i,g??{},x,s),y.adminsync=i.onRequest?i.httpsOptions?i.onRequest(i.httpsOptions,c):i.onRequest(c):c);let p={functions:{...b,...y},adminHandler:c,handleMessage:m.handleMessage,queues:m.queues,shutdown:m.shutdown};for(let u of ["adminHandler","handleMessage","queues","shutdown"])Object.defineProperty(p,u,{enumerable:false});return p}je();je();function yr(e){return e.replace(/([A-Z])/g," $1").replace(/_/g," ").replace(/^\s/,"").replace(/^./,t=>t.toUpperCase())}function fn(e){let t=e,n=true,r=false,s;for(;;){let o=M(t);if(o==="ZodOptional")n=false,t=U(t);else if(o==="ZodNullable")n=false,r=true,t=U(t);else if(o==="ZodDefault")n=false,s=tn(t),t=U(t);else break}return {inner:t,required:n,nullable:r,defaultValue:s}}function re(e,t=""){if(M(e)==="ZodObject"){let r=J(e);return Object.entries(r).map(([s,o])=>mn(t?`${t}.${s}`:s,s,o))}return [mn(t||"value",t||"value",e)]}function mn(e,t,n){let{inner:r,required:s,nullable:o,defaultValue:a}=fn(n),d=M(r),i=yr(t.split(".").pop()??t);switch(d){case "ZodString":{let f=nn(r),g=f.includes("email"),x=f.includes("url");return {name:e,label:i,type:"text",required:s,nullable:o,defaultValue:a,hint:g?"email":x?"url":void 0}}case "ZodNumber":case "ZodBigInt":return {name:e,label:i,type:"number",required:s,nullable:o,defaultValue:a};case "ZodBoolean":return {name:e,label:i,type:"checkbox",required:s,nullable:o,defaultValue:a};case "ZodDate":case "ZodCoerce":return {name:e,label:i,type:"datetime-local",required:s,nullable:o,defaultValue:a};case "ZodEnum":{let f=Ne(r);return {name:e,label:i,type:"select",required:s,nullable:o,defaultValue:a,options:f}}case "ZodNativeEnum":{let f=_e(r),g=Object.values(f).filter(x=>typeof x=="string");return {name:e,label:i,type:"select",required:s,nullable:o,defaultValue:a,options:g}}case "ZodLiteral":{let f=String(Je(r)??"");return {name:e,label:i,type:"select",required:s,nullable:o,defaultValue:a,options:[f]}}case "ZodObject":{let f=re(r,e);return {name:e,label:i,type:"textarea",required:s,nullable:o,defaultValue:a,nested:f,hint:"JSON object"}}case "ZodArray":{let f=en(r);if(!f)return {name:e,label:i,type:"textarea",required:s,nullable:o,defaultValue:a,hint:"JSON array"};let{inner:g}=fn(f),x=M(g),h,b,m;switch(x){case "ZodString":h="text";break;case "ZodNumber":case "ZodBigInt":h="number";break;case "ZodBoolean":h="checkbox";break;case "ZodDate":h="datetime-local";break;case "ZodEnum":h="select",b=Ne(g);break;case "ZodNativeEnum":h="select",b=Object.values(_e(g)).filter(y=>typeof y=="string");break;case "ZodObject":h="object",m=re(g);break;default:return {name:e,label:i,type:"textarea",required:s,nullable:o,defaultValue:a,hint:"JSON array"}}return {name:e,label:i,type:"textarea",required:s,nullable:o,defaultValue:a,arrayElementType:h,arrayElementOptions:b,arrayElementFields:m}}default:return {name:e,label:i,type:"textarea",required:s,nullable:o,defaultValue:a,hint:"JSON"}}}function pt(e,t=0){let n=t>0?`ml-${t*4}`:"",r=`field_${e.name.replace(/\./g,"__")}`,s=e.name,o=e.required?" required":"",a=e.defaultValue==="__null__",d=!a&&e.defaultValue!=null?String(e.defaultValue):"",i=e.nullable&&e.type!=="checkbox"?`<span class="flex items-center gap-1 shrink-0">
92
+ <input type="hidden" id="${r}__isnull" name="${s}__isnull" value="${a?"1":""}">
93
93
  <label class="flex items-center gap-1 cursor-pointer select-none text-xs text-base-content/40 hover:text-base-content/70 border border-base-300 rounded px-2 py-1">
94
94
  <input type="checkbox" class="checkbox checkbox-xs" ${a?"checked":""}
95
95
  onchange="(function(cb){
@@ -100,61 +100,61 @@ ${n}
100
100
  })(this)">
101
101
  <span>null</span>
102
102
  </label>
103
- </span>`:"",l;switch(e.type){case "checkbox":if(e.nullable){let y=a?"__null__":u==="true"?"true":u==="false"?"false":"__null__";return `
103
+ </span>`:"",f;switch(e.type){case "checkbox":if(e.nullable){let g=a?"__null__":d==="true"?"true":d==="false"?"false":"__null__";return `
104
104
  <div class="form-control mb-3 ${n}">
105
105
  <label for="${r}" class="label pb-1">
106
106
  <span class="label-text font-medium">
107
- ${A(e.label)}
107
+ ${I(e.label)}
108
108
  <span class="text-base-content/40 text-xs ml-1">(nullable)</span>
109
109
  </span>
110
110
  </label>
111
- <select id="${r}" name="${o}" class="select select-bordered select-sm w-full">
112
- <option value="__null__"${y==="__null__"?" selected":""}>\u2014 null \u2014</option>
113
- <option value="true"${y==="true"?" selected":""}>\u2713 true</option>
114
- <option value="false"${y==="false"?" selected":""}>\u2717 false</option>
111
+ <select id="${r}" name="${s}" class="select select-bordered select-sm w-full">
112
+ <option value="__null__"${g==="__null__"?" selected":""}>\u2014 null \u2014</option>
113
+ <option value="true"${g==="true"?" selected":""}>\u2713 true</option>
114
+ <option value="false"${g==="false"?" selected":""}>\u2717 false</option>
115
115
  </select>
116
116
  </div>`}return `
117
117
  <div class="form-control ${n}">
118
118
  <label class="label cursor-pointer justify-start gap-3">
119
- <input type="checkbox" id="${r}" name="${o}" value="true"${u==="true"?" checked":""} class="checkbox checkbox-primary checkbox-sm">
119
+ <input type="checkbox" id="${r}" name="${s}" value="true"${d==="true"?" checked":""} class="checkbox checkbox-primary checkbox-sm">
120
120
  <span class="label-text font-medium">
121
- ${A(e.label)}${e.required?' <span class="text-error">*</span>':""}
121
+ ${I(e.label)}${e.required?' <span class="text-error">*</span>':""}
122
122
  </span>
123
123
  </label>
124
- </div>`;case "select":l=`<select id="${r}" name="${o}"${s}${a?' disabled style="opacity:0.35"':""} class="select select-bordered select-sm w-full">
124
+ </div>`;case "select":f=`<select id="${r}" name="${s}"${o}${a?' disabled style="opacity:0.35"':""} class="select select-bordered select-sm w-full">
125
125
  ${e.required&&!e.nullable?"":'<option value="">\u2014 optional \u2014</option>'}
126
- ${(e.options??[]).map(y=>`<option value="${A(y)}"${u===y?" selected":""}>${A(y)}</option>`).join(`
126
+ ${(e.options??[]).map(g=>`<option value="${I(g)}"${d===g?" selected":""}>${I(g)}</option>`).join(`
127
127
  `)}
128
- </select>`;break;case "textarea":if(e.arrayElementType)return Qn(e,t);if(e.nested&&e.nested.length>0){let y=e.nested.map(h=>Ve(h,t+1)).join(`
128
+ </select>`;break;case "textarea":if(e.arrayElementType)return gr(e,t);if(e.nested&&e.nested.length>0){let g=e.nested.map(x=>pt(x,t+1)).join(`
129
129
  `);return `
130
130
  <fieldset class="fieldset border border-base-300 rounded-box p-3 mb-3 ${n}">
131
131
  <legend class="fieldset-legend text-xs font-semibold text-base-content/60 px-1">
132
- ${A(e.label)}${e.required?' <span class="text-error">*</span>':""}
132
+ ${I(e.label)}${e.required?' <span class="text-error">*</span>':""}
133
133
  </legend>
134
- ${y}
135
- </fieldset>`}l=`<textarea id="${r}" name="${o}"${s} rows="3"${a?' disabled style="opacity:0.35"':""}
134
+ ${g}
135
+ </fieldset>`}f=`<textarea id="${r}" name="${s}"${o} rows="3"${a?' disabled style="opacity:0.35"':""}
136
136
  data-json
137
137
  class="textarea textarea-bordered textarea-sm w-full font-mono text-xs"
138
- placeholder="${A(e.hint??"JSON")}">${A(u)}</textarea>`;break;default:l=`<input type="${e.type}" id="${r}" name="${o}"${s}${a?' disabled style="opacity:0.35"':""}
139
- value="${A(u)}"
138
+ placeholder="${I(e.hint??"JSON")}">${I(d)}</textarea>`;break;default:f=`<input type="${e.type}" id="${r}" name="${s}"${o}${a?' disabled style="opacity:0.35"':""}
139
+ value="${I(d)}"
140
140
  class="input input-bordered input-sm w-full"${e.hint==="email"?' autocomplete="email"':e.hint==="url"?' autocomplete="url"':""}>`;}return `
141
141
  <div class="form-control mb-3 ${n}">
142
142
  <label for="${r}" class="label pb-1">
143
143
  <span class="label-text font-medium">
144
- ${A(e.label)}${e.required?' <span class="text-error">*</span>':""}
145
- ${e.hint?`<span class="text-base-content/40 text-xs ml-1">(${A(e.hint)})</span>`:""}
144
+ ${I(e.label)}${e.required?' <span class="text-error">*</span>':""}
145
+ ${e.hint?`<span class="text-base-content/40 text-xs ml-1">(${I(e.hint)})</span>`:""}
146
146
  </span>
147
147
  </label>
148
148
  <div class="flex items-center gap-2">
149
- <div class="flex-1 min-w-0">${l}</div>
149
+ <div class="flex-1 min-w-0">${f}</div>
150
150
  ${i}
151
151
  </div>
152
- </div>`}function A(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#39;")}function Qn(e,t){let n=t>0?`ml-${t*4}`:"",r=`field_${e.name.replace(/\./g,"__")}`,o=e.defaultValue==="__null__",s=e.arrayElementType==="object",a=[];if(e.defaultValue!=null&&e.defaultValue!==""&&e.defaultValue!=="__null__")try{a=JSON.parse(String(e.defaultValue));}catch{}Array.isArray(a)||(a=[]);let u=s?a.map(y=>Wt(e,y??{})).join(`
153
- `):a.map(y=>Ht(e,y)).join(`
154
- `),i=s?Wt(e,{}):Ht(e,""),l=e.nullable?`<span class="flex items-center gap-1 mt-2">
155
- <input type="hidden" id="${r}__isnull" name="${A(e.name)}__isnull" value="${o?"1":""}">
152
+ </div>`}function I(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#39;")}function gr(e,t){let n=t>0?`ml-${t*4}`:"",r=`field_${e.name.replace(/\./g,"__")}`,s=e.defaultValue==="__null__",o=e.arrayElementType==="object",a=[];if(e.defaultValue!=null&&e.defaultValue!==""&&e.defaultValue!=="__null__")try{a=JSON.parse(String(e.defaultValue));}catch{}Array.isArray(a)||(a=[]);let d=o?a.map(g=>gn(e,g??{})).join(`
153
+ `):a.map(g=>yn(e,g)).join(`
154
+ `),i=o?gn(e,{}):yn(e,""),f=e.nullable?`<span class="flex items-center gap-1 mt-2">
155
+ <input type="hidden" id="${r}__isnull" name="${I(e.name)}__isnull" value="${s?"1":""}">
156
156
  <label class="flex items-center gap-1 cursor-pointer select-none text-xs text-base-content/40 hover:text-base-content/70 border border-base-300 rounded px-2 py-1">
157
- <input type="checkbox" class="checkbox checkbox-xs" ${o?"checked":""}
157
+ <input type="checkbox" class="checkbox checkbox-xs" ${s?"checked":""}
158
158
  onchange="(function(cb){
159
159
  var fs = cb.closest('[data-frs-array]');
160
160
  var h = document.getElementById('${r}__isnull');
@@ -166,64 +166,536 @@ ${n}
166
166
  </label>
167
167
  </span>`:"";return `
168
168
  <fieldset class="fieldset border border-base-300 rounded-box p-3 mb-3 ${n}"
169
- data-frs-array="${A(e.name)}" data-frs-array-type="${A(e.arrayElementType??"text")}">
169
+ data-frs-array="${I(e.name)}" data-frs-array-type="${I(e.arrayElementType??"text")}">
170
170
  <legend class="fieldset-legend text-xs font-semibold text-base-content/60 px-1">
171
- ${A(e.label)}${e.required?' <span class="text-error">*</span>':""}
171
+ ${I(e.label)}${e.required?' <span class="text-error">*</span>':""}
172
172
  </legend>
173
- <input type="hidden" id="${r}" name="${A(e.name)}" value="${A(JSON.stringify(a))}"${o?" disabled":""}>
174
- <div data-frs-array-items${o?' style="opacity:0.35"':""}>
175
- ${u}
173
+ <input type="hidden" id="${r}" name="${I(e.name)}" value="${I(JSON.stringify(a))}"${s?" disabled":""}>
174
+ <div data-frs-array-items${s?' style="opacity:0.35"':""}>
175
+ ${d}
176
176
  </div>
177
177
  <template data-frs-array-tpl>${i}</template>
178
178
  <button type="button" class="btn btn-xs btn-outline mt-1" data-frs-array-add>+ Add</button>
179
- ${l}
180
- </fieldset>`}function Ht(e,t){let n=t!=null?String(t):"",r;switch(e.arrayElementType){case "select":r=`<select data-frs-val class="select select-bordered select-sm flex-1">
179
+ ${f}
180
+ </fieldset>`}function yn(e,t){let n=t!=null?String(t):"",r;switch(e.arrayElementType){case "select":r=`<select data-frs-val class="select select-bordered select-sm flex-1">
181
181
  <option value="">\u2014</option>
182
- ${(e.arrayElementOptions??[]).map(o=>`<option value="${A(o)}"${n===o?" selected":""}>${A(o)}</option>`).join("")}
182
+ ${(e.arrayElementOptions??[]).map(s=>`<option value="${I(s)}"${n===s?" selected":""}>${I(s)}</option>`).join("")}
183
183
  </select>`;break;case "checkbox":r=`<label class="flex items-center gap-2 flex-1 cursor-pointer">
184
184
  <input type="checkbox" data-frs-val class="checkbox checkbox-sm checkbox-primary"${n==="true"?" checked":""}>
185
185
  <span class="text-sm">true</span>
186
- </label>`;break;case "number":r=`<input type="number" data-frs-val value="${A(n)}" class="input input-bordered input-sm flex-1">`;break;case "datetime-local":r=`<input type="datetime-local" data-frs-val value="${A(n)}" class="input input-bordered input-sm flex-1">`;break;default:r=`<input type="text" data-frs-val value="${A(n)}" class="input input-bordered input-sm flex-1">`;}return `<div class="flex items-center gap-2 mb-2" data-frs-array-item>
186
+ </label>`;break;case "number":r=`<input type="number" data-frs-val value="${I(n)}" class="input input-bordered input-sm flex-1">`;break;case "datetime-local":r=`<input type="datetime-local" data-frs-val value="${I(n)}" class="input input-bordered input-sm flex-1">`;break;default:r=`<input type="text" data-frs-val value="${I(n)}" class="input input-bordered input-sm flex-1">`;}return `<div class="flex items-center gap-2 mb-2" data-frs-array-item>
187
187
  ${r}
188
188
  <button type="button" class="btn btn-xs btn-ghost text-error" data-frs-array-rm>&times;</button>
189
- </div>`}function Wt(e,t){return `<div class="border border-base-200 rounded p-3 mb-2" data-frs-array-item>
189
+ </div>`}function gn(e,t){return `<div class="border border-base-200 rounded p-3 mb-2" data-frs-array-item>
190
190
  <div class="flex justify-end mb-1">
191
191
  <button type="button" class="btn btn-xs btn-ghost text-error" data-frs-array-rm>&times;</button>
192
192
  </div>
193
- ${(e.arrayElementFields??[]).map(o=>{let s=t[o.name],a=s==null?"":typeof s=="object"?JSON.stringify(s):String(s);switch(o.type){case "checkbox":return `<div class="form-control mb-2">
193
+ ${(e.arrayElementFields??[]).map(s=>{let o=t[s.name],a=o==null?"":typeof o=="object"?JSON.stringify(o):String(o);switch(s.type){case "checkbox":return `<div class="form-control mb-2">
194
194
  <label class="label cursor-pointer justify-start gap-3">
195
- <input type="checkbox" data-frs-key="${A(o.name)}" class="checkbox checkbox-sm checkbox-primary"${a==="true"?" checked":""}>
196
- <span class="label-text text-sm">${A(o.label)}</span>
195
+ <input type="checkbox" data-frs-key="${I(s.name)}" class="checkbox checkbox-sm checkbox-primary"${a==="true"?" checked":""}>
196
+ <span class="label-text text-sm">${I(s.label)}</span>
197
197
  </label>
198
198
  </div>`;case "select":return `<div class="form-control mb-2">
199
- <label class="label pb-1"><span class="label-text text-sm">${A(o.label)}</span></label>
200
- <select data-frs-key="${A(o.name)}" class="select select-bordered select-sm w-full">
201
- ${o.required?"":'<option value="">\u2014</option>'}
202
- ${(o.options??[]).map(u=>`<option value="${A(u)}"${a===u?" selected":""}>${A(u)}</option>`).join("")}
199
+ <label class="label pb-1"><span class="label-text text-sm">${I(s.label)}</span></label>
200
+ <select data-frs-key="${I(s.name)}" class="select select-bordered select-sm w-full">
201
+ ${s.required?"":'<option value="">\u2014</option>'}
202
+ ${(s.options??[]).map(d=>`<option value="${I(d)}"${a===d?" selected":""}>${I(d)}</option>`).join("")}
203
203
  </select>
204
204
  </div>`;case "number":return `<div class="form-control mb-2">
205
- <label class="label pb-1"><span class="label-text text-sm">${A(o.label)}</span></label>
206
- <input type="number" data-frs-key="${A(o.name)}" value="${A(a)}" class="input input-bordered input-sm w-full">
205
+ <label class="label pb-1"><span class="label-text text-sm">${I(s.label)}</span></label>
206
+ <input type="number" data-frs-key="${I(s.name)}" value="${I(a)}" class="input input-bordered input-sm w-full">
207
207
  </div>`;case "datetime-local":return `<div class="form-control mb-2">
208
- <label class="label pb-1"><span class="label-text text-sm">${A(o.label)}</span></label>
209
- <input type="datetime-local" data-frs-key="${A(o.name)}" value="${A(a)}" class="input input-bordered input-sm w-full">
208
+ <label class="label pb-1"><span class="label-text text-sm">${I(s.label)}</span></label>
209
+ <input type="datetime-local" data-frs-key="${I(s.name)}" value="${I(a)}" class="input input-bordered input-sm w-full">
210
210
  </div>`;case "textarea":return `<div class="form-control mb-2">
211
- <label class="label pb-1"><span class="label-text text-sm">${A(o.label)}</span></label>
212
- <textarea data-frs-key="${A(o.name)}" rows="2" class="textarea textarea-bordered textarea-sm w-full font-mono text-xs" placeholder="JSON">${A(a)}</textarea>
211
+ <label class="label pb-1"><span class="label-text text-sm">${I(s.label)}</span></label>
212
+ <textarea data-frs-key="${I(s.name)}" rows="2" class="textarea textarea-bordered textarea-sm w-full font-mono text-xs" placeholder="JSON">${I(a)}</textarea>
213
213
  </div>`;default:return `<div class="form-control mb-2">
214
- <label class="label pb-1"><span class="label-text text-sm">${A(o.label)}</span></label>
215
- <input type="text" data-frs-key="${A(o.name)}" value="${A(a)}" class="input input-bordered input-sm w-full">
214
+ <label class="label pb-1"><span class="label-text text-sm">${I(s.label)}</span></label>
215
+ <input type="text" data-frs-key="${I(s.name)}" value="${I(a)}" class="input input-bordered input-sm w-full">
216
216
  </div>`}}).join(`
217
217
  `)}
218
- </div>`}function se(e,t,n,r="Save"){let o=e.map(s=>Ve(s)).join(`
218
+ </div>`}function de(e,t,n,r="Save"){let s=e.map(o=>pt(o)).join(`
219
219
  `);return `
220
- <form action="${A(t)}" method="${n}" novalidate data-frs-form>
221
- ${o}
220
+ <form action="${I(t)}" method="${n}" novalidate data-frs-form>
221
+ ${s}
222
222
  <div class="flex gap-2 mt-4 pt-4 border-t border-base-200">
223
- <button type="submit" class="btn btn-primary btn-sm">${A(r)}</button>
223
+ <button type="submit" class="btn btn-primary btn-sm">${I(r)}</button>
224
224
  <button type="button" class="btn btn-ghost btn-sm" onclick="history.back()">Cancel</button>
225
225
  </div>
226
- </form>`}function Vt({val:e}){return jsx("span",{class:"text-sm text-base-content/80 font-mono tabular-nums whitespace-nowrap",children:e.toLocaleString()})}function Je({val:e}){if(e==null)return jsx("span",{class:"opacity-30 italic text-xs",children:"\u2014"});if(typeof e=="boolean")return e?jsx("span",{class:"badge badge-success badge-sm",children:"true"}):jsx("span",{class:"badge badge-error badge-sm",children:"false"});if(e instanceof Date)return jsx(Vt,{val:e});if(typeof e=="object"&&e!==null&&typeof e.toDate=="function")return jsx(Vt,{val:e.toDate()});if(typeof e=="number")return jsx("span",{class:"text-sm font-mono tabular-nums",children:String(e)});if(Array.isArray(e))return e.length===0?jsx("span",{class:"text-xs text-base-content/30",children:"[]"}):jsxs("ul",{class:"list-none p-0 m-0 space-y-0.5 text-xs",children:[e.slice(0,8).map((n,r)=>jsx("li",{class:"break-all",children:typeof n=="object"?JSON.stringify(n):String(n)},r)),e.length>8&&jsxs("li",{class:"text-base-content/40 italic",children:["+",e.length-8," more\u2026"]})]});if(typeof e=="object"&&e!==null){let n=Object.entries(e);return n.length===0?jsx("span",{class:"text-xs text-base-content/30",children:"{}"}):jsxs("dl",{class:"grid grid-cols-[auto_1fr] gap-x-2 gap-y-0.5 text-xs m-0",children:[n.slice(0,8).map(([r,o])=>jsxs(Fragment,{children:[jsx("dt",{class:"text-base-content/50 font-semibold whitespace-nowrap",children:r}),jsx("dd",{class:"break-all",children:String(o??"")})]})),n.length>8&&jsxs("dt",{class:"col-span-2 text-base-content/40 italic",children:["+",n.length-8," more\u2026"]})]})}let t=String(e);return jsx("span",{class:"text-sm break-all",children:t})}var Jt=`// \u2500\u2500 Form validation + array serialization \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
226
+ </form>`}et();var bn=`// \u2500\u2500 Right panel (relations preview) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
227
+ (function () {
228
+ function panelEls() {
229
+ return {
230
+ root: document.querySelector("[data-frs-panel-root]"),
231
+ backdrop: document.querySelector("[data-frs-panel-backdrop]"),
232
+ panel: document.querySelector("[data-frs-panel]"),
233
+ title: document.querySelector("[data-frs-panel-title]"),
234
+ body: document.querySelector("[data-frs-panel-body]"),
235
+ };
236
+ }
237
+ function openPanel(label) {
238
+ var els = panelEls();
239
+ if (!els.root || !els.panel) return;
240
+ els.root.classList.remove("hidden");
241
+ els.root.setAttribute("aria-hidden", "false");
242
+ requestAnimationFrame(function () {
243
+ if (els.backdrop) els.backdrop.style.opacity = "1";
244
+ if (els.panel) els.panel.style.transform = "translateX(0)";
245
+ });
246
+ if (els.title) els.title.textContent = label || "Relation";
247
+ if (els.body) {
248
+ els.body.innerHTML =
249
+ '<div class="flex items-center justify-center py-12 text-base-content/40"><span class="loading loading-spinner loading-md"></span></div>';
250
+ }
251
+ }
252
+ function closePanel() {
253
+ var els = panelEls();
254
+ if (!els.root || !els.panel) return;
255
+ if (els.backdrop) els.backdrop.style.opacity = "0";
256
+ if (els.panel) els.panel.style.transform = "translateX(100%)";
257
+ setTimeout(function () {
258
+ els.root.classList.add("hidden");
259
+ els.root.setAttribute("aria-hidden", "true");
260
+ }, 200);
261
+ }
262
+ function basePathFromLocation() {
263
+ // The current list page is at <basePath>/<repoName>...; we strip everything after the basePath.
264
+ // We can't know basePath without context, so we derive it from the current URL up to the repoName segment.
265
+ // Caller passes the absolute fragment URL via data attribute; we just need the prefix here.
266
+ return window.location.pathname.replace(/\\/[^/]*\\/?$/, "");
267
+ }
268
+ function fetchPanel(url, label) {
269
+ openPanel(label);
270
+ fetch(url, { credentials: "same-origin" })
271
+ .then(function (r) {
272
+ return r.text();
273
+ })
274
+ .then(function (html) {
275
+ var els = panelEls();
276
+ if (els.body) els.body.innerHTML = html;
277
+ })
278
+ .catch(function (err) {
279
+ var els = panelEls();
280
+ if (els.body) {
281
+ els.body.innerHTML =
282
+ '<div class="p-6 text-error text-sm">Error: ' +
283
+ (err && err.message ? err.message : String(err)) +
284
+ "</div>";
285
+ }
286
+ });
287
+ }
288
+ document.addEventListener("click", function (e) {
289
+ var trigger = e.target.closest("[data-frs-relation]");
290
+ if (trigger) {
291
+ e.preventDefault();
292
+ var type = trigger.getAttribute("data-frs-rel-type");
293
+ var repo = trigger.getAttribute("data-frs-rel-repo");
294
+ var fk = trigger.getAttribute("data-frs-rel-fk");
295
+ var val = trigger.getAttribute("data-frs-rel-val");
296
+ var label = trigger.getAttribute("data-frs-rel-label") || "Relation";
297
+ var bp = basePathFromLocation();
298
+ var url;
299
+ if (type === "one") {
300
+ url =
301
+ bp +
302
+ "/" +
303
+ encodeURIComponent(repo) +
304
+ "/_panel?type=one&id=" +
305
+ encodeURIComponent(val);
306
+ } else {
307
+ url =
308
+ bp +
309
+ "/" +
310
+ encodeURIComponent(repo) +
311
+ "/_panel?type=many&fk=" +
312
+ encodeURIComponent(fk) +
313
+ "&fv=" +
314
+ encodeURIComponent(val);
315
+ }
316
+ fetchPanel(url, repo + " \xB7 " + label);
317
+ return;
318
+ }
319
+ if (e.target.closest("[data-frs-panel-close]")) {
320
+ closePanel();
321
+ return;
322
+ }
323
+ if (e.target.closest("[data-frs-panel-backdrop]")) {
324
+ closePanel();
325
+ return;
326
+ }
327
+ var pageBtn = e.target.closest("[data-frs-panel-page]");
328
+ if (pageBtn) {
329
+ // Recompute URL by inspecting the current open panel context \u2014 encoded in the body's first anchor "Full view \u2192"
330
+ // Simpler: rebuild via the previous URL stored in data-attr.
331
+ var fullViewLink = document.querySelector(
332
+ "[data-frs-panel-body] a.btn-outline[href]",
333
+ );
334
+ if (!fullViewLink) return;
335
+ var dir = pageBtn.getAttribute("data-frs-panel-page");
336
+ var cursor = pageBtn.getAttribute("data-cursor") || "";
337
+ var fullViewUrl = new URL(fullViewLink.href, window.location.href);
338
+ var repo = fullViewUrl.pathname.split("/").filter(Boolean).pop();
339
+ var bp = fullViewUrl.pathname.replace(/\\/[^/]+\\/?$/, "");
340
+ var fk = "";
341
+ var fv = "";
342
+ fullViewUrl.searchParams.forEach(function (v, k) {
343
+ if (k.indexOf("fv_") === 0) {
344
+ fk = k.slice(3);
345
+ fv = v;
346
+ }
347
+ });
348
+ var url =
349
+ bp +
350
+ "/" +
351
+ encodeURIComponent(repo) +
352
+ "/_panel?type=many&fk=" +
353
+ encodeURIComponent(fk) +
354
+ "&fv=" +
355
+ encodeURIComponent(fv) +
356
+ "&cursor=" +
357
+ encodeURIComponent(cursor) +
358
+ "&dir=" +
359
+ encodeURIComponent(dir);
360
+ fetchPanel(url, repo);
361
+ }
362
+ });
363
+ document.addEventListener("keydown", function (e) {
364
+ if (e.key === "Escape") closePanel();
365
+ });
366
+ })();
367
+
368
+ // \u2500\u2500 Bulk selection + actions \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
369
+ (function () {
370
+ var state = {
371
+ selectAllAcrossQuery: false,
372
+ // Keep a Set of selected ids on the current page (cleared on navigation).
373
+ selected: new Set(),
374
+ };
375
+
376
+ function bar() {
377
+ return document.querySelector("[data-frs-bulk-bar]");
378
+ }
379
+ function summarize() {
380
+ var b = bar();
381
+ if (!b) return;
382
+ var total = parseInt(b.getAttribute("data-frs-total") || "0", 10);
383
+ var summary = b.querySelector("[data-frs-bulk-summary]");
384
+ var selectAllBtn = b.querySelector("[data-frs-bulk-select-all]");
385
+ var allActive = b.querySelector("[data-frs-bulk-all-active]");
386
+ var pageSize = parseInt(b.getAttribute("data-frs-page-size") || "0", 10);
387
+ var n = state.selected.size;
388
+ if (state.selectAllAcrossQuery) {
389
+ if (allActive) allActive.classList.remove("hidden");
390
+ if (selectAllBtn) selectAllBtn.classList.add("hidden");
391
+ if (summary) summary.textContent = "";
392
+ b.classList.remove("hidden");
393
+ return;
394
+ }
395
+ if (n === 0) {
396
+ b.classList.add("hidden");
397
+ if (selectAllBtn) selectAllBtn.classList.add("hidden");
398
+ if (allActive) allActive.classList.add("hidden");
399
+ return;
400
+ }
401
+ b.classList.remove("hidden");
402
+ if (allActive) allActive.classList.add("hidden");
403
+ if (summary) {
404
+ summary.textContent =
405
+ n + " selected" + (pageSize ? " on this page" : "");
406
+ }
407
+ if (selectAllBtn && total > n) {
408
+ selectAllBtn.classList.remove("hidden");
409
+ } else if (selectAllBtn) {
410
+ selectAllBtn.classList.add("hidden");
411
+ }
412
+ }
413
+ function syncHeader() {
414
+ var head = document.querySelector("[data-frs-select-page]");
415
+ if (!head) return;
416
+ var rows = document.querySelectorAll("[data-frs-select-row]");
417
+ var checked = 0;
418
+ rows.forEach(function (r) {
419
+ if (r.checked) checked++;
420
+ });
421
+ head.indeterminate = checked > 0 && checked < rows.length;
422
+ head.checked = rows.length > 0 && checked === rows.length;
423
+ }
424
+ document.addEventListener("change", function (e) {
425
+ if (e.target.matches("[data-frs-select-row]")) {
426
+ var id = e.target.value;
427
+ if (e.target.checked) state.selected.add(id);
428
+ else state.selected.delete(id);
429
+ state.selectAllAcrossQuery = false;
430
+ syncHeader();
431
+ summarize();
432
+ } else if (e.target.matches("[data-frs-select-page]")) {
433
+ var rows = document.querySelectorAll("[data-frs-select-row]");
434
+ rows.forEach(function (r) {
435
+ r.checked = e.target.checked;
436
+ if (e.target.checked) state.selected.add(r.value);
437
+ else state.selected.delete(r.value);
438
+ });
439
+ state.selectAllAcrossQuery = false;
440
+ summarize();
441
+ }
442
+ });
443
+ document.addEventListener("click", function (e) {
444
+ var sa = e.target.closest("[data-frs-bulk-select-all]");
445
+ if (sa) {
446
+ state.selectAllAcrossQuery = true;
447
+ summarize();
448
+ return;
449
+ }
450
+ var clear = e.target.closest("[data-frs-bulk-clear]");
451
+ if (clear) {
452
+ state.selectAllAcrossQuery = false;
453
+ state.selected.clear();
454
+ document
455
+ .querySelectorAll("[data-frs-select-row]")
456
+ .forEach(function (r) {
457
+ r.checked = false;
458
+ });
459
+ syncHeader();
460
+ summarize();
461
+ return;
462
+ }
463
+ var actBtn = e.target.closest("[data-frs-bulk-action]");
464
+ if (actBtn) {
465
+ var action = actBtn.getAttribute("data-frs-bulk-action");
466
+ if (action === "delete") doBulkDelete();
467
+ else if (action === "update") openBulkUpdateModal();
468
+ }
469
+ });
470
+
471
+ function buildPayload() {
472
+ var b = bar();
473
+ if (!b) return null;
474
+ if (state.selectAllAcrossQuery) {
475
+ var filters = [];
476
+ try {
477
+ filters = JSON.parse(b.getAttribute("data-frs-filters") || "[]");
478
+ } catch (_) {}
479
+ return { selectAll: true, filters: filters };
480
+ }
481
+ return { ids: Array.from(state.selected) };
482
+ }
483
+
484
+ function repoEndpoint(action) {
485
+ var b = bar();
486
+ if (!b) return null;
487
+ var repo = b.getAttribute("data-frs-repo");
488
+ var bp = window.location.pathname.replace(/\\/[^/]*\\/?$/, "");
489
+ return bp + "/" + encodeURIComponent(repo) + "/_bulk/" + action;
490
+ }
491
+
492
+ function doBulkDelete() {
493
+ var payload = buildPayload();
494
+ if (!payload) return;
495
+ var n = state.selectAllAcrossQuery
496
+ ? "all matching"
497
+ : payload.ids.length + "";
498
+ if (!confirm("Delete " + n + " documents? This cannot be undone.")) return;
499
+ var url = repoEndpoint("delete");
500
+ if (!url) return;
501
+ fetch(url, {
502
+ method: "POST",
503
+ credentials: "same-origin",
504
+ headers: { "Content-Type": "application/json" },
505
+ body: JSON.stringify(payload),
506
+ })
507
+ .then(function (r) {
508
+ return r.json().then(function (j) {
509
+ return { ok: r.ok, body: j };
510
+ });
511
+ })
512
+ .then(function (res) {
513
+ if (!res.ok) {
514
+ alert("Bulk delete failed: " + (res.body && res.body.error));
515
+ return;
516
+ }
517
+ window.location.reload();
518
+ })
519
+ .catch(function (err) {
520
+ alert("Bulk delete failed: " + err.message);
521
+ });
522
+ }
523
+
524
+ function openBulkUpdateModal() {
525
+ var b = bar();
526
+ var dialog = document.getElementById("frs-bulk-update-modal");
527
+ if (!b || !dialog) return;
528
+ var fields = [];
529
+ try {
530
+ fields = JSON.parse(b.getAttribute("data-frs-fields") || "[]");
531
+ } catch (_) {}
532
+ if (fields.length === 0) return;
533
+ var summary = dialog.querySelector(
534
+ "[data-frs-bulk-update-summary]",
535
+ );
536
+ if (summary) {
537
+ summary.textContent = state.selectAllAcrossQuery
538
+ ? "Update one field on all matching documents."
539
+ : "Update one field on " +
540
+ state.selected.size +
541
+ " selected document" +
542
+ (state.selected.size !== 1 ? "s" : "") +
543
+ ".";
544
+ }
545
+ var select = dialog.querySelector("[data-frs-bulk-field-select]");
546
+ var valueContainer = dialog.querySelector(
547
+ "[data-frs-bulk-value-container]",
548
+ );
549
+ var renderValueInput = function () {
550
+ if (!valueContainer || !select) return;
551
+ var name = select.value;
552
+ var meta = fields.filter(function (f) {
553
+ return f.name === name;
554
+ })[0];
555
+ if (!meta) {
556
+ valueContainer.innerHTML = "";
557
+ return;
558
+ }
559
+ var inputHtml = "";
560
+ var label =
561
+ '<div class="label"><span class="label-text text-xs uppercase tracking-wide">Value</span></div>';
562
+ if (meta.type === "boolean") {
563
+ inputHtml =
564
+ '<select name="value" class="select select-bordered select-sm w-full">' +
565
+ '<option value="true">true</option><option value="false">false</option>' +
566
+ "</select>";
567
+ } else if (
568
+ (meta.type === "enum" || meta.type === "literal") &&
569
+ meta.enumValues &&
570
+ meta.enumValues.length
571
+ ) {
572
+ inputHtml =
573
+ '<select name="value" class="select select-bordered select-sm w-full">';
574
+ meta.enumValues.forEach(function (v) {
575
+ inputHtml +=
576
+ '<option value="' + escapeAttr(v) + '">' + escapeHtml(v) + "</option>";
577
+ });
578
+ inputHtml += "</select>";
579
+ } else if (meta.type === "number" || meta.type === "bigint") {
580
+ inputHtml =
581
+ '<input type="number" step="any" name="value" class="input input-bordered input-sm w-full" />';
582
+ } else {
583
+ inputHtml =
584
+ '<input type="text" name="value" class="input input-bordered input-sm w-full" />';
585
+ }
586
+ valueContainer.innerHTML =
587
+ '<label class="form-control w-full">' + label + inputHtml + "</label>";
588
+ };
589
+ if (select) {
590
+ select.value = "";
591
+ select.onchange = renderValueInput;
592
+ }
593
+ if (valueContainer) valueContainer.innerHTML = "";
594
+ var form = dialog.querySelector("[data-frs-bulk-update-form]");
595
+ if (form) {
596
+ form.onsubmit = function (e) {
597
+ e.preventDefault();
598
+ if (!select || !select.value) return;
599
+ var meta = fields.filter(function (f) {
600
+ return f.name === select.value;
601
+ })[0];
602
+ if (!meta) return;
603
+ var input = valueContainer.querySelector(
604
+ 'input[name="value"], select[name="value"]',
605
+ );
606
+ if (!input) return;
607
+ var raw = input.value;
608
+ var typed;
609
+ if (meta.type === "boolean") typed = raw === "true";
610
+ else if (meta.type === "number") typed = raw === "" ? null : Number(raw);
611
+ else if (meta.type === "bigint") typed = raw === "" ? null : Number(raw);
612
+ else typed = raw;
613
+ var payload = buildPayload();
614
+ if (!payload) return;
615
+ payload.field = select.value;
616
+ payload.value = typed;
617
+ var url = repoEndpoint("update");
618
+ if (!url) return;
619
+ fetch(url, {
620
+ method: "POST",
621
+ credentials: "same-origin",
622
+ headers: { "Content-Type": "application/json" },
623
+ body: JSON.stringify(payload),
624
+ })
625
+ .then(function (r) {
626
+ return r.json().then(function (j) {
627
+ return { ok: r.ok, body: j };
628
+ });
629
+ })
630
+ .then(function (res) {
631
+ if (!res.ok) {
632
+ alert("Bulk update failed: " + (res.body && res.body.error));
633
+ return;
634
+ }
635
+ dialog.close();
636
+ window.location.reload();
637
+ })
638
+ .catch(function (err) {
639
+ alert("Bulk update failed: " + err.message);
640
+ });
641
+ };
642
+ }
643
+ var cancel = dialog.querySelector("[data-frs-bulk-update-cancel]");
644
+ if (cancel)
645
+ cancel.onclick = function () {
646
+ dialog.close();
647
+ };
648
+ if (typeof dialog.showModal === "function") dialog.showModal();
649
+ else dialog.setAttribute("open", "");
650
+ }
651
+
652
+ function escapeHtml(s) {
653
+ return String(s).replace(/[&<>"']/g, function (c) {
654
+ return (
655
+ { "&": "&amp;", "<": "&lt;", ">": "&gt;", '"': "&quot;", "'": "&#39;" }[
656
+ c
657
+ ] || c
658
+ );
659
+ });
660
+ }
661
+ function escapeAttr(s) {
662
+ return escapeHtml(s);
663
+ }
664
+ })();
665
+
666
+ // \u2500\u2500 Theme switcher \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
667
+ (function () {
668
+ function syncThemeUI() {
669
+ var current =
670
+ document.documentElement.getAttribute("data-theme") || "corporate";
671
+ var label = document.querySelector("[data-frs-theme-current]");
672
+ if (label) label.textContent = current;
673
+ document.querySelectorAll("[data-frs-theme-check]").forEach(function (el) {
674
+ el.classList.toggle(
675
+ "hidden",
676
+ el.getAttribute("data-frs-theme-check") !== current,
677
+ );
678
+ });
679
+ }
680
+ document.addEventListener("DOMContentLoaded", syncThemeUI);
681
+ document.addEventListener("click", function (e) {
682
+ var btn = e.target.closest("[data-frs-theme]");
683
+ if (!btn) return;
684
+ var theme = btn.getAttribute("data-frs-theme");
685
+ if (!theme) return;
686
+ document.documentElement.setAttribute("data-theme", theme);
687
+ try {
688
+ localStorage.setItem("frs-admin-theme", theme);
689
+ } catch (_) {}
690
+ syncThemeUI();
691
+ // Close the dropdown by blurring focus (DaisyUI dropdown closes on focusout)
692
+ if (document.activeElement && document.activeElement.blur) {
693
+ document.activeElement.blur();
694
+ }
695
+ });
696
+ })();
697
+
698
+ // \u2500\u2500 Form validation + array serialization \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
227
699
  document.addEventListener("submit", function (e) {
228
700
  var form = e.target;
229
701
  if (!form.hasAttribute("data-frs-form")) return;
@@ -642,7 +1114,7 @@ function initColumnReorder(table) {
642
1114
  });
643
1115
  });
644
1116
  }
645
- `;function ke(){return jsx("script",{dangerouslySetInnerHTML:{__html:Jt}})}function ae(e){return "<!DOCTYPE html>"+renderToString(e)}var ie=({opts:e,children:t})=>{let{title:n,breadcrumb:r,flash:o,basePath:s="/"}=e;return jsxs("html",{lang:"en","data-theme":"corporate",children:[jsxs("head",{children:[jsx("meta",{charset:"UTF-8"}),jsx("meta",{name:"viewport",content:"width=device-width, initial-scale=1"}),jsxs("title",{children:[n," \u2014 FRS Admin"]}),jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/themes.css",rel:"stylesheet",type:"text/css"}),jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/daisyui.css",rel:"stylesheet",type:"text/css"}),jsx("script",{src:"https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4"})]}),jsxs("body",{class:"bg-base-200/50 min-h-screen flex flex-col",children:[jsx("div",{class:"navbar bg-neutral text-neutral-content shadow-sm sticky top-0 z-50 px-6",children:jsx("div",{class:"flex-1",children:jsx("a",{href:s,class:"font-bold text-lg tracking-tight hover:opacity-80 transition-opacity",children:"FRS Admin"})})}),jsxs("main",{class:"px-6 py-8 w-full flex-1",children:[r&&r.length>0&&jsx("div",{class:"text-sm breadcrumbs mb-4",children:jsx("ul",{children:r.map((a,u)=>a.href?jsx("li",{children:jsx("a",{href:a.href,children:a.label})},u):jsx("li",{class:"text-base-content/60",children:a.label},u))})}),jsx("h1",{class:"text-2xl font-bold mb-6",children:n}),o&&jsxs("div",{role:"alert",class:`alert ${o.type==="success"?"alert-success":o.type==="warning"?"alert-warning":"alert-error"} mb-6`,children:[jsx("span",{class:"flex-1",children:o.message}),o.action&&jsx("a",{href:o.action.href,...o.action.external?{target:"_blank",rel:"noopener noreferrer"}:{},class:"btn btn-sm btn-outline",children:o.action.label})]}),t]}),jsx(ke,{})]})]})};function Xe(e,t){return ae(jsx(ie,{opts:t,children:jsx("div",{dangerouslySetInnerHTML:{__html:e}})}))}function Ye(e,t){return ae(jsx(ie,{opts:{title:"Repositories",basePath:t},children:e.length===0?jsxs("div",{class:"text-center py-20 text-base-content/50",children:[jsx("p",{class:"text-lg font-medium mb-1",children:"No repositories configured"}),jsx("p",{class:"text-sm",children:"Add a repository to your FRS config to get started."})]}):jsx("div",{class:"grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4",children:e.map(n=>jsx("a",{href:`${t}/${n.name}`,class:"card bg-base-100 border border-base-300 hover:shadow-md no-underline transition-shadow",children:jsxs("div",{class:"card-body p-5",children:[jsx("h2",{class:"card-title text-sm font-semibold",children:n.name}),jsx("p",{class:"text-xs text-base-content/50 font-mono",children:n.path})]})},n.name))})}))}var Yt=[{value:"==",label:"="},{value:"!=",label:"\u2260"},{value:"in",label:"in"},{value:"not-in",label:"not in"}],Wn=[{value:"==",label:"="},{value:"!=",label:"\u2260"},{value:"<",label:"<"},{value:"<=",label:"\u2264"},{value:">",label:">"},{value:">=",label:"\u2265"},{value:"in",label:"in"},{value:"not-in",label:"not in"}],Vn=[{value:"array-contains",label:"contains"},{value:"array-contains-any",label:"contains any"}];function Jn(e){switch(e){case "ZodNumber":case "ZodBigInt":case "ZodDate":return Wn;case "ZodBoolean":return Yt;case "ZodArray":return Vn;default:return Yt}}var ce="__null__";function Xn(e){return `(function(cb){var i=document.getElementById('${e}');if(!i)return;if(cb.checked){i.dataset._prev=i.value;if(i.tagName==='SELECT'){var o=i.querySelector('option[value="${ce}"]');if(!o){o=document.createElement('option');o.value='${ce}';o.textContent='\u2205 null';o.dataset._auto='1';i.appendChild(o);}o.selected=true;}else{if(i.type==='number'||i.type==='datetime-local'){i.dataset._type=i.type;i.type='text';}i.value='${ce}';i.readOnly=true;}i.style.opacity='0.55';}else{i.style.opacity='';if(i.tagName==='SELECT'){var o2=i.querySelector('option[value="${ce}"][data-_auto="1"]');if(o2)o2.remove();var prev=i.dataset._prev||'';for(var k=0;k<i.options.length;k++)i.options[k].selected=(i.options[k].value===prev);}else{if(i.dataset._type){i.type=i.dataset._type;delete i.dataset._type;}i.readOnly=false;i.value=(i.dataset._prev&&i.dataset._prev!=='${ce}')?i.dataset._prev:'';}}})(this)`}function Yn(e,t){return `(function(){var h=document.getElementById('${e}');var boxes=document.querySelectorAll('input[data-enum-group="${t}"]');h.value=Array.from(boxes).filter(function(b){return b.checked;}).map(function(b){return b.value;}).join(',');})()`}function Ae({inputId:e,active:t}){return jsxs("label",{class:"flex items-center gap-1 cursor-pointer select-none text-xs text-base-content/60 hover:text-base-content border border-base-300 rounded-md px-1.5 py-1 shrink-0 leading-none h-8",title:"Filter where field IS NULL",children:[jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",checked:t,onchange:Xn(e)}),jsx("span",{children:"\u2205"})]})}function er({col:e,active:t}){let n=t?.value??"",r=n===ce,o=`fv_input_${e.name.replace(/\./g,"__")}`,s=t?.op,a=s==="in"||s==="not-in";if(e.enumValues&&e.enumValues.length>0){if(a){let u=new Set(n.split(",").map(l=>l.trim()).filter(Boolean)),i=`eg_${e.name.replace(/\./g,"__")}`;return jsxs("div",{class:"flex flex-wrap items-center gap-1 w-full",children:[jsx("input",{type:"hidden",id:o,name:`fv_${e.name}`,value:n}),e.enumValues.map(l=>jsxs("label",{class:"flex items-center gap-1 text-xs border border-base-300 rounded px-2 cursor-pointer hover:bg-base-200",children:[jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",value:l,checked:u.has(l),"data-enum-group":i,onchange:Yn(o,i)}),jsx("span",{children:l})]},l))]})}return jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxs("select",{id:o,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),e.enumValues.map(u=>jsx("option",{value:u,selected:n===u,children:u},u)),e.nullable&&jsx("option",{value:ce,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsx(Ae,{inputId:o,active:r})]})}if(e.zodType==="ZodBoolean")return jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxs("select",{id:o,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),jsx("option",{value:"true",selected:n==="true",children:"true"}),jsx("option",{value:"false",selected:n==="false",children:"false"}),e.nullable&&jsx("option",{value:ce,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsx(Ae,{inputId:o,active:r})]});if(e.zodType==="ZodArray"){let u=t?.op==="array-contains-any";return jsx("input",{id:o,type:"text",name:`fv_${e.name}`,value:n,placeholder:u?"val1, val2, \u2026":"value",class:"input input-sm input-bordered w-full"})}return e.zodType==="ZodNumber"||e.zodType==="ZodBigInt"?jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsx("input",{id:o,type:r?"text":"number",name:`fv_${e.name}`,value:n,placeholder:"value",class:"input input-sm input-bordered w-full",readOnly:r,style:r?"opacity:0.55":void 0,"data-_type":r?"number":void 0}),e.nullable&&jsx(Ae,{inputId:o,active:r})]}):e.zodType==="ZodDate"?jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsx("input",{id:o,type:r?"text":"datetime-local",name:`fv_${e.name}`,value:n,class:"input input-sm input-bordered w-full",readOnly:r,style:r?"opacity:0.55":void 0,"data-_type":r?"datetime-local":void 0}),e.nullable&&jsx(Ae,{inputId:o,active:r})]}):jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsx("input",{id:o,type:"text",name:`fv_${e.name}`,value:n,placeholder:"value",class:"input input-sm input-bordered w-full",readOnly:r,style:r?"opacity:0.55":void 0}),e.nullable&&jsx(Ae,{inputId:o,active:r})]})}function et({action:e,columnMeta:t,activeFilters:n,isGroup:r}){let o=Object.fromEntries(n.map(i=>[i.field,i])),s=n.length>0,a=n.length>=2||r&&s,u=t.filter(i=>i.zodType!=="ZodObject"&&i.zodType!=="ZodRecord");return jsxs("details",{class:"collapse collapse-arrow bg-base-100 border border-base-300 rounded-box mb-6 shadow-sm",open:s?true:void 0,children:[jsxs("summary",{class:"collapse-title text-sm font-medium py-2 min-h-0",children:["Filters",s&&jsxs("span",{class:"badge badge-primary badge-sm ml-2",children:[n.length," active"]})]}),jsx("div",{class:"collapse-content pb-4 pt-2",children:jsxs("form",{method:"get",action:e,children:[jsx("div",{class:"grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4",children:u.map(i=>{let l=Jn(i.zodType),y=o[i.name],h=y?.op??l[0].value;return jsxs("div",{class:"flex flex-col gap-1.5",children:[jsx("label",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:i.name}),jsxs("div",{class:"flex gap-1.5",children:[l.length>1?jsx("select",{name:`fo_${i.name}`,class:"select select-sm select-bordered w-20 shrink-0",children:l.map(c=>jsx("option",{value:c.value,selected:c.value===h,children:c.label},c.value))}):jsx("input",{type:"hidden",name:`fo_${i.name}`,value:l[0].value}),jsx(er,{col:i,active:y})]})]},i.name)})}),jsxs("div",{class:"flex flex-wrap gap-2 mt-4 pt-3 border-t border-base-200 items-center",children:[jsx("button",{type:"submit",class:"btn btn-sm btn-primary",children:"Apply"}),s&&jsx("a",{href:e,class:"btn btn-sm btn-ghost",children:"Clear"}),a&&jsxs("span",{class:"text-xs text-warning ml-auto flex items-center gap-1",children:[jsx("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:jsx("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"})}),r?"Collection group queries require a composite index":"Multiple filters may require a composite index"]})]})]})})]})}function tt(e,t,n,r,o,s){let a=n==="create"?`Create ${e}`:`Edit ${e} / ${r??""}`,u=n==="create"?[{label:"Repositories",href:o},{label:e,href:`${o}/${e}`},{label:"New document"}]:[{label:"Repositories",href:o},{label:e,href:`${o}/${e}`},{label:`Edit ${r??""}`}];return ae(jsx(ie,{opts:{title:a,breadcrumb:u,basePath:o,flash:s},children:jsx("div",{class:"card bg-base-100 border border-base-300",children:jsx("div",{class:"card-body p-6",children:jsx("div",{dangerouslySetInnerHTML:{__html:t}})})})}))}function nt(e,t,n){let r=new URLSearchParams;for(let o of e)r.set(`fv_${o.field}`,o.value),r.set(`fo_${o.field}`,o.op);return t&&(r.set("ob",t.field),r.set("od",t.dir)),n&&r.set("ps",String(n)),r}function en(e,t,n,r,o){let s=nt(e,r,o);return s.set("cursor",t),s.set("dir",n),`?${s.toString()}`}function tr(e,t,n,r){let o=nt(n,void 0,r);return t?.field===e?t.dir==="asc"&&(o.set("ob",e),o.set("od","desc")):(o.set("ob",e),o.set("od","asc")),`?${o.toString()}`}function nr(e,t,n){return `?${nt(t,n,e).toString()}`}function rt(e,t,n,r,o,s,a=[],u=[],i=false,l=[],y,h,c,d){let p=`${r}/${e}`,m=`${p}/create`;return ae(jsxs(ie,{opts:{title:e,breadcrumb:[{label:"Repositories",href:r},{label:e}],basePath:r,flash:s},children:[a.length>0&&jsx(et,{action:p,columnMeta:a,activeFilters:u,isGroup:d}),c&&jsxs("div",{role:"alert",class:`alert ${c.type==="index"?"alert-warning":"alert-error"} mb-6 shadow-sm`,children:[jsx("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-6 w-6 shrink-0 stroke-current",fill:"none",viewBox:"0 0 24 24",children:c.type==="index"?jsx("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z"}):jsx("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z"})}),jsxs("div",{class:"flex-1",children:[jsx("h3",{class:"font-bold",children:c.type==="index"?"Composite index required":"Query failed"}),jsx("div",{class:"text-sm",children:c.message})]}),c.indexUrl&&jsx("a",{href:c.indexUrl,target:"_blank",rel:"noopener noreferrer",class:"btn btn-sm btn-outline",children:"Create Index \u2192"})]}),jsxs("div",{class:"flex flex-wrap justify-between items-center mb-4 gap-3",children:[jsxs("div",{class:"flex items-center gap-3",children:[jsxs("span",{class:"text-sm text-base-content/60",children:[t.length," document",t.length!==1&&"s"]}),jsxs("div",{class:"flex items-center gap-1.5 text-sm text-base-content/60",children:[jsx("span",{children:"Rows"}),jsx("div",{class:"join",children:[10,25,50,100].map(f=>jsx("a",{href:nr(f,u,y),class:`join-item btn btn-xs ${h===f?"btn-active btn-primary":"btn-outline"}`,children:f},f))})]})]}),jsx("a",{href:m,class:"btn btn-primary btn-sm",children:"+ New"})]}),jsx("div",{class:"overflow-x-auto rounded-box border border-base-300 bg-base-100","data-frs-table-wrap":true,children:jsxs("table",{class:"table table-sm w-full","data-frs-table":true,"data-frs-repo":e,"data-frs-colcount":n.length,children:[jsx("thead",{children:jsxs("tr",{class:"bg-base-200/50",children:[[...n].map((f,v)=>{let g=y?.field===f,b=g?y.dir==="asc"?" \u25B2":" \u25BC":"";return jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:jsxs("a",{href:tr(f,y,u,h),class:`hover:text-base-content inline-flex items-center gap-0.5${g?" text-primary font-bold":""}`,children:[f,b]})},v)}),l.map((f,v)=>jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:f.column},`rel-${v}`)),jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide text-right",children:"Actions"})]})}),jsx("tbody",{children:t.length===0?jsx("tr",{children:jsx("td",{colspan:n.length+l.length+1,class:"text-center py-16 text-base-content/40",children:"No documents found"})}):t.map((f,v)=>{let g=String(f.docId??f.id??""),b=`${r}/${e}/${encodeURIComponent(g)}/edit`,R=`${r}/${e}/${encodeURIComponent(g)}/delete`;return jsxs("tr",{class:"hover",children:[n.map((x,S)=>jsx("td",{class:"align-top py-2",children:jsx(Je,{val:f[x]})},S)),l.map((x,S)=>{let C=f[x.key];if(C==null||C==="")return jsx("td",{class:"py-2"},`rel-${S}`);let P=`${r}/${x.targetRepo}?fv_${x.targetKey}=${encodeURIComponent(String(C))}`;return jsx("td",{class:"align-middle py-2",children:jsx("a",{href:P,class:"btn btn-xs btn-ghost btn-outline",children:x.column})},`rel-${S}`)}),jsx("td",{class:"align-middle text-right whitespace-nowrap py-2",children:jsxs("div",{class:"flex gap-1 justify-end",children:[jsx("a",{href:b,class:"btn btn-xs btn-outline",children:"Edit"}),i&&jsx("form",{method:"post",action:R,onsubmit:"return confirm('Delete this document?')",children:jsx("button",{type:"submit",class:"btn btn-xs btn-error btn-outline",children:"Delete"})})]})})]},v)})})]})}),(o.hasPrev||o.hasNext)&&jsxs("div",{class:"flex justify-center items-center mt-6 gap-2",children:[o.hasPrev?jsx("a",{href:en(u,o.prevCursor,"prev",y,h),class:"btn btn-sm btn-outline",children:"\u2190 Previous"}):jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"\u2190 Previous"}),o.hasNext?jsx("a",{href:en(u,o.nextCursor,"next",y,h),class:"btn btn-sm btn-outline",children:"Next \u2192"}):jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"Next \u2192"})]})]}))}function Me(e,t){return Xe(e,t)}function ot(e,t){return Ye(e,t)}function st(e,t,n,r,o,s,a,u,i,l,y,h,c,d){return rt(e,t,n,r,o,s,a,u,i,l,y,h,c,d)}function le(e,t,n,r,o,s){return tt(e,t,n,r,o,s)}var rr=new Set(["<","<=",">",">=","!="]),or=new Set(["array-contains","array-contains-any"]);function at(e){return e==="desc"?"DESCENDING":"ASCENDING"}function sr(e){let t=e.split("/").filter(Boolean);return t[t.length-1]??e}function ar(e,t,n,r,o){let s=[],a=new Set;for(let i of r)if(i.op==="=="||i.op==="in"||i.op==="not-in"){if(a.has(i.field))continue;a.add(i.field),s.push({fieldPath:i.field,order:"ASCENDING"});}for(let i of r)if(or.has(i.op)){if(a.has(i.field))continue;a.add(i.field),s.push({fieldPath:i.field,arrayConfig:"CONTAINS"});}for(let i of r)if(rr.has(i.op)){if(a.has(i.field))continue;a.add(i.field);let l=o?.field===i.field?at(o.dir):"ASCENDING";s.push({fieldPath:i.field,order:l});}if(o&&!a.has(o.field)&&s.push({fieldPath:o.field,order:at(o.dir)}),s.length===1&&n)return lr(e,t,s[0]);let u=o&&s.some(i=>i.fieldPath===o.field)?at(o.dir):"ASCENDING";return s.push({fieldPath:"__name__",order:u}),ir(e,t,n,s)}function ir(e,t,n,r,o="(default)"){let s=`projects/${e}/databases/${o}/collectionGroups/${t}/indexes/_`,a=[...lt(1,s),...Be(2,n?2:1)];for(let l of r)a.push(...tn(3,nn(l)));let u=o==="(default)"?"-default-":o,i=encodeURIComponent(rn(a));return `https://console.firebase.google.com/project/${e}/firestore/databases/${u}/indexes?create_composite=${i}`}function cr(e){return e.match(/https:\/\/console\.firebase\.google\.com[^\s)"]*/)?.[0]}function it(e){let t=[],n=e>>>0;for(;n>=128;)t.push(n&127|128),n>>>=7;return t.push(n&127),t}function ct(e,t){return e<<3|t}function lt(e,t){let n=Array.from(new TextEncoder().encode(t));return [ct(e,2),...it(n.length),...n]}function Be(e,t){return [ct(e,0),...it(t)]}function tn(e,t){return [ct(e,2),...it(t.length),...t]}function nn(e){let t=[...lt(1,e.fieldPath)];return e.arrayConfig==="CONTAINS"?t.push(...Be(3,1)):t.push(...Be(2,e.order==="DESCENDING"?2:1)),t}function rn(e){let t=String.fromCharCode(...e),n;if(typeof Buffer<"u")n=Buffer.from(e).toString("base64");else if(typeof btoa<"u")n=btoa(t);else throw new Error("No base64 encoder available");return n.replace(/=+$/,"")}function lr(e,t,n,r="(default)"){let o=`projects/${e}/databases/${r}/collectionGroups/${t}/fields/${n.fieldPath}`,s=[...lt(1,o),...Be(2,2),...tn(3,nn(n))],a=r==="(default)"?"-default-":r,u=encodeURIComponent(rn(s));return `https://console.firebase.google.com/project/${e}/firestore/databases/${a}/indexes/automatic?create_exemption=${u}`}function dr(e){let t=e,n=[t?.firestore?.projectId,t?.firestore?.app?.options?.projectId,t?.firestore?._settings?.projectId,t?.firestore?.databaseId?.projectId,t?._firestore?.projectId];for(let o of n)if(typeof o=="string"&&o.length>0)return o;return process.env.GCLOUD_PROJECT||process.env.GOOGLE_CLOUD_PROJECT||process.env.FIREBASE_PROJECT_ID||void 0}function me(e){let t=e;return t?t.code===9?true:typeof t.message=="string"?t.message.includes("requires an index"):false:false}function Pe(e,t){let n=e??{},r=me(e),o;if(r&&(o=n.message?cr(n.message):void 0,!o)){let s=dr(t.ref);if(s){let a=sr(t.path);o=ar(s,a,t.isGroup,t.filters,t.sort);}}return {type:r?"index":"error",message:r?"This query requires a composite index that does not exist yet.":n.message??"Query failed",indexUrl:o}}var on="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function pr(){let e="";for(let t=0;t<20;t++)e+=on.charAt(Math.floor(Math.random()*on.length));return e}function sn(e,t){if(!t)return;let n=e[t];if(typeof n!="string"||!n)return;let r=n.split("/").filter(Boolean),o=[];for(let s=1;s<r.length;s+=2)o.push(r[s]);return o.length>0?o:void 0}async function dt(e,t){let n=e.documentKey??"docId",r=`by${n.charAt(0).toUpperCase()}${n.slice(1)}`;if(typeof e.repo.get[r]=="function")try{let s=await e.repo.get[r](t);if(s)return s}catch{}return (await e.repo.query.by({where:[[n,"==",t]],limit:1}))[0]??null}function ut(e,t,n){let r=e.documentKey??"docId",o=Pe(n,{ref:e.repo.ref,path:e.path,isGroup:!!e.isGroup,filters:[{field:r,op:"==",value:t}]});return o.type==="index"?{type:"warning",message:"Loading this document requires a composite index that does not exist yet.",...o.indexUrl?{action:{href:o.indexUrl,label:"Create Index \u2192",external:true}}:{}}:{type:"error",message:o.message}}function N(e,t,n=200){e.status(n).set("Content-Type","text/html; charset=utf-8").send(t);}function pt(e,t){e.status(302).set("Location",t).send("");}function ft(e,t){let n=t.shape,r={};for(let[o,s]of Object.entries(n)){let a=mt(s);if(a==="ZodObject"){if(e[o+"__isnull"]==="1"){r[o]=null;continue}let l={},y=false;for(let[d,p]of Object.entries(e))d.startsWith(`${o}.`)&&(l[d.slice(o.length+1)]=p,y=true);if(y){let d=s;for(;;){let p=j(d);if(p==="ZodOptional"||p==="ZodNullable"||p==="ZodDefault")d=q(d);else break}r[o]=ft(l,d);continue}let h=e[o],c=Array.isArray(h)?h[h.length-1]:h;if(c)try{r[o]=JSON.parse(c);}catch{r[o]=c;}continue}let u=e[o],i=Array.isArray(u)?u[u.length-1]:u;if(e[o+"__isnull"]==="1"){r[o]=null;continue}if(i===void 0||i===""){a==="ZodBoolean"&&(r[o]=false);continue}switch(a){case "ZodBoolean":i==="__null__"?r[o]=null:r[o]=i==="true"||i==="on"||i==="1";break;case "ZodNumber":case "ZodBigInt":r[o]=Number(i);break;case "ZodDate":r[o]=new Date(i);break;case "ZodArray":try{r[o]=JSON.parse(i);}catch{r[o]=i;}break;default:if(i.startsWith("{")||i.startsWith("["))try{r[o]=JSON.parse(i);break}catch{}r[o]=i;}}return r}function an(e){let t=null;if(e instanceof Date)t=e;else if(typeof e=="object"&&e!==null&&typeof e.toDate=="function")t=e.toDate();else if(typeof e=="object"&&e!==null&&"_seconds"in e&&"_nanoseconds"in e)t=new Date(e._seconds*1e3+Math.floor(e._nanoseconds/1e6));else if(typeof e=="string"||typeof e=="number"){let r=new Date(e);isNaN(r.getTime())||(t=r);}if(!t||isNaN(t.getTime()))return null;let n=r=>String(r).padStart(2,"0");return `${t.getFullYear()}-${n(t.getMonth()+1)}-${n(t.getDate())}T${n(t.getHours())}:${n(t.getMinutes())}`}function mt(e){let t=e;for(;;){let n=j(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=q(t);else return n}}function fr(e){let t=e;for(;;){let n=j(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=q(t);else return t}}function cn(e){let t=e;for(;;){let n=j(t);if(n==="ZodOptional"||n==="ZodNullable")return true;if(n==="ZodDefault"){t=q(t);continue}return false}}function ln(e){let t=fr(e),n=j(t);if(n==="ZodEnum"){let r=Se(t);return r.length>0?r:void 0}if(n==="ZodNativeEnum"){let r=Te(t),o=Object.values(r).filter(s=>typeof s=="string");return o.length>0?o:void 0}if(n==="ZodLiteral"){let r=je(t);return typeof r=="string"?[r]:void 0}}function dn(e,t,n=""){let r={};for(let o of Object.keys(t.shape)){let s=n?`${n}.${o}`:o,a=e[o];if(a===null){r[s]="__null__";continue}if(a===void 0)continue;let u=t.shape[o];for(;;){let l=j(u);if(l==="ZodOptional"||l==="ZodNullable"||l==="ZodDefault")u=q(u);else break}let i=j(u);if(i==="ZodObject"&&typeof a=="object"&&a!==null&&!Array.isArray(a)){let l=dn(a,u,s);Object.assign(r,l);}else if(i==="ZodDate"){let l=an(a);l!==null&&(r[s]=l);}else if(typeof a=="object"&&a!==null&&!Array.isArray(a)&&("_seconds"in a||typeof a.toDate=="function")){let l=an(a);r[s]=l??JSON.stringify(a,null,2);}else typeof a=="object"?r[s]=JSON.stringify(a,null,2):r[s]=String(a);}return r}function yt(e,t){return e.map(n=>({...n,defaultValue:t[n.name]??n.defaultValue,nested:n.nested?yt(n.nested,t):void 0}))}function yr(e,t){let n=new Set(["==","!=","<","<=",">",">=","in","not-in","array-contains","array-contains-any"]),r=[];for(let[o,s]of Object.entries(e)){if(!o.startsWith("fv_"))continue;let a=o.slice(3);if(!t.has(a))continue;let u=(s??"").trim();if(!u)continue;let i=e[`fo_${a}`]??"==",l=n.has(i)?i:"==";r.push({field:a,op:l,value:u});}return r}function mr(e){let t="__null__",n=r=>r===t?null:r==="true"?true:r==="false"?false:r!==""&&!isNaN(Number(r))?Number(r):r;return e.map(r=>{if(r.op==="array-contains-any"||r.op==="in"||r.op==="not-in"){let o=r.value.split(",").map(s=>s.trim()).filter(s=>s!==""&&s!==t).map(s=>n(s));return [r.field,r.op,o]}return [r.field,r.op,n(r.value)]})}function un(e,t,n=""){let r=[];for(let o of e){let s=n?`${n}.${o}`:o,a=t.shape[o];if(!a){r.push({name:s,zodType:"ZodString"});continue}let u=mt(a);if(u==="ZodObject"){let i=a;for(;;){let y=j(i);if(y==="ZodOptional"||y==="ZodNullable"||y==="ZodDefault")i=q(i);else break}let l=re(i);r.push(...un(Object.keys(l),i,s));}else r.push({name:s,zodType:u,nullable:cn(a),enumValues:ln(a)});}return r}function gr(e,t){let n=t.split("."),r=e;for(let o of n){for(;;){let a=j(r);if(a==="ZodOptional"||a==="ZodNullable"||a==="ZodDefault")r=q(r);else break}let s=re(r);if(!(o in s))return null;r=s[o];}return r}function he(e,t){if(!t||t.length===0)return e;let n=[],r=new Map;for(let s of t){let a=s.indexOf(".");if(a===-1)n.push(s);else {let u=s.slice(0,a),i=s.slice(a+1);r.has(u)||r.set(u,[]),r.get(u).push(i);}}let o={};for(let s of n)s in e.shape&&(o[s]=e.shape[s]);for(let[s,a]of r){if(!(s in e.shape))continue;let u=e.shape[s];for(;;){let i=j(u);if(i==="ZodOptional"||i==="ZodNullable"||i==="ZodDefault")u=q(u);else break}if(j(u)!=="ZodObject"){o[s]=e.shape[s];continue}o[s]=he(u,a);}return z.object(o)}function ge(e,t){let n=t==="/"?"":t.replace(/\/$/,"");if(process.env.FUNCTIONS_EMULATOR==="true"){let s=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??"demo-project",a=process.env.FUNCTION_REGION??"us-central1",u=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${s}/${a}/${u}${n}`}let r=process.env.K_SERVICE,o=e.hostname??e.headers?.host??"";return r&&o.includes("cloudfunctions.net")?`/${r.toLowerCase()}${n}`:n}function pn(e,t){return {handleDashboard:(l,y)=>{let h=ge(l,t),c=Object.values(e).map(d=>({name:d.name,path:d.path}));N(y,ot(c,h));},handleList:async(l,y)=>{let h=l.params.repoName;if(!h){N(y,"Bad request",400);return}let c=e[h];if(!c){N(y,"Repository not found",404);return}let d=c.pageSize??25,p=l.query??{},m=p.cursor,f=p.dir==="prev"?"prev":"next",v=p.ob??"",g=p.od==="desc"?"desc":"asc",b=v?{field:v,dir:g}:void 0,R=parseInt(p.ps??""),x=Number.isFinite(R)&&R>0?Math.min(R,200):d,S=c.listColumns??Object.keys(c.schema.shape),C=c.documentKey??"docId",P=[C,...S.filter(B=>B!==C)],w=c.filterableFields?(()=>{let B=[];for(let Y of c.filterableFields)(Y.includes(".")||S.includes(Y))&&B.push(Y);return B})():S,T=(()=>{let B=[];for(let Y of w)if(Y.includes(".")){let be=gr(c.schema,Y);B.push({name:Y,zodType:be?mt(be):"ZodString",nullable:be?cn(be):false,enumValues:be?ln(be):void 0});}else B.push(...un([Y],c.schema));return B})(),O=new Set(T.map(B=>B.name)),k=yr(p,O),I=mr(k),$;if(m)try{let B=c.repo.ref;typeof B.doc=="function"&&($=await B.doc(m).get());}catch{}let K=await c.repo.query.paginate({pageSize:x,cursor:$,direction:f,...I.length>0?{where:I}:{},...b?{orderBy:[{field:b.field,direction:b.dir}]}:{}}).catch(B=>({queryError:Pe(B,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:k,sort:b})})),_="queryError"in K,te=_?[]:K.data,Tn=_?"":K.nextCursor?.id??"",Cn=_?"":K.prevCursor?.id??"",$n=_?K.queryError:void 0,On=ge(l,t);N(y,st(c.name,te,P,On,{hasPrev:_?false:K.hasPrevPage,hasNext:_?false:K.hasNextPage,prevCursor:Cn,nextCursor:Tn},void 0,T,k,c.allowDelete??false,c.relationalMeta,b,x,$n,c.isGroup));},handleCreateForm:(l,y)=>{let h=l.params.repoName;if(!h){N(y,"Bad request",400);return}let c=e[h];if(!c){N(y,"Repository not found",404);return}let d=ge(l,t),p=he(c.schema,c.createFields),m=J(p),f=`${d}/${c.name}/create`,v=se(m,f,"POST","Create document");N(y,le(c.name,v,"create",null,d));},handleCreateSubmit:async(l,y)=>{let h=l.params.repoName;if(!h){N(y,"Bad request",400);return}let c=e[h];if(!c){N(y,"Repository not found",404);return}let d=ge(l,t),p=l.body??{},m=ft(p,c.schema),f=he(c.schema,c.createFields),v=f.safeParse(m);if(!v.success){let g=J(f),b=`${d}/${c.name}/create`,R=se(g,b,"POST","Create document"),x=v.error.issues.map(S=>`${S.path.join(".")}: ${S.message}`).join(", ");N(y,le(c.name,R,"create",null,d,{type:"error",message:`Validation error: ${x}`}),422);return}try{if(c.isGroup&&c.parentKeys&&c.parentKeys.length>0){let g={...v.data};c.createdKey&&(g[c.createdKey]=new Date);let b=c.parentKeys.filter(C=>!g[C]);if(b.length>0)throw new Error(`Missing parent key(s) for subcollection create: ${b.join(", ")}`);let R=c.parentKeys.map(C=>g[C]),x=c.documentKey??"docId",S=g[x]||pr();await c.repo.set(...R,S,g);}else await c.repo.create(v.data);pt(y,`${d}/${c.name}?flash=created`);}catch(g){let b=he(c.schema,c.createFields),R=J(b),x=`${d}/${c.name}/create`,S=se(R,x,"POST","Create document");N(y,le(c.name,S,"create",null,d,{type:"error",message:`Save error: ${g.message}`}),500);}},handleEditForm:async(l,y)=>{let h=l.params.repoName,c=l.params.id;if(!h||!c){N(y,"Bad request",400);return}let d=e[h];if(!d){N(y,"Repository not found",404);return}let p=ge(l,t),m=null;try{m=await dt(d,c);}catch(x){let S=ut(d,c,x),C=me(x)?424:500;N(y,Me("",{title:`Edit ${d.name} / ${c}`,basePath:p,breadcrumb:[{label:"Repositories",href:p},{label:d.name,href:`${p}/${d.name}`},{label:`Edit ${c}`}],flash:S}),C);return}if(!m){N(y,"Document not found",404);return}let f=dn(m,d.schema),v=he(d.schema,d.mutableFields),g=yt(J(v),f),b=`${p}/${d.name}/${encodeURIComponent(c)}/edit`,R=se(g,b,"POST","Save changes");N(y,le(d.name,R,"edit",c,p));},handleEditSubmit:async(l,y)=>{let h=l.params.repoName,c=l.params.id;if(!h||!c){N(y,"Bad request",400);return}let d=e[h];if(!d){N(y,"Repository not found",404);return}let p=ge(l,t),m=l.body??{},f=ft(m,d.schema),v=he(d.schema,d.mutableFields),b=v.partial().safeParse(f);if(!b.success){let R=Object.fromEntries(Object.entries(m).map(([w,T])=>[w,Array.isArray(T)?T.join(","):T??""])),x=yt(J(v),R),S=`${p}/${d.name}/${encodeURIComponent(c)}/edit`,C=se(x,S,"POST","Save changes"),P=b.error.issues.map(w=>`${w.path.join(".")}: ${w.message}`).join(", ");N(y,le(d.name,C,"edit",c,p,{type:"error",message:`Validation error: ${P}`}),422);return}try{let R=await dt(d,c),x=(R&&sn(R,d.pathKey))??[c];await d.repo.update(...x,b.data),pt(y,`${p}/${d.name}?flash=updated`);}catch(R){let x=he(d.schema,d.mutableFields),S=J(x),C=`${p}/${d.name}/${encodeURIComponent(c)}/edit`,P=se(S,C,"POST","Save changes"),w=me(R)?ut(d,c,R):{type:"error",message:`Save error: ${R.message}`},T=me(R)?424:500;N(y,le(d.name,P,"edit",c,p,w),T);}},handleDelete:async(l,y)=>{let h=l.params.repoName,c=l.params.id;if(!h||!c){N(y,"Bad request",400);return}let d=e[h];if(!d){N(y,"Repository not found",404);return}if(!d.allowDelete){N(y,"Delete is not allowed for this repository",403);return}let p=ge(l,t);try{let m=await dt(d,c),f=(m&&sn(m,d.pathKey))??[c];await d.repo.delete(...f),pt(y,`${p}/${d.name}?flash=deleted`);}catch(m){let f=me(m)?ut(d,c,m):{type:"error",message:`Delete error: ${m.message}`},v=me(m)?424:500;N(y,Me("",{title:`Delete ${d.name} / ${c}`,basePath:p,breadcrumb:[{label:"Repositories",href:p},{label:d.name,href:`${p}/${d.name}`},{label:`Delete ${c}`}],flash:f}),v);}}}}async function hr(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function br(e){let t={};if(!e)return t;for(let n of e.split("&")){let r=n.indexOf("=");if(r===-1)continue;let o=decodeURIComponent(n.slice(0,r).replace(/\+/g," ")),s=decodeURIComponent(n.slice(r+1).replace(/\+/g," ")),a=t[o];a===void 0?t[o]=s:Array.isArray(a)?a.push(s):t[o]=[a,s];}return t}function fn(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:o,middleware:s=[],httpsOptions:a}=e,u=t==="/"?"":t.replace(/\/$/,""),i={};for(let[c,d]of Object.entries(n)){let p=d.schema??d.repo.schema??null;if(!p)throw new Error(`[createAdminServer] Repository "${c}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let m,f,v;if(d.fieldsConfig){let R=d.fieldsConfig;m=[],f=[],v=[];for(let[x,S]of Object.entries(R))for(let C of S)C==="filterable"?m.push(x):C==="mutable"?f.push(x):C==="create"&&v.push(x);m.length===0&&(m=void 0),f.length===0&&(f=void 0),v.length===0&&(v=void 0);}let g=(()=>{let R=d.repo._parentKeys;return R&&R.length>0?R:void 0})();if(g&&v)for(let R of g)v.includes(R)||v.push(R);let b={name:c,path:d.path,repo:d.repo,schema:p,documentKey:d.documentKey??"docId",pathKey:d.repo._pathKey??void 0,isGroup:!!d.repo._isGroup,parentKeys:g,createdKey:d.repo._createdKey??void 0,listColumns:d.listColumns,pageSize:d.pageSize,filterableFields:m,mutableFields:f,createFields:v,allowDelete:d.allowDelete??false,relationalMeta:(()=>{if(!d.relationalFields||d.relationalFields.length===0)return;let R=d.repo.relationalKeys??{},x=[];for(let S of d.relationalFields){let C=R[S.key];C&&x.push({key:S.key,column:S.column,targetRepo:String(C.repo),targetKey:String(C.key),type:C.type});}return x.length>0?x:void 0})()};i[c]=b;}let l=pn(i,u),y=new G;if(r&&y.use(async(c,d,p)=>{let m=c,f=String(m.headers?.["content-type"]??"");if(f.includes("application/x-www-form-urlencoded")){let v=await hr(m);c.body=br(v);}else if(f.includes("application/json")&&typeof m.body=="string")try{c.body=JSON.parse(m.body);}catch{}await p();}),o)if(typeof o=="function")y.use(o);else {let c=o.realm??"Admin",d="Basic "+Buffer.from(`${o.username}:${o.password}`).toString("base64");y.use((p,m,f)=>{if((p.headers?.authorization??"")!==d){m.status(401).set("WWW-Authenticate",`Basic realm="${c}"`).set("Content-Type","text/plain").send("Unauthorized");return}f();});}for(let c of s)y.use(c);y.get(`${u}/`,l.handleDashboard),y.get(`${u}`,l.handleDashboard),y.get(`${u}/:repoName`,l.handleList),y.get(`${u}/:repoName/create`,l.handleCreateForm),y.post(`${u}/:repoName/create`,l.handleCreateSubmit),y.get(`${u}/:repoName/:id/edit`,l.handleEditForm),y.post(`${u}/:repoName/:id/edit`,l.handleEditSubmit),y.post(`${u}/:repoName/:id/delete`,l.handleDelete);let h=async(c,d)=>{await y.handle(c,d);};return a&&(h.httpsOptions=a),h}function ht(e,t,n=200){let r=F(t);e.status(n).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(r));}function Re(e,t,n,r=200){ht(e,{success:true,data:t,meta:n},r);}function M(e,t,n=400){ht(e,{success:false,error:t},n);}function gt(e,t,n,r,o){let s=Pe(t,n),a=s.type==="index",u=a?424:500,l={success:false,error:a?s.message:o&&t instanceof Error?t.message:r};a&&(l.errorType="index",s.indexUrl&&(l.indexUrl=s.indexUrl)),ht(e,l,u);}var yn="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function vr(){let e="";for(let t=0;t<20;t++)e+=yn.charAt(Math.floor(Math.random()*yn.length));return e}function xe(e){let t=e._def??e.def;if(!t)return e;let n=t.typeName??t.type;if(n==="ZodDate"||n==="date")return z.preprocess(r=>Ge(r)??r,e);if(n==="ZodObject"||n==="object"){let r=e.shape,o={};for(let[s,a]of Object.entries(r))o[s]=xe(a);return z.object(o)}if(n==="ZodArray"||n==="array"){let r=t.element??t.type;if(r)return z.array(xe(r))}if(n==="ZodOptional"||n==="optional"){let r=t.innerType;if(r)return xe(r).optional()}if(n==="ZodNullable"||n==="nullable"){let r=t.innerType;if(r)return xe(r).nullable()}if(n==="ZodDefault"||n==="default"){let r=t.innerType,o=t.defaultValue;if(r){let s=xe(r);return typeof o=="function"?s.default(o()):s.default(o)}}return e}function Rr(e,t,n=[]){let r=e.shape,o={},s=t&&t.length>0?t:Object.keys(r);for(let a of s){if(n.includes(a))continue;let u=a.split(".")[0];u&&r[u]&&(o[u]=r[u]);}return z.object(o)}function mn(e,t,n,r=false,o=[]){try{let s=Rr(e,n,o),a=r?s.partial():s;return {success:!0,data:(Le()==="normalize"?xe(a):a).parse(t)}}catch(s){return s instanceof z.ZodError?{success:false,error:`Validation failed: ${s.issues.map(u=>`${u.path.join(".")}: ${u.message}`).join(", ")}`}:{success:false,error:"Validation failed"}}}function xr(e,t){let n=[],r=t?new Set(t):null,o={eq:"==",ne:"!=",lt:"<",lte:"<=",gt:">",gte:">=",in:"in",nin:"not-in",contains:"array-contains",containsAny:"array-contains-any"};for(let[s,a]of Object.entries(e)){if(a===void 0||["cursor","limit","pageSize","orderBy","orderDir","select"].includes(s))continue;let u=Array.isArray(a)?a[0]:a;if(u===void 0||u==="")continue;let i=s.match(/^(\w+)__(\w+)$/),l,y="==";if(i&&i[1]&&i[2]){l=i[1];let c=i[2];if(o[c])y=o[c];else continue}else if(!i)l=s;else continue;if(r&&!r.has(l))continue;let h=u;y==="in"||y==="not-in"||y==="array-contains-any"?h=u.split(",").map(c=>gn(c.trim())):h=gn(u),n.push({field:l,op:y,value:h});}return n}function gn(e){if(e==="true")return true;if(e==="false")return false;if(e==="null")return null;let t=Number(e);return !isNaN(t)&&e!==""?t:e}function Ke(e){return e?{docId:e.id}:null}async function hn(e,t){if(!t||typeof t!="object")return;let n=t.docId;if(typeof n=="string")try{let r=e.repo.ref;if(typeof r.doc!="function")return;let o=await r.doc(n).get();return o.exists?o:void 0}catch{return}}function bn(e,t,n){function r(d,p){return !d||!e[d]?(M(p,`Repository "${d}" not found`,404),null):e[d]}function o(d,p){if(!p)return;let m=d[p];if(typeof m!="string"||!m)return;let f=m.split("/").filter(Boolean),v=[];for(let g=1;g<f.length;g+=2)v.push(f[g]);return v.length>0?v:void 0}async function s(d,p){let m=`by${d.documentKey.charAt(0).toUpperCase()}${d.documentKey.slice(1)}`,f=d.repo.get[m];if(typeof f=="function")try{let g=await f(p);if(g)return g}catch{}return (await d.repo.query.by({where:[[d.documentKey,"==",p]],limit:1}))[0]??null}async function a(d,p){let m=d.params||{},f=r(m.repoName,p);if(!f)return;let v=[],g;try{let b=d.query??{},R=Math.min(Number(b.pageSize)||f.pageSize,100),x=b.cursor,S=b.direction?.toLowerCase()==="prev"?"prev":"next",C=b.orderBy,P=b.orderDir?.toLowerCase()==="desc"?"desc":"asc",w=b.select,T=w?w.split(",").map(_=>_.trim()):void 0,O;f.allowedIncludes&&b.includes&&(O=(typeof b.includes=="string"?b.includes.split(",").map(te=>te.trim()):Array.isArray(b.includes)?b.includes:[]).filter(te=>typeof te=="string"&&f.allowedIncludes.includes(te)),O?.length===0&&(O=void 0));let k=xr(b,f.filterableFields);v=k.map(_=>({field:_.field,op:_.op,value:String(_.value??"")})),C&&(g={field:C,dir:P});let I={pageSize:R,direction:S};if(x)try{let _=typeof x=="string"?JSON.parse(x):x;I.cursor=await hn(f,_);}catch{}C&&(I.orderBy=[{field:C,direction:P}]),k.length>0&&(I.where=k.map(_=>[_.field,_.op,_.value])),T&&(I.select=T),O&&(I.include=O);let $=await f.repo.query.paginate(I),K={items:$.data,hasNextPage:$.hasNextPage,hasPrevPage:$.hasPrevPage,nextCursor:Ke($.nextCursor),prevCursor:Ke($.prevCursor)};Re(p,K,{pageSize:R,hasMore:$.hasNextPage});}catch(b){gt(p,b,{ref:f.repo.ref,path:f.path,isGroup:!!f.isGroup,filters:v,sort:g},"Failed to fetch documents",n);}}async function u(d,p){let m=d.params||{},f=r(m.repoName,p);if(!f)return;let v=[],g;try{let b=d.body??{},R=Math.min(b.pageSize||f.pageSize,100),x=b.direction==="prev"?"prev":"next";b.where&&(v=b.where.map(w=>({field:String(w[0]),op:w[1],value:String(w[2]??"")}))),b.orderBy&&b.orderBy[0]&&(g={field:b.orderBy[0].field,dir:b.orderBy[0].direction==="desc"?"desc":"asc"});let S={pageSize:R,direction:x};if(b.cursor)try{let w=typeof b.cursor=="string"?JSON.parse(b.cursor):b.cursor;S.cursor=await hn(f,w);}catch{}if(f.allowedIncludes&&b.includes&&b.includes.length>0){let w=b.includes.filter(T=>typeof T=="string"?f.allowedIncludes.includes(T):typeof T=="object"&&T!==null&&"relation"in T&&typeof T.relation=="string"?f.allowedIncludes.includes(T.relation):!1);w.length>0&&(S.include=w);}if(b.where&&b.where.length>0){if(f.filterableFields){let w=new Set(f.filterableFields),T=b.where.filter(O=>!w.has(O[0]));if(T.length>0){M(p,`Fields not filterable: ${T.map(O=>O[0]).join(", ")}`,400);return}}S.where=b.where;}if(b.orWhere&&b.orWhere.length>0){if(f.filterableFields){let w=new Set(f.filterableFields),T=b.orWhere.filter(O=>!w.has(O[0]));if(T.length>0){M(p,`Fields not filterable: ${T.map(O=>O[0]).join(", ")}`,400);return}}S.orWhere=b.orWhere;}if(b.orWhereGroups&&b.orWhereGroups.length>0){if(f.filterableFields){let w=new Set(f.filterableFields);for(let T of b.orWhereGroups){let O=T.filter(k=>!w.has(k[0]));if(O.length>0){M(p,`Fields not filterable: ${O.map(k=>k[0]).join(", ")}`,400);return}}}S.orWhereGroups=b.orWhereGroups;}b.orderBy&&b.orderBy.length>0&&(S.orderBy=b.orderBy),b.select&&b.select.length>0&&(S.select=b.select);let C=await f.repo.query.paginate(S),P={items:C.data,hasNextPage:C.hasNextPage,hasPrevPage:C.hasPrevPage,nextCursor:Ke(C.nextCursor),prevCursor:Ke(C.prevCursor)};Re(p,P,{pageSize:R,hasMore:C.hasNextPage});}catch(b){gt(p,b,{ref:f.repo.ref,path:f.path,isGroup:!!f.isGroup,filters:v,sort:g},"Failed to query documents",n);}}async function i(d,p){let m=d.params||{},f=r(m.repoName,p);if(!f)return;let v=m.id;if(!v){M(p,"Document ID required",400);return}try{let g=await s(f,v);if(!g){M(p,"Document not found",404);return}Re(p,g);}catch(g){gt(p,g,{ref:f.repo.ref,path:f.path,isGroup:!!f.isGroup,filters:[{field:f.documentKey,op:"==",value:v}]},"Failed to fetch document",n);}}async function l(d,p){let m=d.params||{},f=r(m.repoName,p);if(f)try{let v=d.body??{},g=mn(f.schema,v,f.createFields,!1,f.systemKeys);if(!g.success){M(p,g.error,400);return}if(f.validate){let R=await f.validate(g.data,"create");if(R){M(p,R,400);return}}let b;if(f.isGroup&&f.parentKeys&&f.parentKeys.length>0){let R={...g.data};f.createdKey&&(R[f.createdKey]=new Date);let x=f.parentKeys.filter(P=>!R[P]);if(x.length>0){M(p,`Missing parent key(s) for subcollection create: ${x.join(", ")}`,400);return}let S=f.parentKeys.map(P=>R[P]),C=R[f.documentKey]||vr();b=await f.repo.set(...S,C,R);}else b=await f.repo.create(g.data);Re(p,b,void 0,201);}catch(v){let g=n&&v instanceof Error?v.message:"Failed to create document";M(p,g,500);}}async function y(d,p,m){let f=d.params||{},v=r(f.repoName,p);if(!v)return;let g=f.id;if(!g){M(p,"Document ID required",400);return}try{let b=d.body??{},R=mn(v.schema,b,v.mutableFields,m,v.systemKeys);if(!R.success){M(p,R.error,400);return}if(v.validate){let P=await v.validate(R.data,"update");if(P){M(p,P,400);return}}let x=await s(v,g),S=(x&&o(x,v.pathKey))??[g],C=await v.repo.update(...S,R.data);Re(p,C);}catch(b){let R=n&&b instanceof Error?b.message:"Failed to update document";M(p,R,500);}}async function h(d,p){let m=d.params||{},f=r(m.repoName,p);if(!f)return;if(!f.allowDelete){M(p,"Delete not allowed for this repository",403);return}let v=m.id;if(!v){M(p,"Document ID required",400);return}try{let g=await s(f,v),b=(g&&o(g,f.pathKey))??[v];await f.repo.delete(...b),Re(p,{deleted:!0});}catch(g){let b=n&&g instanceof Error?g.message:"Failed to delete document";M(p,b,500);}}function c(d,p){p.status(204).set("Access-Control-Allow-Methods","GET, POST, PUT, PATCH, DELETE, OPTIONS").set("Access-Control-Allow-Headers","Content-Type, Authorization").set("Access-Control-Max-Age","86400").send("");}return {handleList:a,handleQuery:u,handleGet:i,handleCreate:l,handleUpdate:y,handleDelete:h,handleOptions:c}}function bt(e){try{return z.toJSONSchema(e,{target:"openapi-3.1",unrepresentable:"any",override:t=>{let n=t.zodSchema?._zod?.def;n&&(n.type==="date"?(t.jsonSchema.type="string",t.jsonSchema.format="date-time"):n.type==="bigint"&&(t.jsonSchema.type="string",t.jsonSchema.format="int64"));}})}catch(t){return typeof console<"u"&&console.warn&&console.warn("[generateOpenAPISpec] Failed to convert Zod schema to JSON Schema; falling back to {type:object}.",t),{type:"object"}}}function H(e){return {$ref:`#/components/schemas/${e}`}}function Q(e){return {description:e,content:{"application/json":{schema:H("ErrorResponse")}}}}function De(e,t){return {description:e,content:{"application/json":{schema:{type:"object",properties:{success:{type:"boolean",enum:[true]},data:t},required:["success","data"]}}}}}function vn(e){return {description:"Paginated list of documents",content:{"application/json":{schema:{type:"object",properties:{success:{type:"boolean",enum:[true]},data:{type:"object",properties:{items:{type:"array",items:e},nextCursor:{oneOf:[{type:"object"},{type:"null"}]},prevCursor:{oneOf:[{type:"object"},{type:"null"}]},hasNextPage:{type:"boolean"},hasPrevPage:{type:"boolean"}},required:["items","hasNextPage","hasPrevPage"]},meta:{type:"object",properties:{pageSize:{type:"integer"},hasMore:{type:"boolean"},cursor:{oneOf:[{type:"string"},{type:"null"}]}}}},required:["success","data"]}}}}}function wr(e){return [{name:"pageSize",in:"query",schema:{type:"integer",default:e.pageSize,maximum:100},description:"Number of items per page"},{name:"cursor",in:"query",schema:{type:"string"},description:"Base64 pagination cursor"},{name:"orderBy",in:"query",schema:{type:"string"},description:"Field name to order by"},{name:"orderDir",in:"query",schema:{type:"string",enum:["asc","desc"]},description:"Order direction"},{name:"select",in:"query",schema:{type:"string"},description:"Comma-separated list of fields to return"}]}function Sr(e){let t=e.filterableFields??Object.keys(e.schema.shape),n=["eq","ne","lt","lte","gt","gte","in","nin","contains"],r=[];for(let o of t){r.push({name:o,in:"query",schema:{type:"string"},description:`Filter by ${o} (equality)`});for(let s of n)r.push({name:`${o}__${s}`,in:"query",schema:{type:"string"},description:`Filter ${o} with operator ${s}`});}return r}function Tr(){return {type:"object",properties:{where:{type:"array",items:{type:"array",items:{},minItems:3,maxItems:3},description:"AND conditions: [field, operator, value][]"},orWhere:{type:"array",items:{type:"array",items:{},minItems:3,maxItems:3},description:"Simple OR conditions (each independently OR'd)"},orWhereGroups:{type:"array",items:{type:"array",items:{type:"array",items:{},minItems:3,maxItems:3}},description:"Advanced OR groups (AND within, OR across groups)"},orderBy:{type:"array",items:{type:"object",properties:{field:{type:"string"},direction:{type:"string",enum:["asc","desc"]}},required:["field"]}},select:{type:"array",items:{type:"string"},description:"Fields to select (projection)"},pageSize:{type:"integer",maximum:100,description:"Number of items per page"},cursor:{oneOf:[{type:"string"},{type:"object"}],description:"Pagination cursor"},direction:{type:"string",enum:["next","prev"],description:"Pagination direction"},includes:{type:"array",items:{oneOf:[{type:"string"},{type:"object",properties:{relation:{type:"string"},select:{type:"array",items:{type:"string"}}},required:["relation"]}]},description:"Relations to include (populate)"}}}}function Cr(e,t,n,r,o){let s={},a=e.name,u=`${t}/${e.name}`,i=`${u}/{${e.documentKey}}`,l={name:e.documentKey,in:"path",required:true,schema:{type:"string"},description:"Unique document identifier"};s[u]={get:{operationId:`list${de(e.name)}`,summary:`List ${e.name} (paginated)`,tags:[a],parameters:[...wr(e),...Sr(e)],responses:{200:vn(H(n)),500:Q("Internal server error")}},post:{operationId:`create${de(e.name)}`,summary:`Create a ${X(e.name)}`,tags:[a],requestBody:{required:true,content:{"application/json":{schema:H(r??n)}}},responses:{201:De("Document created",H(n)),400:Q("Validation error"),500:Q("Internal server error")}}},s[`${u}/query`]={post:{operationId:`query${de(e.name)}`,summary:`Query ${e.name} with advanced filters`,tags:[a],requestBody:{required:true,content:{"application/json":{schema:H("QueryRequestBody")}}},responses:{200:vn(H(n)),400:Q("Invalid query"),500:Q("Internal server error")}}};let y={};return y.get={operationId:`get${de(X(e.name))}`,summary:`Get a single ${X(e.name)}`,tags:[a],parameters:[l],responses:{200:De("Document found",H(n)),404:Q("Document not found"),500:Q("Internal server error")}},y.put={operationId:`update${de(X(e.name))}`,summary:`Update a ${X(e.name)} (full replace)`,tags:[a],parameters:[l],requestBody:{required:true,content:{"application/json":{schema:H(o??n)}}},responses:{200:De("Document updated",H(n)),400:Q("Validation error"),404:Q("Document not found"),500:Q("Internal server error")}},y.patch={operationId:`patch${de(X(e.name))}`,summary:`Partially update a ${X(e.name)}`,tags:[a],parameters:[l],requestBody:{required:true,content:{"application/json":{schema:{allOf:[H(o??n)],description:"All fields are optional for partial updates"}}}},responses:{200:De("Document patched",H(n)),400:Q("Validation error"),404:Q("Document not found"),500:Q("Internal server error")}},e.allowDelete&&(y.delete={operationId:`delete${de(X(e.name))}`,summary:`Delete a ${X(e.name)}`,tags:[a],parameters:[l],responses:{200:De("Document deleted",{type:"object",properties:{id:{type:"string"}}}),404:Q("Document not found"),500:Q("Internal server error")}}),s[i]=y,s}function Rt(e,t,n={}){let{title:r="CRUD API",version:o="1.0.0",description:s,servers:a,auth:u=false}=n,i=t==="/"?"":t.replace(/\/$/,""),l={},y={},h=[];l.ErrorResponse={type:"object",properties:{success:{type:"boolean",enum:[false]},error:{type:"string"}},required:["success","error"]},l.QueryRequestBody=Tr();for(let[m,f]of Object.entries(e)){let v=de(X(m)),g=`${v}Create`,b=`${v}Update`;l[v]=bt(f.schema);let R=T=>{let O=T&&T.length>0?T:Object.keys(f.schema.shape),k={};for(let I of O){let $=I.split(".")[0];$&&f.schema.shape[$]&&!f.systemKeys.includes($)&&(k[$]=f.schema.shape[$]);}return k},x=null,S=R(f.createFields);Object.keys(S).length>0&&(l[g]=bt(z.object(S)),x=g);let C=null,P=R(f.mutableFields);Object.keys(P).length>0&&(l[b]=bt(z.object(P)),C=b);let w=Cr(f,i,v,x,C);Object.assign(y,w),h.push({name:m,description:`Operations on ${m} (collection: ${f.path})`});}let c={},d;return u==="basic"?(c.basicAuth={type:"http",scheme:"basic"},d=[{basicAuth:[]}]):u==="bearer"&&(c.bearerAuth={type:"http",scheme:"bearer",bearerFormat:"JWT"},d=[{bearerAuth:[]}]),{openapi:"3.1.0",info:{title:r,version:o,...s?{description:s}:{}},...a&&a.length>0?{servers:a}:{},paths:y,components:{schemas:l,...Object.keys(c).length>0?{securitySchemes:c}:{}},...d?{security:d}:{},tags:h}}function de(e){return e.charAt(0).toUpperCase()+e.slice(1)}function X(e){return e.endsWith("ies")?e.slice(0,-3)+"y":e.endsWith("ses")||e.endsWith("xes")||e.endsWith("zes")?e.slice(0,-2):e.endsWith("s")&&!e.endsWith("ss")?e.slice(0,-1):e}function $r(e,t){return `<!DOCTYPE html>
1117
+ `;function Ze(){return jsx("script",{dangerouslySetInnerHTML:{__html:bn}})}gt();function ue(e){return "<!DOCTYPE html>"+renderToString(e)}var $r=["corporate","silk","dark"],Ar=()=>jsxs("div",{class:"dropdown dropdown-end","data-frs-theme-switcher":true,children:[jsxs("button",{type:"button",tabIndex:0,class:"btn btn-sm btn-ghost text-neutral-content gap-1.5","aria-label":"Switch theme",children:[jsxs("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24",fill:"none",stroke:"currentColor","stroke-width":"2",class:"size-4",children:[jsx("circle",{cx:"12",cy:"12",r:"4"}),jsx("path",{d:"M12 2v2M12 20v2M4.93 4.93l1.41 1.41M17.66 17.66l1.41 1.41M2 12h2M20 12h2M4.93 19.07l1.41-1.41M17.66 6.34l1.41-1.41"})]}),jsx("span",{class:"text-xs hidden sm:inline","data-frs-theme-current":true,children:"Theme"})]}),jsx("ul",{tabIndex:0,class:"dropdown-content menu menu-sm bg-base-100 text-base-content rounded-box z-50 mt-2 w-40 p-1 shadow-lg border border-base-300",children:$r.map(e=>jsx("li",{children:jsxs("button",{type:"button","data-frs-theme":e,class:"capitalize justify-between",children:[jsx("span",{children:e}),jsx("span",{class:"hidden text-primary","data-frs-theme-check":e,children:"\u2713"})]})},e))})]}),pe=({opts:e,children:t})=>{let{title:n,breadcrumb:r,flash:s,basePath:o="/"}=e;return jsxs("html",{lang:"en","data-theme":"corporate",children:[jsxs("head",{children:[jsx("meta",{charset:"UTF-8"}),jsx("meta",{name:"viewport",content:"width=device-width, initial-scale=1"}),jsxs("title",{children:[n," \u2014 FRS Admin"]}),jsx("script",{dangerouslySetInnerHTML:{__html:"(function(){try{var t=localStorage.getItem('frs-admin-theme');if(t){document.documentElement.setAttribute('data-theme',t);}}catch(_){}})();"}}),jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/themes.css",rel:"stylesheet",type:"text/css"}),jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/daisyui.css",rel:"stylesheet",type:"text/css"}),jsx("script",{src:"https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4"})]}),jsxs("body",{class:"bg-base-200/50 min-h-screen flex flex-col",children:[jsxs("div",{class:"navbar bg-neutral text-neutral-content shadow-sm sticky top-0 z-50 px-6",children:[jsx("div",{class:"flex-1",children:jsx("a",{href:o,class:"font-bold text-lg tracking-tight hover:opacity-80 transition-opacity",children:"FRS Admin"})}),jsx("div",{class:"flex-none",children:jsx(Ar,{})})]}),jsxs("main",{class:"px-6 py-8 w-full flex-1",children:[r&&r.length>0&&jsx("div",{class:"text-sm breadcrumbs mb-4",children:jsx("ul",{children:r.map((a,d)=>a.href?jsx("li",{children:jsx("a",{href:a.href,children:a.label})},d):jsx("li",{class:"text-base-content/60",children:a.label},d))})}),jsx("h1",{class:"text-2xl font-bold mb-6",children:n}),s&&jsxs("div",{role:"alert",class:`alert ${s.type==="success"?"alert-success":s.type==="warning"?"alert-warning":"alert-error"} mb-6`,children:[jsx("span",{class:"flex-1",children:s.message}),s.action&&jsx("a",{href:s.action.href,...s.action.external?{target:"_blank",rel:"noopener noreferrer"}:{},class:"btn btn-sm btn-outline",children:s.action.label})]}),t]}),jsx(yt,{}),jsx(Ze,{})]})]})};function ht(e,t){return ue(jsx(pe,{opts:t,children:jsx("div",{dangerouslySetInnerHTML:{__html:e}})}))}function bt(e,t){return ue(jsx(pe,{opts:{title:"Repositories",basePath:t},children:e.length===0?jsxs("div",{class:"text-center py-20 text-base-content/50",children:[jsx("p",{class:"text-lg font-medium mb-1",children:"No repositories configured"}),jsx("p",{class:"text-sm",children:"Add a repository to your FRS config to get started."})]}):jsx("div",{class:"grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4",children:e.map(n=>jsx("a",{href:`${t}/${n.name}`,class:"card bg-base-100 border border-base-300 hover:shadow-md no-underline transition-shadow",children:jsxs("div",{class:"card-body p-5",children:[jsx("h2",{class:"card-title text-sm font-semibold",children:n.name}),jsx("p",{class:"text-xs text-base-content/50 font-mono",children:n.path})]})},n.name))})}))}var wn=[{value:"==",label:"="},{value:"!=",label:"\u2260"},{value:"in",label:"in"},{value:"not-in",label:"not in"}],Or=[{value:"==",label:"="},{value:"!=",label:"\u2260"},{value:"<",label:"<"},{value:"<=",label:"\u2264"},{value:">",label:">"},{value:">=",label:"\u2265"},{value:"in",label:"in"},{value:"not-in",label:"not in"}],Pr=[{value:"array-contains",label:"contains"},{value:"array-contains-any",label:"contains any"}];function Er(e){switch(e){case "ZodNumber":case "ZodBigInt":case "ZodDate":return Or;case "ZodBoolean":return wn;case "ZodArray":return Pr;default:return wn}}var fe="__null__";function Ir(e){return `(function(cb){var i=document.getElementById('${e}');if(!i)return;if(cb.checked){i.dataset._prev=i.value;if(i.tagName==='SELECT'){var o=i.querySelector('option[value="${fe}"]');if(!o){o=document.createElement('option');o.value='${fe}';o.textContent='\u2205 null';o.dataset._auto='1';i.appendChild(o);}o.selected=true;}else{if(i.type==='number'||i.type==='datetime-local'){i.dataset._type=i.type;i.type='text';}i.value='${fe}';i.readOnly=true;}i.style.opacity='0.55';}else{i.style.opacity='';if(i.tagName==='SELECT'){var o2=i.querySelector('option[value="${fe}"][data-_auto="1"]');if(o2)o2.remove();var prev=i.dataset._prev||'';for(var k=0;k<i.options.length;k++)i.options[k].selected=(i.options[k].value===prev);}else{if(i.dataset._type){i.type=i.dataset._type;delete i.dataset._type;}i.readOnly=false;i.value=(i.dataset._prev&&i.dataset._prev!=='${fe}')?i.dataset._prev:'';}}})(this)`}function Dr(e,t){return `(function(){var h=document.getElementById('${e}');var boxes=document.querySelectorAll('input[data-enum-group="${t}"]');h.value=Array.from(boxes).filter(function(b){return b.checked;}).map(function(b){return b.value;}).join(',');})()`}function Be({inputId:e,active:t}){return jsxs("label",{class:"flex items-center gap-1 cursor-pointer select-none text-xs text-base-content/60 hover:text-base-content border border-base-300 rounded-md px-1.5 py-1 shrink-0 leading-none h-8",title:"Filter where field IS NULL",children:[jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",checked:t,onchange:Ir(e)}),jsx("span",{children:"\u2205"})]})}function Nr({col:e,active:t}){let n=t?.value??"",r=n===fe,s=`fv_input_${e.name.replace(/\./g,"__")}`,o=t?.op,a=o==="in"||o==="not-in";if(e.enumValues&&e.enumValues.length>0){if(a){let d=new Set(n.split(",").map(f=>f.trim()).filter(Boolean)),i=`eg_${e.name.replace(/\./g,"__")}`;return jsxs("div",{class:"flex flex-wrap items-center gap-1 w-full",children:[jsx("input",{type:"hidden",id:s,name:`fv_${e.name}`,value:n}),e.enumValues.map(f=>jsxs("label",{class:"flex items-center gap-1 text-xs border border-base-300 rounded px-2 cursor-pointer hover:bg-base-200",children:[jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",value:f,checked:d.has(f),"data-enum-group":i,onchange:Dr(s,i)}),jsx("span",{children:f})]},f))]})}return jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxs("select",{id:s,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),e.enumValues.map(d=>jsx("option",{value:d,selected:n===d,children:d},d)),e.nullable&&jsx("option",{value:fe,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsx(Be,{inputId:s,active:r})]})}if(e.zodType==="ZodBoolean")return jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxs("select",{id:s,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),jsx("option",{value:"true",selected:n==="true",children:"true"}),jsx("option",{value:"false",selected:n==="false",children:"false"}),e.nullable&&jsx("option",{value:fe,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsx(Be,{inputId:s,active:r})]});if(e.zodType==="ZodArray"){let d=t?.op==="array-contains-any";return jsx("input",{id:s,type:"text",name:`fv_${e.name}`,value:n,placeholder:d?"val1, val2, \u2026":"value",class:"input input-sm input-bordered w-full"})}return e.zodType==="ZodNumber"||e.zodType==="ZodBigInt"?jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsx("input",{id:s,type:r?"text":"number",name:`fv_${e.name}`,value:n,placeholder:"value",class:"input input-sm input-bordered w-full",readOnly:r,style:r?"opacity:0.55":void 0,"data-_type":r?"number":void 0}),e.nullable&&jsx(Be,{inputId:s,active:r})]}):e.zodType==="ZodDate"?jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsx("input",{id:s,type:r?"text":"datetime-local",name:`fv_${e.name}`,value:n,class:"input input-sm input-bordered w-full",readOnly:r,style:r?"opacity:0.55":void 0,"data-_type":r?"datetime-local":void 0}),e.nullable&&jsx(Be,{inputId:s,active:r})]}):jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsx("input",{id:s,type:"text",name:`fv_${e.name}`,value:n,placeholder:"value",class:"input input-sm input-bordered w-full",readOnly:r,style:r?"opacity:0.55":void 0}),e.nullable&&jsx(Be,{inputId:s,active:r})]})}function vt({action:e,columnMeta:t,activeFilters:n,isGroup:r}){let s=Object.fromEntries(n.map(i=>[i.field,i])),o=n.length>0,a=n.length>=2||r&&o,d=t.filter(i=>i.zodType!=="ZodObject"&&i.zodType!=="ZodRecord");return jsxs("details",{class:"collapse collapse-arrow bg-base-100 border border-base-300 rounded-box mb-6 shadow-sm",open:o?true:void 0,children:[jsxs("summary",{class:"collapse-title text-sm font-medium py-2 min-h-0",children:["Filters",o&&jsxs("span",{class:"badge badge-primary badge-sm ml-2",children:[n.length," active"]})]}),jsx("div",{class:"collapse-content pb-4 pt-2",children:jsxs("form",{method:"get",action:e,children:[jsx("div",{class:"grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4",children:d.map(i=>{let f=Er(i.zodType),g=s[i.name],x=g?.op??f[0].value;return jsxs("div",{class:"flex flex-col gap-1.5",children:[jsx("label",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:i.name}),jsxs("div",{class:"flex gap-1.5",children:[f.length>1?jsx("select",{name:`fo_${i.name}`,class:"select select-sm select-bordered w-20 shrink-0",children:f.map(h=>jsx("option",{value:h.value,selected:h.value===x,children:h.label},h.value))}):jsx("input",{type:"hidden",name:`fo_${i.name}`,value:f[0].value}),jsx(Nr,{col:i,active:g})]})]},i.name)})}),jsxs("div",{class:"flex flex-wrap gap-2 mt-4 pt-3 border-t border-base-200 items-center",children:[jsx("button",{type:"submit",class:"btn btn-sm btn-primary",children:"Apply"}),o&&jsx("a",{href:e,class:"btn btn-sm btn-ghost",children:"Clear"}),a&&jsxs("span",{class:"text-xs text-warning ml-auto flex items-center gap-1",children:[jsx("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:jsx("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"})}),r?"Collection group queries require a composite index":"Multiple filters may require a composite index"]})]})]})})]})}function xt(e,t,n,r,s,o){let a=n==="create"?`Create ${e}`:`Edit ${e} / ${r??""}`,d=n==="create"?[{label:"Repositories",href:s},{label:e,href:`${s}/${e}`},{label:"New document"}]:[{label:"Repositories",href:s},{label:e,href:`${s}/${e}`},{label:`Edit ${r??""}`}];return ue(jsx(pe,{opts:{title:a,breadcrumb:d,basePath:s,flash:o},children:jsx("div",{class:"card bg-base-100 border border-base-300",children:jsx("div",{class:"card-body p-6",children:jsx("div",{dangerouslySetInnerHTML:{__html:t}})})})}))}et();mt();function Rt(e,t,n){let r=new URLSearchParams;for(let s of e)r.set(`fv_${s.field}`,s.value),r.set(`fo_${s.field}`,s.op);return t&&(r.set("ob",t.field),r.set("od",t.dir)),n&&r.set("ps",String(n)),r}function Sn(e,t,n,r,s){let o=Rt(e,r,s);return o.set("cursor",t),o.set("dir",n),`?${o.toString()}`}function _r(e,t,n,r){let s=Rt(n,void 0,r);return t?.field===e?t.dir==="asc"&&(s.set("ob",e),s.set("od","desc")):(s.set("ob",e),s.set("od","asc")),`?${s.toString()}`}function jr(e,t,n){return `?${Rt(t,n,e).toString()}`}function wt(e,t,n,r,s,o,a=[],d=[],i=false,f=[],g,x,h,b,m,y,c){let p=`${r}/${e}`,u=`${p}/create`,l={};if(c)for(let w of n)l[w]=Ae(Oe(c,w));let v=(y??[]).filter(w=>{let S=l[w]??Ae(Oe(c,w));return S==="string"||S==="number"||S==="bigint"||S==="boolean"||S==="enum"||S==="literal"}),R=v.length>0,T=i,$=T||R,O=v.map(w=>{let S=Oe(c,w),k=Ae(S),A=a.find(E=>E.name===w);return {name:w,type:k,enumValues:A?.enumValues??null,nullable:A?.nullable??false}});return ue(jsxs(pe,{opts:{title:e,breadcrumb:[{label:"Repositories",href:r},{label:e}],basePath:r,flash:o},children:[a.length>0&&jsx(vt,{action:p,columnMeta:a,activeFilters:d,isGroup:b}),h&&jsxs("div",{role:"alert",class:`alert ${h.type==="index"?"alert-warning":"alert-error"} mb-6 shadow-sm`,children:[jsx("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-6 w-6 shrink-0 stroke-current",fill:"none",viewBox:"0 0 24 24",children:h.type==="index"?jsx("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z"}):jsx("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z"})}),jsxs("div",{class:"flex-1",children:[jsx("h3",{class:"font-bold",children:h.type==="index"?"Composite index required":"Query failed"}),jsx("div",{class:"text-sm",children:h.message})]}),h.indexUrl&&jsx("a",{href:h.indexUrl,target:"_blank",rel:"noopener noreferrer",class:"btn btn-sm btn-outline",children:"Create Index \u2192"})]}),jsxs("div",{class:"flex flex-wrap justify-between items-center mb-4 gap-3",children:[jsxs("div",{class:"flex items-center gap-3",children:[jsxs("span",{class:"text-sm text-base-content/60",children:[t.length,typeof m=="number"&&jsxs(Fragment,{children:[" ","of ",jsx("span",{class:"font-semibold text-base-content/80",children:m})]})," ","document",(typeof m=="number"?m:t.length)!==1&&"s"]}),jsxs("div",{class:"flex items-center gap-1.5 text-sm text-base-content/60",children:[jsx("span",{children:"Rows"}),jsx("div",{class:"join",children:[10,25,50,100].map(w=>jsx("a",{href:jr(w,d,g),class:`join-item btn btn-xs ${x===w?"btn-active btn-primary":"btn-outline"}`,children:w},w))})]})]}),jsx("a",{href:u,class:"btn btn-primary btn-sm",children:"+ New"})]}),$&&jsxs("div",{class:"hidden mb-3 alert alert-info py-2 px-3","data-frs-bulk-bar":true,"data-frs-repo":e,"data-frs-total":typeof m=="number"?String(m):"","data-frs-page-size":String(x??t.length),"data-frs-allow-delete":T?"1":"0","data-frs-allow-update":R?"1":"0","data-frs-fields":JSON.stringify(O),"data-frs-filters":JSON.stringify(d),children:[jsxs("div",{class:"flex-1 text-sm",children:[jsx("span",{"data-frs-bulk-summary":true,children:"0 selected"}),typeof m=="number"&&m>t.length&&jsxs("button",{type:"button",class:"ml-3 link link-primary text-sm hidden","data-frs-bulk-select-all":true,children:["Select all ",m," matching documents"]}),jsxs("span",{class:"hidden ml-3 italic","data-frs-bulk-all-active":true,children:["All ",m??"?"," matching documents are selected."," ",jsx("button",{type:"button",class:"link","data-frs-bulk-clear":true,children:"Clear selection"})]})]}),jsxs("div",{class:"flex gap-2",children:[R&&jsx("button",{type:"button",class:"btn btn-sm btn-outline","data-frs-bulk-action":"update",children:"Update field\u2026"}),T&&jsx("button",{type:"button",class:"btn btn-sm btn-error btn-outline","data-frs-bulk-action":"delete",children:"Delete"})]})]}),jsx("div",{class:"overflow-x-auto rounded-box border border-base-300 bg-base-100","data-frs-table-wrap":true,children:jsxs("table",{class:"table table-sm w-full","data-frs-table":true,"data-frs-repo":e,"data-frs-colcount":n.length,children:[jsx("thead",{children:jsxs("tr",{class:"bg-base-200/50",children:[$&&jsx("th",{class:"w-8",children:jsx("input",{type:"checkbox",class:"checkbox checkbox-xs checkbox-primary","data-frs-select-page":true,"aria-label":"Select all on this page"})}),[...n].map((w,S)=>{let k=g?.field===w,A=k?g.dir==="asc"?" \u25B2":" \u25BC":"";return jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:jsxs("a",{href:_r(w,g,d,x),class:`hover:text-base-content inline-flex items-center gap-0.5${k?" text-primary font-bold":""}`,children:[w,A]})},S)}),f.map((w,S)=>jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:w.column},`rel-${S}`)),jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide text-right",children:"Actions"})]})}),jsx("tbody",{children:t.length===0?jsx("tr",{children:jsx("td",{colspan:n.length+f.length+1+($?1:0),class:"text-center py-16 text-base-content/40",children:"No documents found"})}):t.map((w,S)=>{let k=String(w.docId??w.id??""),A=`${r}/${e}/${encodeURIComponent(k)}/edit`,E=`${r}/${e}/${encodeURIComponent(k)}/delete`;return jsxs("tr",{class:"hover","data-frs-row-id":k,children:[$&&jsx("td",{class:"align-middle py-2",children:jsx("input",{type:"checkbox",class:"checkbox checkbox-xs checkbox-primary","data-frs-select-row":true,value:k,"aria-label":`Select ${k}`})}),n.map((C,B)=>{let D=w[C],_=l[C],oe=_?Me(_,D):null;return jsx("td",{class:"align-top py-2",children:jsx($e,{val:D,mismatch:oe})},B)}),f.map((C,B)=>{let D=w[C.key];if(D==null||D==="")return jsx("td",{class:"py-2"},`rel-${B}`);let _=C.type==="one"?`${r}/${C.targetRepo}/${encodeURIComponent(String(D))}/edit`:`${r}/${C.targetRepo}?fv_${C.targetKey}=${encodeURIComponent(String(D))}`;return jsx("td",{class:"align-middle py-2",children:jsx("a",{href:_,class:"btn btn-xs btn-ghost btn-outline","data-frs-relation":true,"data-frs-rel-type":C.type,"data-frs-rel-repo":C.targetRepo,"data-frs-rel-fk":C.targetKey,"data-frs-rel-val":String(D),"data-frs-rel-label":C.column,children:C.column})},`rel-${B}`)}),jsx("td",{class:"align-middle text-right whitespace-nowrap py-2",children:jsxs("div",{class:"flex gap-1 justify-end",children:[jsx("a",{href:A,class:"btn btn-xs btn-outline",children:"Edit"}),i&&jsx("form",{method:"post",action:E,onsubmit:"return confirm('Delete this document?')",children:jsx("button",{type:"submit",class:"btn btn-xs btn-error btn-outline",children:"Delete"})})]})})]},S)})})]})}),(s.hasPrev||s.hasNext)&&jsxs("div",{class:"flex flex-col items-center mt-6 gap-2",children:[jsxs("div",{class:"flex justify-center items-center gap-2",children:[s.hasPrev?jsx("a",{href:Sn(d,s.prevCursor,"prev",g,x),class:"btn btn-sm btn-outline",children:"\u2190 Previous"}):jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"\u2190 Previous"}),s.hasNext?jsx("a",{href:Sn(d,s.nextCursor,"next",g,x),class:"btn btn-sm btn-outline",children:"Next \u2192"}):jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"Next \u2192"})]}),typeof m=="number"&&jsxs("div",{class:"text-xs text-base-content/50",children:[m," total document",m!==1?"s":"",d.length>0?" matching filters":""]})]}),$&&R&&jsxs("dialog",{id:"frs-bulk-update-modal",class:"modal",children:[jsxs("div",{class:"modal-box",children:[jsx("h3",{class:"font-bold text-lg mb-3",children:"Bulk update field"}),jsx("p",{class:"text-sm text-base-content/60 mb-4","data-frs-bulk-update-summary":true,children:"Update one field on the selected documents."}),jsxs("form",{method:"dialog","data-frs-bulk-update-form":true,children:[jsxs("label",{class:"form-control w-full mb-3",children:[jsx("div",{class:"label",children:jsx("span",{class:"label-text text-xs uppercase tracking-wide",children:"Field"})}),jsxs("select",{class:"select select-bordered select-sm w-full",name:"field",required:true,"data-frs-bulk-field-select":true,children:[jsx("option",{value:"",children:"\u2014 Select a field \u2014"}),v.map(w=>jsx("option",{value:w,children:w},w))]})]}),jsx("div",{class:"mb-4","data-frs-bulk-value-container":true}),jsxs("div",{class:"modal-action",children:[jsx("button",{type:"button",class:"btn btn-sm btn-ghost","data-frs-bulk-update-cancel":true,children:"Cancel"}),jsx("button",{type:"submit",class:"btn btn-sm btn-primary","data-frs-bulk-update-submit":true,children:"Apply"})]})]})]}),jsx("form",{method:"dialog",class:"modal-backdrop",children:jsx("button",{children:"close"})})]})]}))}function nt(e,t){return ht(e,t)}function St(e,t){return bt(e,t)}function Tt(e,t,n,r,s,o,a,d,i,f,g,x,h,b,m,y,c){return wt(e,t,n,r,s,o,a,d,i,f,g,x,h,b,m,y,c)}function me(e,t,n,r,s,o){return xt(e,t,n,r,s,o)}var Fr=new Set(["<","<=",">",">=","!="]),Zr=new Set(["array-contains","array-contains-any"]);function kt(e){return e==="desc"?"DESCENDING":"ASCENDING"}function Mr(e){let t=e.split("/").filter(Boolean);return t[t.length-1]??e}function Br(e,t,n,r,s){let o=[],a=new Set;for(let i of r)if(i.op==="=="||i.op==="in"||i.op==="not-in"){if(a.has(i.field))continue;a.add(i.field),o.push({fieldPath:i.field,order:"ASCENDING"});}for(let i of r)if(Zr.has(i.op)){if(a.has(i.field))continue;a.add(i.field),o.push({fieldPath:i.field,arrayConfig:"CONTAINS"});}for(let i of r)if(Fr.has(i.op)){if(a.has(i.field))continue;a.add(i.field);let f=s?.field===i.field?kt(s.dir):"ASCENDING";o.push({fieldPath:i.field,order:f});}if(s&&!a.has(s.field)&&o.push({fieldPath:s.field,order:kt(s.dir)}),o.length===1&&n)return Ur(e,t,o[0]);let d=s&&o.some(i=>i.fieldPath===s.field)?kt(s.dir):"ASCENDING";return o.push({fieldPath:"__name__",order:d}),qr(e,t,n,o)}function qr(e,t,n,r,s="(default)"){let o=`projects/${e}/databases/${s}/collectionGroups/${t}/indexes/_`,a=[...At(1,o),...rt(2,n?2:1)];for(let f of r)a.push(...Tn(3,kn(f)));let d=s==="(default)"?"-default-":s,i=encodeURIComponent(Cn(a));return `https://console.firebase.google.com/project/${e}/firestore/databases/${d}/indexes?create_composite=${i}`}function Kr(e){return e.match(/https:\/\/console\.firebase\.google\.com[^\s)"]*/)?.[0]}function Ct(e){let t=[],n=e>>>0;for(;n>=128;)t.push(n&127|128),n>>>=7;return t.push(n&127),t}function $t(e,t){return e<<3|t}function At(e,t){let n=Array.from(new TextEncoder().encode(t));return [$t(e,2),...Ct(n.length),...n]}function rt(e,t){return [$t(e,0),...Ct(t)]}function Tn(e,t){return [$t(e,2),...Ct(t.length),...t]}function kn(e){let t=[...At(1,e.fieldPath)];return e.arrayConfig==="CONTAINS"?t.push(...rt(3,1)):t.push(...rt(2,e.order==="DESCENDING"?2:1)),t}function Cn(e){let t=String.fromCharCode(...e),n;if(typeof Buffer<"u")n=Buffer.from(e).toString("base64");else if(typeof btoa<"u")n=btoa(t);else throw new Error("No base64 encoder available");return n.replace(/=+$/,"")}function Ur(e,t,n,r="(default)"){let s=`projects/${e}/databases/${r}/collectionGroups/${t}/fields/${n.fieldPath}`,o=[...At(1,s),...rt(2,2),...Tn(3,kn(n))],a=r==="(default)"?"-default-":r,d=encodeURIComponent(Cn(o));return `https://console.firebase.google.com/project/${e}/firestore/databases/${a}/indexes/automatic?create_exemption=${d}`}function Lr(e){let t=e,n=[t?.firestore?.projectId,t?.firestore?.app?.options?.projectId,t?.firestore?._settings?.projectId,t?.firestore?.databaseId?.projectId,t?._firestore?.projectId];for(let s of n)if(typeof s=="string"&&s.length>0)return s;return process.env.GCLOUD_PROJECT||process.env.GOOGLE_CLOUD_PROJECT||process.env.FIREBASE_PROJECT_ID||void 0}function we(e){let t=e;return t?t.code===9?true:typeof t.message=="string"?t.message.includes("requires an index"):false:false}function qe(e,t){let n=e??{},r=we(e),s;if(r&&(s=n.message?Kr(n.message):void 0,!s)){let o=Lr(t.ref);if(o){let a=Mr(t.path);s=Br(o,a,t.isGroup,t.filters,t.sort);}}return {type:r?"index":"error",message:r?"This query requires a composite index that does not exist yet.":n.message??"Query failed",indexUrl:s}}var $n="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function Hr(){let e="";for(let t=0;t<20;t++)e+=$n.charAt(Math.floor(Math.random()*$n.length));return e}function Ot(e,t){if(!t)return;let n=e[t];if(typeof n!="string"||!n)return;let r=n.split("/").filter(Boolean),s=[];for(let o=1;o<r.length;o+=2)s.push(r[o]);return s.length>0?s:void 0}async function Ke(e,t){let n=e.documentKey??"docId",r=`by${n.charAt(0).toUpperCase()}${n.slice(1)}`;if(typeof e.repo.get[r]=="function")try{let o=await e.repo.get[r](t);if(o)return o}catch{}return (await e.repo.query.by({where:[[n,"==",t]],limit:1}))[0]??null}function Pt(e,t,n){let r=e.documentKey??"docId",s=qe(n,{ref:e.repo.ref,path:e.path,isGroup:!!e.isGroup,filters:[{field:r,op:"==",value:t}]});return s.type==="index"?{type:"warning",message:"Loading this document requires a composite index that does not exist yet.",...s.indexUrl?{action:{href:s.indexUrl,label:"Create Index \u2192",external:true}}:{}}:{type:"error",message:s.message}}function z(e,t,n=200){e.status(n).set("Content-Type","text/html; charset=utf-8").send(t);}function Et(e,t){e.status(302).set("Location",t).send("");}function It(e,t){let n=t.shape,r={};for(let[s,o]of Object.entries(n)){let a=Nt(o);if(a==="ZodObject"){if(e[s+"__isnull"]==="1"){r[s]=null;continue}let f={},g=false;for(let[b,m]of Object.entries(e))b.startsWith(`${s}.`)&&(f[b.slice(s.length+1)]=m,g=true);if(g){let b=o;for(;;){let m=M(b);if(m==="ZodOptional"||m==="ZodNullable"||m==="ZodDefault")b=U(b);else break}r[s]=It(f,b);continue}let x=e[s],h=Array.isArray(x)?x[x.length-1]:x;if(h)try{r[s]=JSON.parse(h);}catch{r[s]=h;}continue}let d=e[s],i=Array.isArray(d)?d[d.length-1]:d;if(e[s+"__isnull"]==="1"){r[s]=null;continue}if(i===void 0||i===""){a==="ZodBoolean"&&(r[s]=false);continue}switch(a){case "ZodBoolean":i==="__null__"?r[s]=null:r[s]=i==="true"||i==="on"||i==="1";break;case "ZodNumber":case "ZodBigInt":r[s]=Number(i);break;case "ZodDate":r[s]=new Date(i);break;case "ZodArray":try{r[s]=JSON.parse(i);}catch{r[s]=i;}break;default:if(i.startsWith("{")||i.startsWith("["))try{r[s]=JSON.parse(i);break}catch{}r[s]=i;}}return r}function An(e){let t=null;if(e instanceof Date)t=e;else if(typeof e=="object"&&e!==null&&typeof e.toDate=="function")t=e.toDate();else if(typeof e=="object"&&e!==null&&"_seconds"in e&&"_nanoseconds"in e)t=new Date(e._seconds*1e3+Math.floor(e._nanoseconds/1e6));else if(typeof e=="string"||typeof e=="number"){let r=new Date(e);isNaN(r.getTime())||(t=r);}if(!t||isNaN(t.getTime()))return null;let n=r=>String(r).padStart(2,"0");return `${t.getFullYear()}-${n(t.getMonth()+1)}-${n(t.getDate())}T${n(t.getHours())}:${n(t.getMinutes())}`}function Nt(e){let t=e;for(;;){let n=M(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=U(t);else return n}}function Gr(e){let t=e;for(;;){let n=M(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=U(t);else return t}}function Pn(e){let t=e;for(;;){let n=M(t);if(n==="ZodOptional"||n==="ZodNullable")return true;if(n==="ZodDefault"){t=U(t);continue}return false}}function En(e){let t=Gr(e),n=M(t);if(n==="ZodEnum"){let r=Ne(t);return r.length>0?r:void 0}if(n==="ZodNativeEnum"){let r=_e(t),s=Object.values(r).filter(o=>typeof o=="string");return s.length>0?s:void 0}if(n==="ZodLiteral"){let r=Je(t);return typeof r=="string"?[r]:void 0}}function In(e,t,n=""){let r={};for(let s of Object.keys(t.shape)){let o=n?`${n}.${s}`:s,a=e[s];if(a===null){r[o]="__null__";continue}if(a===void 0)continue;let d=t.shape[s];for(;;){let f=M(d);if(f==="ZodOptional"||f==="ZodNullable"||f==="ZodDefault")d=U(d);else break}let i=M(d);if(i==="ZodObject"&&typeof a=="object"&&a!==null&&!Array.isArray(a)){let f=In(a,d,o);Object.assign(r,f);}else if(i==="ZodDate"){let f=An(a);f!==null&&(r[o]=f);}else if(typeof a=="object"&&a!==null&&!Array.isArray(a)&&("_seconds"in a||typeof a.toDate=="function")){let f=An(a);r[o]=f??JSON.stringify(a,null,2);}else typeof a=="object"?r[o]=JSON.stringify(a,null,2):r[o]=String(a);}return r}function Dt(e,t){return e.map(n=>({...n,defaultValue:t[n.name]??n.defaultValue,nested:n.nested?Dt(n.nested,t):void 0}))}function Wr(e,t){let n=new Set(["==","!=","<","<=",">",">=","in","not-in","array-contains","array-contains-any"]),r=[];for(let[s,o]of Object.entries(e)){if(!s.startsWith("fv_"))continue;let a=s.slice(3);if(!t.has(a))continue;let d=(o??"").trim();if(!d)continue;let i=e[`fo_${a}`]??"==",f=n.has(i)?i:"==";r.push({field:a,op:f,value:d});}return r}function On(e){let t="__null__",n=r=>r===t?null:r==="true"?true:r==="false"?false:r!==""&&!isNaN(Number(r))?Number(r):r;return e.map(r=>{if(r.op==="array-contains-any"||r.op==="in"||r.op==="not-in"){let s=r.value.split(",").map(o=>o.trim()).filter(o=>o!==""&&o!==t).map(o=>n(o));return [r.field,r.op,s]}return [r.field,r.op,n(r.value)]})}function Dn(e,t,n=""){let r=[];for(let s of e){let o=n?`${n}.${s}`:s,a=t.shape[s];if(!a){r.push({name:o,zodType:"ZodString"});continue}let d=Nt(a);if(d==="ZodObject"){let i=a;for(;;){let g=M(i);if(g==="ZodOptional"||g==="ZodNullable"||g==="ZodDefault")i=U(i);else break}let f=J(i);r.push(...Dn(Object.keys(f),i,o));}else r.push({name:o,zodType:d,nullable:Pn(a),enumValues:En(a)});}return r}function Vr(e,t){let n=t.split("."),r=e;for(let s of n){for(;;){let a=M(r);if(a==="ZodOptional"||a==="ZodNullable"||a==="ZodDefault")r=U(r);else break}let o=J(r);if(!(s in o))return null;r=o[s];}return r}function Se(e,t){if(!t||t.length===0)return e;let n=[],r=new Map;for(let o of t){let a=o.indexOf(".");if(a===-1)n.push(o);else {let d=o.slice(0,a),i=o.slice(a+1);r.has(d)||r.set(d,[]),r.get(d).push(i);}}let s={};for(let o of n)o in e.shape&&(s[o]=e.shape[o]);for(let[o,a]of r){if(!(o in e.shape))continue;let d=e.shape[o];for(;;){let i=M(d);if(i==="ZodOptional"||i==="ZodNullable"||i==="ZodDefault")d=U(d);else break}if(M(d)!=="ZodObject"){s[o]=e.shape[o];continue}s[o]=Se(d,a);}return z$1.object(s)}function ye(e,t){let n=t==="/"?"":t.replace(/\/$/,"");if(process.env.FUNCTIONS_EMULATOR==="true"){let o=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??"demo-project",a=process.env.FUNCTION_REGION??"us-central1",d=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${o}/${a}/${d}${n}`}let r=process.env.K_SERVICE,s=e.hostname??e.headers?.host??"";return r&&s.includes("cloudfunctions.net")?`/${r.toLowerCase()}${n}`:n}function Nn(e,t){let n=(c,p)=>{let u=ye(c,t),l=Object.values(e).map(v=>({name:v.name,path:v.path}));z(p,St(l,u));},r=async(c,p)=>{let u=c.params.repoName;if(!u){z(p,"Bad request",400);return}let l=e[u];if(!l){z(p,"Repository not found",404);return}let v=l.pageSize??25,R=c.query??{},T=R.cursor,$=R.dir==="prev"?"prev":"next",O=R.ob??"",w=R.od==="desc"?"desc":"asc",S=O?{field:O,dir:w}:void 0,k=parseInt(R.ps??""),A=Number.isFinite(k)&&k>0?Math.min(k,200):v,E=l.listColumns??Object.keys(l.schema.shape),C=l.documentKey??"docId",B=[C,...E.filter(H=>H!==C)],D=l.filterableFields?(()=>{let H=[];for(let ae of l.filterableFields)(ae.includes(".")||E.includes(ae))&&H.push(ae);return H})():E,_=(()=>{let H=[];for(let ae of D)if(ae.includes(".")){let ke=Vr(l.schema,ae);H.push({name:ae,zodType:ke?Nt(ke):"ZodString",nullable:ke?Pn(ke):false,enumValues:ke?En(ke):void 0});}else H.push(...Dn([ae],l.schema));return H})(),oe=new Set(_.map(H=>H.name)),Ie=Wr(R,oe),Qe=On(Ie),Bt;if(T)try{let H=l.repo.ref;typeof H.doc=="function"&&(Bt=await H.doc(T).get());}catch{}let[he,Qn]=await Promise.all([l.repo.query.paginate({pageSize:A,cursor:Bt,direction:$,...Qe.length>0?{where:Qe}:{},...S?{orderBy:[{field:S.field,direction:S.dir}]}:{}}).catch(H=>({queryError:qe(H,{ref:l.repo.ref,path:l.path,isGroup:!!l.isGroup,filters:Ie,sort:S})})),l.repo.aggregate.count(Qe.length>0?{where:Qe}:{}).catch(()=>{})]),Te="queryError"in he,Hn=Te?[]:he.data,Gn=Te?"":he.nextCursor?.id??"",Wn=Te?"":he.prevCursor?.id??"",Vn=Te?he.queryError:void 0,Jn=ye(c,t);z(p,Tt(l.name,Hn,B,Jn,{hasPrev:Te?false:he.hasPrevPage,hasNext:Te?false:he.hasNextPage,prevCursor:Wn,nextCursor:Gn},void 0,_,Ie,l.allowDelete??false,l.relationalMeta,S,A,Vn,l.isGroup,Qn,l.mutableFields,l.schema));},s=(c,p)=>{let u=c.params.repoName;if(!u){z(p,"Bad request",400);return}let l=e[u];if(!l){z(p,"Repository not found",404);return}let v=ye(c,t),R=Se(l.schema,l.createFields),T=re(R),$=`${v}/${l.name}/create`,O=de(T,$,"POST","Create document");z(p,me(l.name,O,"create",null,v));},o=async(c,p)=>{let u=c.params.repoName;if(!u){z(p,"Bad request",400);return}let l=e[u];if(!l){z(p,"Repository not found",404);return}let v=ye(c,t),R=c.body??{},T=It(R,l.schema),$=Se(l.schema,l.createFields),O=$.safeParse(T);if(!O.success){let w=re($),S=`${v}/${l.name}/create`,k=de(w,S,"POST","Create document"),A=O.error.issues.map(E=>`${E.path.join(".")}: ${E.message}`).join(", ");z(p,me(l.name,k,"create",null,v,{type:"error",message:`Validation error: ${A}`}),422);return}try{if(l.isGroup&&l.parentKeys&&l.parentKeys.length>0){let w={...O.data};l.createdKey&&(w[l.createdKey]=new Date);let S=l.parentKeys.filter(C=>!w[C]);if(S.length>0)throw new Error(`Missing parent key(s) for subcollection create: ${S.join(", ")}`);let k=l.parentKeys.map(C=>w[C]),A=l.documentKey??"docId",E=w[A]||Hr();await l.repo.set(...k,E,w);}else await l.repo.create(O.data);Et(p,`${v}/${l.name}?flash=created`);}catch(w){let S=Se(l.schema,l.createFields),k=re(S),A=`${v}/${l.name}/create`,E=de(k,A,"POST","Create document");z(p,me(l.name,E,"create",null,v,{type:"error",message:`Save error: ${w.message}`}),500);}},a=async(c,p)=>{let u=c.params.repoName,l=c.params.id;if(!u||!l){z(p,"Bad request",400);return}let v=e[u];if(!v){z(p,"Repository not found",404);return}let R=ye(c,t),T=null;try{T=await Ke(v,l);}catch(A){let E=Pt(v,l,A),C=we(A)?424:500;z(p,nt("",{title:`Edit ${v.name} / ${l}`,basePath:R,breadcrumb:[{label:"Repositories",href:R},{label:v.name,href:`${R}/${v.name}`},{label:`Edit ${l}`}],flash:E}),C);return}if(!T){z(p,"Document not found",404);return}let $=In(T,v.schema),O=Se(v.schema,v.mutableFields),w=Dt(re(O),$),S=`${R}/${v.name}/${encodeURIComponent(l)}/edit`,k=de(w,S,"POST","Save changes");z(p,me(v.name,k,"edit",l,R));},d=async(c,p)=>{let u=c.params.repoName,l=c.params.id;if(!u||!l){z(p,"Bad request",400);return}let v=e[u];if(!v){z(p,"Repository not found",404);return}let R=ye(c,t),T=c.body??{},$=It(T,v.schema),O=Se(v.schema,v.mutableFields),S=O.partial().safeParse($);if(!S.success){let k=Object.fromEntries(Object.entries(T).map(([D,_])=>[D,Array.isArray(_)?_.join(","):_??""])),A=Dt(re(O),k),E=`${R}/${v.name}/${encodeURIComponent(l)}/edit`,C=de(A,E,"POST","Save changes"),B=S.error.issues.map(D=>`${D.path.join(".")}: ${D.message}`).join(", ");z(p,me(v.name,C,"edit",l,R,{type:"error",message:`Validation error: ${B}`}),422);return}try{let k=await Ke(v,l),A=(k&&Ot(k,v.pathKey))??[l];await v.repo.update(...A,S.data),Et(p,`${R}/${v.name}?flash=updated`);}catch(k){let A=Se(v.schema,v.mutableFields),E=re(A),C=`${R}/${v.name}/${encodeURIComponent(l)}/edit`,B=de(E,C,"POST","Save changes"),D=we(k)?Pt(v,l,k):{type:"error",message:`Save error: ${k.message}`},_=we(k)?424:500;z(p,me(v.name,B,"edit",l,R,D),_);}},i=async(c,p)=>{let u=c.params.repoName,l=c.params.id;if(!u||!l){z(p,"Bad request",400);return}let v=e[u];if(!v){z(p,"Repository not found",404);return}if(!v.allowDelete){z(p,"Delete is not allowed for this repository",403);return}let R=ye(c,t);try{let T=await Ke(v,l),$=(T&&Ot(T,v.pathKey))??[l];await v.repo.delete(...$),Et(p,`${R}/${v.name}?flash=deleted`);}catch(T){let $=we(T)?Pt(v,l,T):{type:"error",message:`Delete error: ${T.message}`},O=we(T)?424:500;z(p,nt("",{title:`Delete ${v.name} / ${l}`,basePath:R,breadcrumb:[{label:"Repositories",href:R},{label:v.name,href:`${R}/${v.name}`},{label:`Delete ${l}`}],flash:$}),O);}},f=async(c,p)=>{let u=c.params.repoName;if(!u){z(p,"Bad request",400);return}let l=e[u];if(!l){z(p,"Repository not found",404);return}let v=ye(c,t),R=c.query,T=R?.type==="many"?"many":"one",$=Math.max(1,Math.min(100,Number(R?.ps??25)||25)),O=l.listColumns??Object.keys(J(l.schema)),{PanelOne:w,PanelMany:S}=await Promise.resolve().then(()=>(gt(),xn)),{renderToString:k}=await import('hono/jsx/dom/server');if(T==="one"){let _=String(R?.id??"");if(!_){z(p,"<div class='p-6 text-error'>Missing id parameter.</div>",400);return}try{let oe=await Ke(l,_),Ie=k(w({doc:oe,repoName:l.name,basePath:v,schema:l.schema,columns:O}));z(p,Ie);}catch(oe){z(p,`<div class='p-6 text-error text-sm'>Error: ${oe.message}</div>`,500);}return}let A=String(R?.fk??""),E=String(R?.fv??"");if(!A||!E){z(p,"<div class='p-6 text-error'>Missing fk/fv parameters.</div>",400);return}let C=R?.cursor??"",B=R?.dir==="prev"?"prev":"next",D;if(C)try{let _=l.repo.ref;typeof _.doc=="function"&&(D=await _.doc(C).get());}catch{}try{let _=await l.repo.query.paginate({pageSize:$,cursor:D,direction:B,where:[[A,"==",Jr(E)]]}),oe=k(S({docs:_.data,repoName:l.name,basePath:v,fk:A,fv:E,columns:O,schema:l.schema,pagination:{hasPrev:_.hasPrevPage,hasNext:_.hasNextPage,prevCursor:_.prevCursor?.id??"",nextCursor:_.nextCursor?.id??"",pageSize:$}}));z(p,oe);}catch(_){z(p,`<div class='p-6 text-error text-sm'>Error: ${_.message}</div>`,500);}},g=async(c,p)=>{let u=[];for(let l of p){let v;if(c.isGroup||c.parentKeys?.length){let R=await Ke(c,l);v=R?Ot(R,c.pathKey):void 0;}v||(v=[l]);try{let R=c.repo.documentRef(...v);R&&u.push(R);}catch{}}return u},x=async(c,p)=>{let u=On(p),l=c.documentKey??"docId",v=[],R;for(;;){let T=await c.repo.query.paginate({pageSize:500,cursor:R,direction:"next",...u.length>0?{where:u}:{}});for(let $ of T.data){let O=String($[l]??$.id??"");O&&v.push(O);}if(!T.hasNextPage||!T.nextCursor)break;R=T.nextCursor;}return v},h=async(c,p)=>{let u=c.params.repoName;if(!u){V(p,{error:"Bad request"},400);return}let l=e[u];if(!l){V(p,{error:"Repository not found"},404);return}if(!l.allowDelete){V(p,{error:"Delete is not allowed for this repository"},403);return}let v=c.body??{};try{let R=await m(l,v);if(R.length===0){V(p,{deleted:0});return}let T=await g(l,R);for(let $=0;$<T.length;$+=500)await l.repo.bulk.delete(T.slice($,$+500));V(p,{deleted:T.length});}catch(R){V(p,{error:R.message},500);}},b=async(c,p)=>{let u=c.params.repoName;if(!u){V(p,{error:"Bad request"},400);return}let l=e[u];if(!l){V(p,{error:"Repository not found"},404);return}let v=c.body??{},R=String(v.field??"");if(!R){V(p,{error:"Missing 'field'"},400);return}if(!l.mutableFields||!l.mutableFields.includes(R)){V(p,{error:`Field '${R}' is not bulk-updatable`},403);return}let T=l.schema.shape?.[R],$=v.value;if(T){let O=T.safeParse(v.value);if(!O.success){V(p,{error:`Invalid value for '${R}': ${O.error.message}`},400);return}$=O.data;}try{let O=await m(l,v);if(O.length===0){V(p,{updated:0});return}let S=(await g(l,O)).map(k=>({docRef:k,data:{[R]:$}}));for(let k=0;k<S.length;k+=500)await l.repo.bulk.update(S.slice(k,k+500));V(p,{updated:S.length});}catch(O){V(p,{error:O.message},500);}};async function m(c,p){if(p.selectAll){let u=y(p.filters,c);return await x(c,u)}return Array.isArray(p.ids)?p.ids.filter(u=>typeof u=="string"&&!!u):[]}function y(c,p){if(!Array.isArray(c))return [];let u=new Set((p.filterableFields??Object.keys(J(p.schema))).map(R=>String(R))),l=new Set(["==","!=","<","<=",">",">=","in","not-in","array-contains","array-contains-any"]),v=[];for(let R of c)R&&typeof R=="object"&&typeof R.field=="string"&&u.has(R.field)&&typeof R.value=="string"&&l.has(String(R.op))&&v.push({field:R.field,op:R.op,value:R.value});return v}return {handleDashboard:n,handleList:r,handleCreateForm:s,handleCreateSubmit:o,handleEditForm:a,handleEditSubmit:d,handleDelete:i,handlePanel:f,handleBulkDelete:h,handleBulkUpdate:b}}function V(e,t,n=200){e.status(n).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(t));}function Jr(e){return e==="true"?true:e==="false"?false:e!==""&&!isNaN(Number(e))?Number(e):e}async function Xr(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function Yr(e){let t={};if(!e)return t;for(let n of e.split("&")){let r=n.indexOf("=");if(r===-1)continue;let s=decodeURIComponent(n.slice(0,r).replace(/\+/g," ")),o=decodeURIComponent(n.slice(r+1).replace(/\+/g," ")),a=t[s];a===void 0?t[s]=o:Array.isArray(a)?a.push(o):t[s]=[a,o];}return t}function _n(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:s,middleware:o=[],httpsOptions:a}=e,d=t==="/"?"":t.replace(/\/$/,""),i={};for(let[h,b]of Object.entries(n)){let m=b.schema??b.repo.schema??null;if(!m)throw new Error(`[createAdminServer] Repository "${h}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let y,c,p;if(b.fieldsConfig){let v=b.fieldsConfig;y=[],c=[],p=[];for(let[R,T]of Object.entries(v))for(let $ of T)$==="filterable"?y.push(R):$==="mutable"?c.push(R):$==="create"&&p.push(R);y.length===0&&(y=void 0),c.length===0&&(c=void 0),p.length===0&&(p=void 0);}let u=(()=>{let v=b.repo._parentKeys;return v&&v.length>0?v:void 0})();if(u&&p)for(let v of u)p.includes(v)||p.push(v);let l={name:h,path:b.path,repo:b.repo,schema:m,documentKey:b.documentKey??"docId",pathKey:b.repo._pathKey??void 0,isGroup:!!b.repo._isGroup,parentKeys:u,createdKey:b.repo._createdKey??void 0,listColumns:b.listColumns,pageSize:b.pageSize,filterableFields:y,mutableFields:c,createFields:p,allowDelete:b.allowDelete??false,relationalMeta:(()=>{if(!b.relationalFields||b.relationalFields.length===0)return;let v=b.repo.relationalKeys??{},R=[];for(let T of b.relationalFields){let $=v[T.key];$&&R.push({key:T.key,column:T.column,targetRepo:String($.repo),targetKey:String($.key),type:$.type});}return R.length>0?R:void 0})()};i[h]=l;}let f=Nn(i,d),g=new X;if(r&&g.use(async(h,b,m)=>{let y=h,c=String(y.headers?.["content-type"]??"");if(c.includes("application/x-www-form-urlencoded")){let p=await Xr(y);h.body=Yr(p);}else if(c.includes("application/json")&&typeof y.body=="string")try{h.body=JSON.parse(y.body);}catch{}await m();}),s)if(typeof s=="function")g.use(s);else {let h=s.realm??"Admin",b="Basic "+Buffer.from(`${s.username}:${s.password}`).toString("base64");g.use((m,y,c)=>{if((m.headers?.authorization??"")!==b){y.status(401).set("WWW-Authenticate",`Basic realm="${h}"`).set("Content-Type","text/plain").send("Unauthorized");return}c();});}for(let h of o)g.use(h);g.get(`${d}/`,f.handleDashboard),g.get(`${d}`,f.handleDashboard),g.get(`${d}/:repoName/_panel`,f.handlePanel),g.post(`${d}/:repoName/_bulk/delete`,f.handleBulkDelete),g.post(`${d}/:repoName/_bulk/update`,f.handleBulkUpdate),g.get(`${d}/:repoName`,f.handleList),g.get(`${d}/:repoName/create`,f.handleCreateForm),g.post(`${d}/:repoName/create`,f.handleCreateSubmit),g.get(`${d}/:repoName/:id/edit`,f.handleEditForm),g.post(`${d}/:repoName/:id/edit`,f.handleEditSubmit),g.post(`${d}/:repoName/:id/delete`,f.handleDelete);let x=async(h,b)=>{await g.handle(h,b);};return a&&(x.httpsOptions=a),x}function jt(e,t,n=200){let r=q(t);e.status(n).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(r));}function Pe(e,t,n,r=200){jt(e,{success:true,data:t,meta:n},r);}function Q(e,t,n=400){jt(e,{success:false,error:t},n);}function _t(e,t,n,r,s){let o=qe(t,n),a=o.type==="index",d=a?424:500,f={success:false,error:a?o.message:s&&t instanceof Error?t.message:r};a&&(f.errorType="index",o.indexUrl&&(f.indexUrl=o.indexUrl)),jt(e,f,d);}var jn="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function es(){let e="";for(let t=0;t<20;t++)e+=jn.charAt(Math.floor(Math.random()*jn.length));return e}function Ee(e){let t=e._def??e.def;if(!t)return e;let n=t.typeName??t.type;if(n==="ZodDate"||n==="date")return z$1.preprocess(r=>ct(r)??r,e);if(n==="ZodObject"||n==="object"){let r=e.shape,s={};for(let[o,a]of Object.entries(r))s[o]=Ee(a);return z$1.object(s)}if(n==="ZodArray"||n==="array"){let r=t.element??t.type;if(r)return z$1.array(Ee(r))}if(n==="ZodOptional"||n==="optional"){let r=t.innerType;if(r)return Ee(r).optional()}if(n==="ZodNullable"||n==="nullable"){let r=t.innerType;if(r)return Ee(r).nullable()}if(n==="ZodDefault"||n==="default"){let r=t.innerType,s=t.defaultValue;if(r){let o=Ee(r);return typeof s=="function"?o.default(s()):o.default(s)}}return e}function ts(e,t,n=[]){let r=e.shape,s={},o=t&&t.length>0?t:Object.keys(r);for(let a of o){if(n.includes(a))continue;let d=a.split(".")[0];d&&r[d]&&(s[d]=r[d]);}return z$1.object(s)}function zn(e,t,n,r=false,s=[]){try{let o=ts(e,n,s),a=r?o.partial():o;return {success:!0,data:(lt()==="normalize"?Ee(a):a).parse(t)}}catch(o){return o instanceof z$1.ZodError?{success:false,error:`Validation failed: ${o.issues.map(d=>`${d.path.join(".")}: ${d.message}`).join(", ")}`}:{success:false,error:"Validation failed"}}}function ns(e,t){let n=[],r=t?new Set(t):null,s={eq:"==",ne:"!=",lt:"<",lte:"<=",gt:">",gte:">=",in:"in",nin:"not-in",contains:"array-contains",containsAny:"array-contains-any"};for(let[o,a]of Object.entries(e)){if(a===void 0||["cursor","limit","pageSize","orderBy","orderDir","select"].includes(o))continue;let d=Array.isArray(a)?a[0]:a;if(d===void 0||d==="")continue;let i=o.match(/^(\w+)__(\w+)$/),f,g="==";if(i&&i[1]&&i[2]){f=i[1];let h=i[2];if(s[h])g=s[h];else continue}else if(!i)f=o;else continue;if(r&&!r.has(f))continue;let x=d;g==="in"||g==="not-in"||g==="array-contains-any"?x=d.split(",").map(h=>Fn(h.trim())):x=Fn(d),n.push({field:f,op:g,value:x});}return n}function Fn(e){if(e==="true")return true;if(e==="false")return false;if(e==="null")return null;let t=Number(e);return !isNaN(t)&&e!==""?t:e}function st(e){return e?{docId:e.id}:null}async function Zn(e,t){if(!t||typeof t!="object")return;let n=t.docId;if(typeof n=="string")try{let r=e.repo.ref;if(typeof r.doc!="function")return;let s=await r.doc(n).get();return s.exists?s:void 0}catch{return}}function Mn(e,t,n){function r(b,m){return !b||!e[b]?(Q(m,`Repository "${b}" not found`,404),null):e[b]}function s(b,m){if(!m)return;let y=b[m];if(typeof y!="string"||!y)return;let c=y.split("/").filter(Boolean),p=[];for(let u=1;u<c.length;u+=2)p.push(c[u]);return p.length>0?p:void 0}async function o(b,m){let y=`by${b.documentKey.charAt(0).toUpperCase()}${b.documentKey.slice(1)}`,c=b.repo.get[y];if(typeof c=="function")try{let u=await c(m);if(u)return u}catch{}return (await b.repo.query.by({where:[[b.documentKey,"==",m]],limit:1}))[0]??null}async function a(b,m){let y=b.params||{},c=r(y.repoName,m);if(!c)return;let p=[],u;try{let l=b.query??{},v=Math.min(Number(l.pageSize)||c.pageSize,100),R=l.cursor,T=l.direction?.toLowerCase()==="prev"?"prev":"next",$=l.orderBy,O=l.orderDir?.toLowerCase()==="desc"?"desc":"asc",w=l.select,S=w?w.split(",").map(D=>D.trim()):void 0,k;c.allowedIncludes&&l.includes&&(k=(typeof l.includes=="string"?l.includes.split(",").map(_=>_.trim()):Array.isArray(l.includes)?l.includes:[]).filter(_=>typeof _=="string"&&c.allowedIncludes.includes(_)),k?.length===0&&(k=void 0));let A=ns(l,c.filterableFields);p=A.map(D=>({field:D.field,op:D.op,value:String(D.value??"")})),$&&(u={field:$,dir:O});let E={pageSize:v,direction:T};if(R)try{let D=typeof R=="string"?JSON.parse(R):R;E.cursor=await Zn(c,D);}catch{}$&&(E.orderBy=[{field:$,direction:O}]),A.length>0&&(E.where=A.map(D=>[D.field,D.op,D.value])),S&&(E.select=S),k&&(E.include=k);let C=await c.repo.query.paginate(E),B={items:C.data,hasNextPage:C.hasNextPage,hasPrevPage:C.hasPrevPage,nextCursor:st(C.nextCursor),prevCursor:st(C.prevCursor)};Pe(m,B,{pageSize:v,hasMore:C.hasNextPage});}catch(l){_t(m,l,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:p,sort:u},"Failed to fetch documents",n);}}async function d(b,m){let y=b.params||{},c=r(y.repoName,m);if(!c)return;let p=[],u;try{let l=b.body??{},v=Math.min(l.pageSize||c.pageSize,100),R=l.direction==="prev"?"prev":"next";l.where&&(p=l.where.map(w=>({field:String(w[0]),op:w[1],value:String(w[2]??"")}))),l.orderBy&&l.orderBy[0]&&(u={field:l.orderBy[0].field,dir:l.orderBy[0].direction==="desc"?"desc":"asc"});let T={pageSize:v,direction:R};if(l.cursor)try{let w=typeof l.cursor=="string"?JSON.parse(l.cursor):l.cursor;T.cursor=await Zn(c,w);}catch{}if(c.allowedIncludes&&l.includes&&l.includes.length>0){let w=l.includes.filter(S=>typeof S=="string"?c.allowedIncludes.includes(S):typeof S=="object"&&S!==null&&"relation"in S&&typeof S.relation=="string"?c.allowedIncludes.includes(S.relation):!1);w.length>0&&(T.include=w);}if(l.where&&l.where.length>0){if(c.filterableFields){let w=new Set(c.filterableFields),S=l.where.filter(k=>!w.has(k[0]));if(S.length>0){Q(m,`Fields not filterable: ${S.map(k=>k[0]).join(", ")}`,400);return}}T.where=l.where;}if(l.orWhere&&l.orWhere.length>0){if(c.filterableFields){let w=new Set(c.filterableFields),S=l.orWhere.filter(k=>!w.has(k[0]));if(S.length>0){Q(m,`Fields not filterable: ${S.map(k=>k[0]).join(", ")}`,400);return}}T.orWhere=l.orWhere;}if(l.orWhereGroups&&l.orWhereGroups.length>0){if(c.filterableFields){let w=new Set(c.filterableFields);for(let S of l.orWhereGroups){let k=S.filter(A=>!w.has(A[0]));if(k.length>0){Q(m,`Fields not filterable: ${k.map(A=>A[0]).join(", ")}`,400);return}}}T.orWhereGroups=l.orWhereGroups;}l.orderBy&&l.orderBy.length>0&&(T.orderBy=l.orderBy),l.select&&l.select.length>0&&(T.select=l.select);let $=await c.repo.query.paginate(T),O={items:$.data,hasNextPage:$.hasNextPage,hasPrevPage:$.hasPrevPage,nextCursor:st($.nextCursor),prevCursor:st($.prevCursor)};Pe(m,O,{pageSize:v,hasMore:$.hasNextPage});}catch(l){_t(m,l,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:p,sort:u},"Failed to query documents",n);}}async function i(b,m){let y=b.params||{},c=r(y.repoName,m);if(!c)return;let p=y.id;if(!p){Q(m,"Document ID required",400);return}try{let u=await o(c,p);if(!u){Q(m,"Document not found",404);return}Pe(m,u);}catch(u){_t(m,u,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:[{field:c.documentKey,op:"==",value:p}]},"Failed to fetch document",n);}}async function f(b,m){let y=b.params||{},c=r(y.repoName,m);if(c)try{let p=b.body??{},u=zn(c.schema,p,c.createFields,!1,c.systemKeys);if(!u.success){Q(m,u.error,400);return}if(c.validate){let v=await c.validate(u.data,"create");if(v){Q(m,v,400);return}}let l;if(c.isGroup&&c.parentKeys&&c.parentKeys.length>0){let v={...u.data};c.createdKey&&(v[c.createdKey]=new Date);let R=c.parentKeys.filter(O=>!v[O]);if(R.length>0){Q(m,`Missing parent key(s) for subcollection create: ${R.join(", ")}`,400);return}let T=c.parentKeys.map(O=>v[O]),$=v[c.documentKey]||es();l=await c.repo.set(...T,$,v);}else l=await c.repo.create(u.data);Pe(m,l,void 0,201);}catch(p){let u=n&&p instanceof Error?p.message:"Failed to create document";Q(m,u,500);}}async function g(b,m,y){let c=b.params||{},p=r(c.repoName,m);if(!p)return;let u=c.id;if(!u){Q(m,"Document ID required",400);return}try{let l=b.body??{},v=zn(p.schema,l,p.mutableFields,y,p.systemKeys);if(!v.success){Q(m,v.error,400);return}if(p.validate){let O=await p.validate(v.data,"update");if(O){Q(m,O,400);return}}let R=await o(p,u),T=(R&&s(R,p.pathKey))??[u],$=await p.repo.update(...T,v.data);Pe(m,$);}catch(l){let v=n&&l instanceof Error?l.message:"Failed to update document";Q(m,v,500);}}async function x(b,m){let y=b.params||{},c=r(y.repoName,m);if(!c)return;if(!c.allowDelete){Q(m,"Delete not allowed for this repository",403);return}let p=y.id;if(!p){Q(m,"Document ID required",400);return}try{let u=await o(c,p),l=(u&&s(u,c.pathKey))??[p];await c.repo.delete(...l),Pe(m,{deleted:!0});}catch(u){let l=n&&u instanceof Error?u.message:"Failed to delete document";Q(m,l,500);}}function h(b,m){m.status(204).set("Access-Control-Allow-Methods","GET, POST, PUT, PATCH, DELETE, OPTIONS").set("Access-Control-Allow-Headers","Content-Type, Authorization").set("Access-Control-Max-Age","86400").send("");}return {handleList:a,handleQuery:d,handleGet:i,handleCreate:f,handleUpdate:g,handleDelete:x,handleOptions:h}}function zt(e){try{return z$1.toJSONSchema(e,{target:"openapi-3.1",unrepresentable:"any",override:t=>{let n=t.zodSchema?._zod?.def;n&&(n.type==="date"?(t.jsonSchema.type="string",t.jsonSchema.format="date-time"):n.type==="bigint"&&(t.jsonSchema.type="string",t.jsonSchema.format="int64"));}})}catch(t){return typeof console<"u"&&console.warn&&console.warn("[generateOpenAPISpec] Failed to convert Zod schema to JSON Schema; falling back to {type:object}.",t),{type:"object"}}}function ee(e){return {$ref:`#/components/schemas/${e}`}}function W(e){return {description:e,content:{"application/json":{schema:ee("ErrorResponse")}}}}function Le(e,t){return {description:e,content:{"application/json":{schema:{type:"object",properties:{success:{type:"boolean",enum:[true]},data:t},required:["success","data"]}}}}}function Bn(e){return {description:"Paginated list of documents",content:{"application/json":{schema:{type:"object",properties:{success:{type:"boolean",enum:[true]},data:{type:"object",properties:{items:{type:"array",items:e},nextCursor:{oneOf:[{type:"object"},{type:"null"}]},prevCursor:{oneOf:[{type:"object"},{type:"null"}]},hasNextPage:{type:"boolean"},hasPrevPage:{type:"boolean"}},required:["items","hasNextPage","hasPrevPage"]},meta:{type:"object",properties:{pageSize:{type:"integer"},hasMore:{type:"boolean"},cursor:{oneOf:[{type:"string"},{type:"null"}]}}}},required:["success","data"]}}}}}function rs(e){return [{name:"pageSize",in:"query",schema:{type:"integer",default:e.pageSize,maximum:100},description:"Number of items per page"},{name:"cursor",in:"query",schema:{type:"string"},description:"Base64 pagination cursor"},{name:"orderBy",in:"query",schema:{type:"string"},description:"Field name to order by"},{name:"orderDir",in:"query",schema:{type:"string",enum:["asc","desc"]},description:"Order direction"},{name:"select",in:"query",schema:{type:"string"},description:"Comma-separated list of fields to return"}]}function ss(e){let t=e.filterableFields??Object.keys(e.schema.shape),n=["eq","ne","lt","lte","gt","gte","in","nin","contains"],r=[];for(let s of t){r.push({name:s,in:"query",schema:{type:"string"},description:`Filter by ${s} (equality)`});for(let o of n)r.push({name:`${s}__${o}`,in:"query",schema:{type:"string"},description:`Filter ${s} with operator ${o}`});}return r}function os(){return {type:"object",properties:{where:{type:"array",items:{type:"array",items:{},minItems:3,maxItems:3},description:"AND conditions: [field, operator, value][]"},orWhere:{type:"array",items:{type:"array",items:{},minItems:3,maxItems:3},description:"Simple OR conditions (each independently OR'd)"},orWhereGroups:{type:"array",items:{type:"array",items:{type:"array",items:{},minItems:3,maxItems:3}},description:"Advanced OR groups (AND within, OR across groups)"},orderBy:{type:"array",items:{type:"object",properties:{field:{type:"string"},direction:{type:"string",enum:["asc","desc"]}},required:["field"]}},select:{type:"array",items:{type:"string"},description:"Fields to select (projection)"},pageSize:{type:"integer",maximum:100,description:"Number of items per page"},cursor:{oneOf:[{type:"string"},{type:"object"}],description:"Pagination cursor"},direction:{type:"string",enum:["next","prev"],description:"Pagination direction"},includes:{type:"array",items:{oneOf:[{type:"string"},{type:"object",properties:{relation:{type:"string"},select:{type:"array",items:{type:"string"}}},required:["relation"]}]},description:"Relations to include (populate)"}}}}function as(e,t,n,r,s){let o={},a=e.name,d=`${t}/${e.name}`,i=`${d}/{${e.documentKey}}`,f={name:e.documentKey,in:"path",required:true,schema:{type:"string"},description:"Unique document identifier"};o[d]={get:{operationId:`list${ge(e.name)}`,summary:`List ${e.name} (paginated)`,tags:[a],parameters:[...rs(e),...ss(e)],responses:{200:Bn(ee(n)),500:W("Internal server error")}},post:{operationId:`create${ge(e.name)}`,summary:`Create a ${se(e.name)}`,tags:[a],requestBody:{required:true,content:{"application/json":{schema:ee(r??n)}}},responses:{201:Le("Document created",ee(n)),400:W("Validation error"),500:W("Internal server error")}}},o[`${d}/query`]={post:{operationId:`query${ge(e.name)}`,summary:`Query ${e.name} with advanced filters`,tags:[a],requestBody:{required:true,content:{"application/json":{schema:ee("QueryRequestBody")}}},responses:{200:Bn(ee(n)),400:W("Invalid query"),500:W("Internal server error")}}};let g={};return g.get={operationId:`get${ge(se(e.name))}`,summary:`Get a single ${se(e.name)}`,tags:[a],parameters:[f],responses:{200:Le("Document found",ee(n)),404:W("Document not found"),500:W("Internal server error")}},g.put={operationId:`update${ge(se(e.name))}`,summary:`Update a ${se(e.name)} (full replace)`,tags:[a],parameters:[f],requestBody:{required:true,content:{"application/json":{schema:ee(s??n)}}},responses:{200:Le("Document updated",ee(n)),400:W("Validation error"),404:W("Document not found"),500:W("Internal server error")}},g.patch={operationId:`patch${ge(se(e.name))}`,summary:`Partially update a ${se(e.name)}`,tags:[a],parameters:[f],requestBody:{required:true,content:{"application/json":{schema:{allOf:[ee(s??n)],description:"All fields are optional for partial updates"}}}},responses:{200:Le("Document patched",ee(n)),400:W("Validation error"),404:W("Document not found"),500:W("Internal server error")}},e.allowDelete&&(g.delete={operationId:`delete${ge(se(e.name))}`,summary:`Delete a ${se(e.name)}`,tags:[a],parameters:[f],responses:{200:Le("Document deleted",{type:"object",properties:{id:{type:"string"}}}),404:W("Document not found"),500:W("Internal server error")}}),o[i]=g,o}function Zt(e,t,n={}){let{title:r="CRUD API",version:s="1.0.0",description:o,servers:a,auth:d=false}=n,i=t==="/"?"":t.replace(/\/$/,""),f={},g={},x=[];f.ErrorResponse={type:"object",properties:{success:{type:"boolean",enum:[false]},error:{type:"string"}},required:["success","error"]},f.QueryRequestBody=os();for(let[y,c]of Object.entries(e)){let p=ge(se(y)),u=`${p}Create`,l=`${p}Update`;f[p]=zt(c.schema);let v=S=>{let k=S&&S.length>0?S:Object.keys(c.schema.shape),A={};for(let E of k){let C=E.split(".")[0];C&&c.schema.shape[C]&&!c.systemKeys.includes(C)&&(A[C]=c.schema.shape[C]);}return A},R=null,T=v(c.createFields);Object.keys(T).length>0&&(f[u]=zt(z$1.object(T)),R=u);let $=null,O=v(c.mutableFields);Object.keys(O).length>0&&(f[l]=zt(z$1.object(O)),$=l);let w=as(c,i,p,R,$);Object.assign(g,w),x.push({name:y,description:`Operations on ${y} (collection: ${c.path})`});}let h={},b;return d==="basic"?(h.basicAuth={type:"http",scheme:"basic"},b=[{basicAuth:[]}]):d==="bearer"&&(h.bearerAuth={type:"http",scheme:"bearer",bearerFormat:"JWT"},b=[{bearerAuth:[]}]),{openapi:"3.1.0",info:{title:r,version:s,...o?{description:o}:{}},...a&&a.length>0?{servers:a}:{},paths:g,components:{schemas:f,...Object.keys(h).length>0?{securitySchemes:h}:{}},...b?{security:b}:{},tags:x}}function ge(e){return e.charAt(0).toUpperCase()+e.slice(1)}function se(e){return e.endsWith("ies")?e.slice(0,-3)+"y":e.endsWith("ses")||e.endsWith("xes")||e.endsWith("zes")?e.slice(0,-2):e.endsWith("s")&&!e.endsWith("ss")?e.slice(0,-1):e}function is(e,t){return `<!DOCTYPE html>
646
1118
  <html lang="en">
647
1119
  <head>
648
1120
  <meta charset="utf-8" />
@@ -653,5 +1125,5 @@ function initColumnReorder(table) {
653
1125
  <script id="api-reference" data-url="${t}"></script>
654
1126
  <script src="https://cdn.jsdelivr.net/npm/@scalar/api-reference"></script>
655
1127
  </body>
656
- </html>`}function Or(e,t){let n=t==="/"?"":t.replace(/\/$/,"");if(process.env.FUNCTIONS_EMULATOR==="true"){let s=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??"demo-project",a=process.env.FUNCTION_REGION??"us-central1",u=process.env.FUNCTION_TARGET??"";return `/${s}/${a}/${u}${n}`}let r=process.env.K_SERVICE,o=e?.hostname??e?.headers?.host??"";return r&&o.includes("cloudfunctions.net")?`/${r.toLowerCase()}${n}`:n}async function kr(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function Rn(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:o,middleware:s=[],verbose:a=false,httpsOptions:u}=e,i=t==="/"?"":t.replace(/\/$/,""),l={};for(let[v,g]of Object.entries(n)){let b=g.schema??g.repo.schema??null;if(!b)throw new Error(`[createCrudServer] Repository "${v}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let R,x,S;if(g.fieldsConfig){let w=g.fieldsConfig;R=[],x=[],S=[];for(let[T,O]of Object.entries(w))for(let k of O)k==="filterable"?R.push(T):k==="mutable"?x.push(T):k==="create"&&S.push(T);R.length===0&&(R=void 0),x.length===0&&(x=void 0),S.length===0&&(S=void 0);}let C=(()=>{let w=g.repo._parentKeys;return w&&w.length>0?w:void 0})();if(C&&S)for(let w of C)S.includes(w)||S.push(w);let P={name:v,path:g.path,repo:g.repo,schema:b,systemKeys:g.repo._systemKeys??[g.documentKey??"docId"],documentKey:g.documentKey??"docId",pathKey:g.repo._pathKey??void 0,isGroup:!!g.repo._isGroup,parentKeys:C,createdKey:g.repo._createdKey??void 0,pageSize:g.pageSize??25,filterableFields:R,mutableFields:x,createFields:S,allowDelete:g.allowDelete??false,allowedIncludes:g.allowedIncludes,validate:g.validate};l[v]=P;}let y=bn(l,i,a),h=e.openapi,c=h&&typeof h=="object"?h:{},d=null;function p(){if(!d){let v=o&&typeof o!="function"?"basic":o?"bearer":false;d=Rt(l,i,{...c,auth:c.auth??v});}return d}let m=new G;if(m.use((v,g,b)=>{g.set("Access-Control-Allow-Origin","*"),g.set("Access-Control-Allow-Credentials","true"),b();}),r&&m.use(async(v,g,b)=>{let R=v;if(String(R.headers?.["content-type"]??"").includes("application/json")){if(typeof R.body=="string")try{v.body=JSON.parse(R.body);}catch{}else if(Buffer.isBuffer(v.rawBody))try{let S=await kr(R);v.body=JSON.parse(S);}catch{}}await b();}),o)if(typeof o=="function")m.use(o);else {let v=o.realm??"API",g="Basic "+Buffer.from(`${o.username}:${o.password}`).toString("base64");m.use((b,R,x)=>{if((b.headers?.authorization??"")!==g){R.status(401).set("WWW-Authenticate",`Basic realm="${v}"`).set("Content-Type","application/json").send(JSON.stringify({success:false,error:"Unauthorized"}));return}x();});}for(let v of s)m.use(v);if(h!==false){let v=`${i}/__spec.json`,g=`${i}/__docs`;m.get(v,(b,R)=>{let x=p();R.status(200).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(x,null,2));}),m.get(g,(b,R)=>{let x=Or(b,i)+"/__spec.json",S=$r(c.title??"CRUD API",x);R.status(200).set("Content-Type","text/html; charset=utf-8").send(S);});}m.use((v,g,b)=>{if(v.method==="OPTIONS"){y.handleOptions(v,g);return}b();}),m.get(`${i}/:repoName`,y.handleList),m.post(`${i}/:repoName/query`,y.handleQuery),m.get(`${i}/:repoName/:id`,y.handleGet),m.post(`${i}/:repoName`,y.handleCreate),m.put(`${i}/:repoName/:id`,(v,g)=>y.handleUpdate(v,g,false)),m.patch(`${i}/:repoName/:id`,(v,g)=>y.handleUpdate(v,g,true)),m.delete(`${i}/:repoName/:id`,y.handleDelete);let f=async(v,g)=>{await m.handle(v,g);};return f.spec=p,u&&(f.httpsOptions=u),f}function xn(e,t){if(!t.onRequest)return e;let n=e.httpsOptions??t.httpsOptions;return n?t.onRequest(n,e):t.onRequest(e)}function wn(e,t,n){let r={};for(let[o,s]of Object.entries(t)){if(!s)continue;let a=e[o];if(!a)throw new Error(`[createServers.${n}] Unknown repo "${o}" \u2014 not present in the registry passed to createServers().`);r[o]={...s,repo:a};}return r}function Sn(e,t={}){return {admin(n){let r=wn(e,n.repos,"admin"),o=fn({...n,repos:r,httpsOptions:n.httpsOptions??t.httpsOptions});return xn(o,t)},crud(n){let r=wn(e,n.repos,"crud"),o=Rn({...n,repos:r,httpsOptions:n.httpsOptions??t.httpsOptions});return xn(o,t)},sync(n){let r={...n};return t.onRequest&&r.admin&&!r.admin.onRequest&&(r.admin={...r.admin,onRequest:t.onRequest,httpsOptions:r.admin.httpsOptions??t.httpsOptions}),Ut(e,r)}}}function Vs(e){return t=>({...t,schema:e,type:null,documentRef:null,update:null})}function Js(e,t){let n={...e};for(let r in t)t[r]&&(n[r]={...e[r],relationalKeys:t[r]});return n}var xt=class{constructor(t,n){this.repositoryCache=new Map;this.allRepositories={};this.db=t,this.mapping=n,this.initializeRepositories();}initializeRepositories(){for(let t of Object.keys(this.mapping))this.allRepositories[t]=We(this.db,this.mapping[t],{});for(let t of Object.keys(this.mapping))this.allRepositories[t]=We(this.db,this.mapping[t],this.allRepositories);}getRepository(t){return this.allRepositories[t]}};function Xs(e,t){let n=new xt(e,t),r=Object.keys(t);return new Proxy(n,{get(o,s){return typeof s=="string"&&s in t?o.getRepository(s):o[s]},ownKeys(){return r},getOwnPropertyDescriptor(o,s){if(typeof s=="string"&&s in t)return {configurable:true,enumerable:true,writable:false}}})}export{G as MiniRouter,xt as RepositoryMapping,ne as applyPaginationQueryOptions,ue as buildAndExecuteQuery,Js as buildRepositoryRelations,Ge as coerceToDate,He as createPaginationIterator,Vs as createRepositoryConfig,Xs as createRepositoryMapping,Sn as createServers,_e as executePaginatedQuery,Le as getDateHandling,Ne as normalizeTimestamps,An as setDateHandling};//# sourceMappingURL=index.js.map
1128
+ </html>`}function ls(e,t){let n=t==="/"?"":t.replace(/\/$/,"");if(process.env.FUNCTIONS_EMULATOR==="true"){let o=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??"demo-project",a=process.env.FUNCTION_REGION??"us-central1",d=process.env.FUNCTION_TARGET??"";return `/${o}/${a}/${d}${n}`}let r=process.env.K_SERVICE,s=e?.hostname??e?.headers?.host??"";return r&&s.includes("cloudfunctions.net")?`/${r.toLowerCase()}${n}`:n}async function cs(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function qn(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:s,middleware:o=[],verbose:a=false,httpsOptions:d}=e,i=t==="/"?"":t.replace(/\/$/,""),f={};for(let[p,u]of Object.entries(n)){let l=u.schema??u.repo.schema??null;if(!l)throw new Error(`[createCrudServer] Repository "${p}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let v,R,T;if(u.fieldsConfig){let w=u.fieldsConfig;v=[],R=[],T=[];for(let[S,k]of Object.entries(w))for(let A of k)A==="filterable"?v.push(S):A==="mutable"?R.push(S):A==="create"&&T.push(S);v.length===0&&(v=void 0),R.length===0&&(R=void 0),T.length===0&&(T=void 0);}let $=(()=>{let w=u.repo._parentKeys;return w&&w.length>0?w:void 0})();if($&&T)for(let w of $)T.includes(w)||T.push(w);let O={name:p,path:u.path,repo:u.repo,schema:l,systemKeys:u.repo._systemKeys??[u.documentKey??"docId"],documentKey:u.documentKey??"docId",pathKey:u.repo._pathKey??void 0,isGroup:!!u.repo._isGroup,parentKeys:$,createdKey:u.repo._createdKey??void 0,pageSize:u.pageSize??25,filterableFields:v,mutableFields:R,createFields:T,allowDelete:u.allowDelete??false,allowedIncludes:u.allowedIncludes,validate:u.validate};f[p]=O;}let g=Mn(f,i,a),x=e.openapi,h=x&&typeof x=="object"?x:{},b=null;function m(){if(!b){let p=s&&typeof s!="function"?"basic":s?"bearer":false;b=Zt(f,i,{...h,auth:h.auth??p});}return b}let y=new X;if(y.use((p,u,l)=>{u.set("Access-Control-Allow-Origin","*"),u.set("Access-Control-Allow-Credentials","true"),l();}),r&&y.use(async(p,u,l)=>{let v=p;if(String(v.headers?.["content-type"]??"").includes("application/json")){if(typeof v.body=="string")try{p.body=JSON.parse(v.body);}catch{}else if(Buffer.isBuffer(p.rawBody))try{let T=await cs(v);p.body=JSON.parse(T);}catch{}}await l();}),s)if(typeof s=="function")y.use(s);else {let p=s.realm??"API",u="Basic "+Buffer.from(`${s.username}:${s.password}`).toString("base64");y.use((l,v,R)=>{if((l.headers?.authorization??"")!==u){v.status(401).set("WWW-Authenticate",`Basic realm="${p}"`).set("Content-Type","application/json").send(JSON.stringify({success:false,error:"Unauthorized"}));return}R();});}for(let p of o)y.use(p);if(x!==false){let p=`${i}/__spec.json`,u=`${i}/__docs`;y.get(p,(l,v)=>{let R=m();v.status(200).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(R,null,2));}),y.get(u,(l,v)=>{let R=ls(l,i)+"/__spec.json",T=is(h.title??"CRUD API",R);v.status(200).set("Content-Type","text/html; charset=utf-8").send(T);});}y.use((p,u,l)=>{if(p.method==="OPTIONS"){g.handleOptions(p,u);return}l();}),y.get(`${i}/:repoName`,g.handleList),y.post(`${i}/:repoName/query`,g.handleQuery),y.get(`${i}/:repoName/:id`,g.handleGet),y.post(`${i}/:repoName`,g.handleCreate),y.put(`${i}/:repoName/:id`,(p,u)=>g.handleUpdate(p,u,false)),y.patch(`${i}/:repoName/:id`,(p,u)=>g.handleUpdate(p,u,true)),y.delete(`${i}/:repoName/:id`,g.handleDelete);let c=async(p,u)=>{await y.handle(p,u);};return c.spec=m,d&&(c.httpsOptions=d),c}function Kn(e,t){if(!t.onRequest)return e;let n=e.httpsOptions??t.httpsOptions;return n?t.onRequest(n,e):t.onRequest(e)}function Un(e,t,n){let r={};for(let[s,o]of Object.entries(t)){if(!o)continue;let a=e[s];if(!a)throw new Error(`[createServers.${n}] Unknown repo "${s}" \u2014 not present in the registry passed to createServers().`);r[s]={...o,repo:a};}return r}function Ln(e,t={}){return {admin(n){let r=Un(e,n.repos,"admin"),s=_n({...n,repos:r,httpsOptions:n.httpsOptions??t.httpsOptions});return Kn(s,t)},crud(n){let r=Un(e,n.repos,"crud"),s=qn({...n,repos:r,httpsOptions:n.httpsOptions??t.httpsOptions});return Kn(s,t)},sync(n){let r={...n};return t.onRequest&&r.admin&&!r.admin.onRequest&&(r.admin={...r.admin,onRequest:t.onRequest,httpsOptions:r.admin.httpsOptions??t.httpsOptions}),pn(e,r)}}}function Za(e){return t=>({...t,schema:e,type:null,documentRef:null,update:null})}function Ma(e,t){let n={...e};for(let r in t)t[r]&&(n[r]={...e[r],relationalKeys:t[r]});return n}var Mt=class{constructor(t,n){this.repositoryCache=new Map;this.allRepositories={};this.db=t,this.mapping=n,this.initializeRepositories();}initializeRepositories(){for(let t of Object.keys(this.mapping))this.allRepositories[t]=ut(this.db,this.mapping[t],{});for(let t of Object.keys(this.mapping))this.allRepositories[t]=ut(this.db,this.mapping[t],this.allRepositories);}getRepository(t){return this.allRepositories[t]}};function Ba(e,t){let n=new Mt(e,t),r=Object.keys(t);return new Proxy(n,{get(s,o){return typeof o=="string"&&o in t?s.getRepository(o):s[o]},ownKeys(){return r},getOwnPropertyDescriptor(s,o){if(typeof o=="string"&&o in t)return {configurable:true,enumerable:true,writable:false}}})}export{X as MiniRouter,Mt as RepositoryMapping,le as applyPaginationQueryOptions,be as buildAndExecuteQuery,Ma as buildRepositoryRelations,ct as coerceToDate,dt as createPaginationIterator,Za as createRepositoryConfig,Ba as createRepositoryMapping,Ln as createServers,Ve as executePaginatedQuery,lt as getDateHandling,We as normalizeTimestamps,tr as setDateHandling};//# sourceMappingURL=index.js.map
657
1129
  //# sourceMappingURL=index.js.map