@lpdjs/firestore-repo-service 2.2.5 → 2.2.7
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/{create-servers-DmggzSb3.d.cts → create-servers-B3Ls46bx.d.cts} +3 -3
- package/dist/{create-servers-D4-NGpKm.d.ts → create-servers-CvVZnihM.d.ts} +3 -3
- package/dist/history/index.cjs +2 -0
- package/dist/history/index.cjs.map +1 -0
- package/dist/history/index.d.cts +127 -0
- package/dist/history/index.d.ts +127 -0
- package/dist/history/index.js +2 -0
- package/dist/history/index.js.map +1 -0
- package/dist/{index-DpD4DEqH.d.cts → index-Cw9b1crP.d.cts} +1 -1
- package/dist/{index-Cvip2Sgt.d.ts → index-DGB3Oemk.d.ts} +1 -1
- package/dist/index.cjs +587 -97
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +19 -10
- package/dist/index.d.ts +19 -10
- package/dist/index.js +587 -97
- package/dist/index.js.map +1 -1
- package/dist/{openapi-B3P2F8op.d.ts → openapi-BEf4OuG9.d.ts} +1 -1
- package/dist/{openapi-UJJ1aCFk.d.cts → openapi-D8UTm0lu.d.cts} +1 -1
- package/dist/read-DMvxeFCg.d.cts +232 -0
- package/dist/read-DMvxeFCg.d.ts +232 -0
- package/dist/servers/admin/index.cjs +566 -76
- package/dist/servers/admin/index.cjs.map +1 -1
- package/dist/servers/admin/index.d.cts +4 -2
- package/dist/servers/admin/index.d.ts +4 -2
- package/dist/servers/admin/index.js +566 -76
- package/dist/servers/admin/index.js.map +1 -1
- package/dist/servers/crud/index.d.cts +6 -4
- package/dist/servers/crud/index.d.ts +6 -4
- package/dist/servers/index.cjs +597 -107
- package/dist/servers/index.cjs.map +1 -1
- package/dist/servers/index.d.cts +7 -6
- package/dist/servers/index.d.ts +7 -6
- package/dist/servers/index.js +597 -107
- package/dist/servers/index.js.map +1 -1
- package/dist/{types-Z9Qy8Xmx.d.ts → types-CE1dws32.d.cts} +20 -3
- package/dist/{types-Z9Qy8Xmx.d.cts → types-CjexXskS.d.ts} +20 -3
- package/package.json +14 -1
package/dist/servers/index.cjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
'use strict';var zod=require('zod'),jsxRuntime=require('hono/jsx/jsx-runtime'),server=require('hono/jsx/dom/server'),firestore=require('firebase-admin/firestore');function on(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 sn(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}=on(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=sn(t),s=null,a={};for(let m of this.routes){if(m.method!==r)continue;let y=o.match(m.pattern);if(y){s=m,a={},m.paramNames.forEach((x,i)=>{a[x]=decodeURIComponent(y[i+1]??"");});break}}let h=Object.assign(t,{params:a}),d=s?s.handler:this.notFoundHandler;try{await this.runMiddlewareChain(h,n,d);}catch(m){this.errorHandler(m,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 an={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 an[n]??`Zod${n.charAt(0).toUpperCase()}${n.slice(1)}`;let r=t._def?.typeName;return r||""}function M(e){let t=e;if(t._zod?.def?.innerType)return t._zod.def.innerType;if(t._def?.innerType)return t._def.innerType}function it(e){let t=e;if(t._zod?.def?.element)return t._zod.def.element;if(t._def?.type)return t._def.type}function lt(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 ee(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 be(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 ve(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 Te(e){let t=e;return Array.isArray(t._zod?.def?.values)?t._zod.def.values[0]:t._def?.value}function ct(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 te="__sync_version";var ln=new Set(["ZodOptional","ZodNullable","ZodDefault"]);function cn(e){let t=e,n=false;for(;;){let r=j(t);if(!ln.has(r))break;(r==="ZodOptional"||r==="ZodNullable")&&(n=true);let o=M(t);if(!o)break;t=o;}return {inner:t,nullable:n}}var dn={ZodString:"string",ZodNumber:"number",ZodBigInt:"bigint",ZodBoolean:"boolean",ZodDate:"timestamp",ZodEnum:"string",ZodNativeEnum:"string",ZodLiteral:"string"};function dt(e,t,n,r,o,s,a,h){for(let[d,m]of Object.entries(e)){let y=n?`${n}__${d}`:d;if(o.has(d)||o.has(y))continue;let{inner:x,nullable:i}=cn(m),l=j(x),f=r||i;if(l==="ZodObject"){let u=ee(x);dt(u,t,y,f,o,s,a,h);continue}let p=dn[l]??"json",c=y===a||d===a,b=s[y]??s[d]??y;h.push({name:b,sqlType:t.mapType(p),nullable:c?false:f,isPrimaryKey:c});}}function Ae(e,t,n={}){let{primaryKey:r,exclude:o=[],columnMap:s={}}=n,a=new Set(o),h=ee(e),d=[];return dt(h,t,"",false,a,s,r,d),d.some(m=>m.name===te)||d.push({name:te,sqlType:t.mapType("bigint"),nullable:true,isPrimaryKey:false,description:"Monotonic publish version (Date.now() ms). Internal."}),d}function ut(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(ut)):e}function pt(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)?pt(o,s,n):n[s]=ut(o);}}function xe(e,t){let n=new Set(t?.exclude),r=t?.columnMap??{},o={};pt(e,"",o);let s={};for(let[a,h]of Object.entries(o)){if(n.has(a))continue;let d=a.split("__")[0];if(d!==a&&n.has(d))continue;let m=r[a]??(a.includes("__")?r[a.split("__").pop()]:void 0)??a;s[m]=h;}return s}function le(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 V(e,t,n){return `<!DOCTYPE html>
|
|
1
|
+
'use strict';var jsxRuntime=require('hono/jsx/jsx-runtime'),zod=require('zod'),server=require('hono/jsx/dom/server'),firestore=require('firebase-admin/firestore');var An=Object.defineProperty;var Me=(e,t)=>()=>(e&&(t=e(e=0)),t);var En=(e,t)=>{for(var n in t)An(e,n,{get:t[n],enumerable:true});};function q(e){let t=e,n=t._zod?.def?.type;if(n)return In[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 Ot(e){let t=e;if(t._zod?.def?.element)return t._zod.def.element;if(t._def?.type)return t._def.type}function Tt(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 Q(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 Ae(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 Ee(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 Ue(e){let t=e;return Array.isArray(t._zod?.def?.values)?t._zod.def.values[0]:t._def?.value}function At(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 In,Pe=Me(()=>{In={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 Ut({val:e}){return jsxRuntime.jsx("span",{class:"text-sm text-base-content/80 font-mono tabular-nums whitespace-nowrap",children:e.toLocaleString()})}function Un({message:e}){return jsxRuntime.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:jsxRuntime.jsx("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24",fill:"currentColor",class:"size-3.5",children:jsxRuntime.jsx("path",{d:"M12 2 1 22h22L12 2zm0 6 7.5 13h-15L12 8zm-1 4v4h2v-4h-2zm0 5v2h2v-2h-2z"})})})}function Se({val:e,mismatch:t}){let n=Ln(e);return t?jsxRuntime.jsxs("span",{class:"inline-flex items-start gap-0.5",children:[n,jsxRuntime.jsx(Un,{message:t})]}):n}function Ln(e){if(e==null)return jsxRuntime.jsx("span",{class:"opacity-30 italic text-xs",children:"\u2014"});if(typeof e=="boolean")return e?jsxRuntime.jsx("span",{class:"badge badge-success badge-sm",children:"true"}):jsxRuntime.jsx("span",{class:"badge badge-error badge-sm",children:"false"});if(e instanceof Date)return jsxRuntime.jsx(Ut,{val:e});if(typeof e=="object"&&e!==null&&typeof e.toDate=="function")return jsxRuntime.jsx(Ut,{val:e.toDate()});if(typeof e=="number")return jsxRuntime.jsx("span",{class:"text-sm font-mono tabular-nums",children:String(e)});if(Array.isArray(e))return e.length===0?jsxRuntime.jsx("span",{class:"text-xs text-base-content/30",children:"[]"}):jsxRuntime.jsxs("ul",{class:"list-none p-0 m-0 space-y-0.5 text-xs",children:[e.slice(0,8).map((n,r)=>jsxRuntime.jsx("li",{class:"break-all",children:typeof n=="object"?JSON.stringify(n):String(n)},r)),e.length>8&&jsxRuntime.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?jsxRuntime.jsx("span",{class:"text-xs text-base-content/30",children:"{}"}):jsxRuntime.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])=>jsxRuntime.jsxs(jsxRuntime.Fragment,{children:[jsxRuntime.jsx("dt",{class:"text-base-content/50 font-semibold whitespace-nowrap",children:r}),jsxRuntime.jsx("dd",{class:"break-all",children:String(s??"")})]})),n.length>8&&jsxRuntime.jsxs("dt",{class:"col-span-2 text-base-content/40 italic",children:["+",n.length-8," more\u2026"]})]})}let t=String(e);return jsxRuntime.jsx("span",{class:"text-sm break-all",children:t})}var Ke=Me(()=>{});function ke(e){if(!e)return "unknown";let t=Xe(e);switch(q(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 Xe(e){let t=q(e);if(t==="ZodOptional"||t==="ZodNullable"||t==="ZodDefault"){let n=U(e);return n?Xe(n):e}return e}function $e(e,t){if(!e)return;let n=t.split("."),r=e;for(let s of n){if(!r)return;let o=Xe(r);if(q(o)!=="ZodObject")return;r=Q(o)[s];}return r}function Gn(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 De(e,t){if(e==="unknown")return null;let n=Gn(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 Ye=Me(()=>{Pe();});var Kt={};En(Kt,{PanelMany:()=>Qn,PanelOne:()=>Jn,RightPanel:()=>et});function Jn({doc:e,repoName:t,basePath:n,schema:r,columns:s}){if(!e)return jsxRuntime.jsx("div",{class:"text-center py-12 text-base-content/50",children:"Document not found."});let o=String(e.docId??e.id??""),l=`${n}/${t}/${encodeURIComponent(o)}/edit`,p=Ht(r,s);return jsxRuntime.jsxs("div",{class:"space-y-4",children:[jsxRuntime.jsxs("div",{class:"flex items-center justify-between gap-2",children:[jsxRuntime.jsxs("div",{class:"text-xs text-base-content/60",children:[jsxRuntime.jsx("span",{class:"font-mono",children:t}),jsxRuntime.jsx("span",{class:"opacity-50",children:" \xB7 "}),jsxRuntime.jsx("span",{class:"font-mono break-all",children:o})]}),jsxRuntime.jsx("a",{href:l,class:"btn btn-sm btn-primary",children:"Edit \u2192"})]}),jsxRuntime.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(u=>{let g=e[u],h=p[u],k=h?De(h,g):null;return jsxRuntime.jsxs(jsxRuntime.Fragment,{children:[jsxRuntime.jsx("dt",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide pt-0.5",children:u}),jsxRuntime.jsx("dd",{class:"min-w-0",children:jsxRuntime.jsx(Se,{val:g,mismatch:k})})]})})})]})}function Qn({docs:e,repoName:t,basePath:n,fk:r,fv:s,columns:o,schema:l,pagination:p}){let u=`${n}/${t}?fv_${r}=${encodeURIComponent(s)}`,g=Ht(l,o);return jsxRuntime.jsxs("div",{class:"space-y-4",children:[jsxRuntime.jsxs("div",{class:"flex items-center justify-between gap-2",children:[jsxRuntime.jsxs("div",{class:"text-xs text-base-content/60",children:[jsxRuntime.jsx("span",{class:"font-mono",children:t}),jsxRuntime.jsx("span",{class:"opacity-50",children:" where "}),jsxRuntime.jsx("span",{class:"font-mono",children:r}),jsxRuntime.jsx("span",{class:"opacity-50",children:" = "}),jsxRuntime.jsx("span",{class:"font-mono break-all",children:s}),jsxRuntime.jsx("span",{class:"opacity-50",children:" \xB7 "}),jsxRuntime.jsxs("span",{children:[e.length," doc",e.length!==1?"s":""]})]}),jsxRuntime.jsx("a",{href:u,class:"btn btn-sm btn-outline",children:"Full view \u2192"})]}),jsxRuntime.jsx("div",{class:"overflow-x-auto rounded-box border border-base-300 bg-base-100",children:jsxRuntime.jsxs("table",{class:"table table-xs w-full",children:[jsxRuntime.jsx("thead",{children:jsxRuntime.jsxs("tr",{class:"bg-base-200/50",children:[o.map((h,k)=>jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:h},k)),jsxRuntime.jsx("th",{})]})}),jsxRuntime.jsx("tbody",{children:e.length===0?jsxRuntime.jsx("tr",{children:jsxRuntime.jsx("td",{colspan:o.length+1,class:"text-center py-10 text-base-content/40",children:"No related documents"})}):e.map((h,k)=>{let w=String(h.docId??h.id??""),b=`${n}/${t}/${encodeURIComponent(w)}/edit`;return jsxRuntime.jsxs("tr",{class:"hover",children:[o.map((m,y)=>{let i=h[m],d=g[m],c=d?De(d,i):null;return jsxRuntime.jsx("td",{class:"align-top py-1.5",children:jsxRuntime.jsx(Se,{val:i,mismatch:c})},y)}),jsxRuntime.jsx("td",{class:"text-right py-1.5",children:jsxRuntime.jsx("a",{href:b,class:"btn btn-xs btn-ghost",children:"Edit"})})]},k)})})]})}),(p.hasPrev||p.hasNext)&&jsxRuntime.jsxs("div",{class:"flex justify-center items-center gap-2",children:[p.hasPrev?jsxRuntime.jsx("button",{type:"button",class:"btn btn-xs btn-outline","data-frs-panel-page":"prev","data-cursor":p.prevCursor,children:"\u2190 Previous"}):jsxRuntime.jsx("button",{class:"btn btn-xs btn-outline",disabled:true,children:"\u2190 Previous"}),p.hasNext?jsxRuntime.jsx("button",{type:"button",class:"btn btn-xs btn-outline","data-frs-panel-page":"next","data-cursor":p.nextCursor,children:"Next \u2192"}):jsxRuntime.jsx("button",{class:"btn btn-xs btn-outline",disabled:true,children:"Next \u2192"})]})]})}function Ht(e,t){if(!e)return {};let n={};for(let r of t)n[r]=ke($e(e,r));return n}var et,tt=Me(()=>{Ke();Ye();et=({basePath:e=""})=>jsxRuntime.jsxs("div",{class:"fixed inset-0 z-[100] hidden pointer-events-none","data-frs-panel-root":true,"data-frs-base-path":e,"aria-hidden":"true",children:[jsxRuntime.jsx("div",{class:"absolute inset-0 bg-black/30 opacity-0 transition-opacity duration-200 pointer-events-auto","data-frs-panel-backdrop":true}),jsxRuntime.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:[jsxRuntime.jsxs("header",{class:"flex items-center justify-between px-5 py-3 border-b border-base-300 bg-base-200/40 shrink-0",children:[jsxRuntime.jsx("h2",{class:"font-semibold text-base truncate","data-frs-panel-title":true,children:"Relation"}),jsxRuntime.jsx("button",{type:"button",class:"btn btn-sm btn-ghost btn-circle","data-frs-panel-close":true,"aria-label":"Close panel",children:"\u2715"})]}),jsxRuntime.jsx("div",{class:"flex-1 overflow-auto p-5 text-sm","data-frs-panel-body":true,children:jsxRuntime.jsx("div",{class:"flex items-center justify-center py-12 text-base-content/40",children:jsxRuntime.jsx("span",{class:"loading loading-spinner loading-md"})})})]})]});});function Pn(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 Nn(e){let t=e.path??e.url??"/",n=t.indexOf("?");return n===-1?t:t.slice(0,n)}var Y=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}=Pn(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=Nn(t),o=null,l={};for(let g of this.routes){if(g.method!==r)continue;let h=s.match(g.pattern);if(h){o=g,l={},g.paramNames.forEach((k,w)=>{l[k]=decodeURIComponent(h[w+1]??"");});break}}let p=Object.assign(t,{params:l}),u=o?o.handler:this.notFoundHandler;try{await this.runMiddlewareChain(p,n,u);}catch(g){this.errorHandler(g,t,n);}}async runMiddlewareChain(t,n,r){let s=0,o=async()=>{if(s<this.middlewares.length){let l=this.middlewares[s++];await l(t,n,o);}else await r(t,n);};await o();}};Pe();var ae="__sync_version";var _n=new Set(["ZodOptional","ZodNullable","ZodDefault"]);function Dn(e){let t=e,n=false;for(;;){let r=q(t);if(!_n.has(r))break;(r==="ZodOptional"||r==="ZodNullable")&&(n=true);let s=U(t);if(!s)break;t=s;}return {inner:t,nullable:n}}var jn={ZodString:"string",ZodNumber:"number",ZodBigInt:"bigint",ZodBoolean:"boolean",ZodDate:"timestamp",ZodEnum:"string",ZodNativeEnum:"string",ZodLiteral:"string"};function Et(e,t,n,r,s,o,l,p){for(let[u,g]of Object.entries(e)){let h=n?`${n}__${u}`:u;if(s.has(u)||s.has(h))continue;let{inner:k,nullable:w}=Dn(g),b=q(k),m=r||w;if(b==="ZodObject"){let c=Q(k);Et(c,t,h,m,s,o,l,p);continue}let y=jn[b]??"json",i=h===l||u===l,d=o[h]??o[u]??h;p.push({name:d,sqlType:t.mapType(y),nullable:i?false:m,isPrimaryKey:i});}}function Le(e,t,n={}){let{primaryKey:r,exclude:s=[],columnMap:o={}}=n,l=new Set(s),p=Q(e),u=[];return Et(p,t,"",false,l,o,r,u),u.some(g=>g.name===ae)||u.push({name:ae,sqlType:t.mapType("bigint"),nullable:true,isPrimaryKey:false,description:"Monotonic publish version (Date.now() ms). Internal."}),u}function Pt(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(Pt)):e}function Nt(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)?Nt(s,o,n):n[o]=Pt(s);}}function Ne(e,t){let n=new Set(t?.exclude),r=t?.columnMap??{},s={};Nt(e,"",s);let o={};for(let[l,p]of Object.entries(s)){if(n.has(l))continue;let u=l.split("__")[0];if(u!==l&&n.has(u))continue;let g=r[l]??(l.includes("__")?r[l.split("__").pop()]:void 0)??l;o[g]=p;}return o}function ye(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",l=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${s}/${o}/${l}${t}`}let n=process.env.K_SERVICE,r=e.hostname??e.headers?.host??"";return n&&r.includes("cloudfunctions.net")?`/${n.toLowerCase()}${t}`:t}function ee(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,70 +28,70 @@
|
|
|
28
28
|
<nav><a href="${t}/">\u2190 Dashboard</a></nav>
|
|
29
29
|
<h1>${e}</h1>
|
|
30
30
|
${n}
|
|
31
|
-
</body></html>`}function
|
|
32
|
-
<td><strong>${
|
|
33
|
-
<td>${
|
|
34
|
-
<td>${
|
|
35
|
-
<td>${
|
|
36
|
-
<td>${
|
|
31
|
+
</body></html>`}function te(e,t,n=200){e.status(n).set("Content-Type","text/html; charset=utf-8").send(t);}function ge(e,t,n=200){e.status(n).set("Content-Type","application/json").send(JSON.stringify(t,null,2));}function Ie(e){return (e.headers?.accept??"").includes("application/json")}function It(e,t,n,r,s,o,l,p){let u=(s.basePath??"/").replace(/\/$/,"")||"",g=s.featuresFlag??{},h=[];for(let[w,b]of Object.entries(e)){let m=o[w];h.push({name:w,schema:b.schema??null,documentKey:b._systemKeys?.[0]??b.documentKey??"docId",tableName:m?.tableName??w,isGroup:!!b._isGroup,repoCfg:m,repo:b});}let k=new Y;if(s.auth)if(typeof s.auth=="function")k.use(s.auth);else {let w=s.auth.realm??"Sync Admin",b="Basic "+Buffer.from(`${s.auth.username}:${s.auth.password}`).toString("base64");k.use((m,y,i)=>{if((m.headers?.authorization??"")!==b){y.status(401).set("WWW-Authenticate",`Basic realm="${w}"`).set("Content-Type","text/plain").send("Unauthorized");return}i();});}return k.get(`${u}/`,(w,b)=>{let m=ye(w,u),y=h.map(a=>{let f=[];return g.healthCheck&&f.push(`<a class="btn" href="${m}/${a.name}/health">Health</a>`),g.manualSync&&f.push(`<a class="btn btn-primary" href="${m}/${a.name}/force-sync">Force Sync</a>`),`<tr>
|
|
32
|
+
<td><strong>${a.name}</strong></td>
|
|
33
|
+
<td>${a.tableName}</td>
|
|
34
|
+
<td>${a.isGroup?'<span class="badge badge-warn">group</span>':'<span class="badge badge-ok">collection</span>'}</td>
|
|
35
|
+
<td>${a.schema?"\u2713":"\u2717"}</td>
|
|
36
|
+
<td>${f.join(" ")}</td>
|
|
37
37
|
</tr>`}).join(`
|
|
38
|
-
`),
|
|
38
|
+
`),i=g.viewQueue?`<p><a class="btn" href="${m}/queues">View Queues</a></p>`:"",d=g.configCheck?`<p style="margin-top:.5rem"><a class="btn" href="${m}/config-check">\u2699 Config Check</a></p>`:"",c=ee("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>${
|
|
41
|
+
<tbody>${y}</tbody>
|
|
42
42
|
</table>
|
|
43
|
-
${
|
|
44
|
-
${
|
|
45
|
-
</div>`);
|
|
46
|
-
`),T=
|
|
47
|
-
`),
|
|
48
|
-
<p>Table: <code>${
|
|
49
|
-
${
|
|
43
|
+
${i}
|
|
44
|
+
${d}
|
|
45
|
+
</div>`);te(b,c);}),k.get(`${u}`,(w,b)=>{let m=ye(w,u);b.status(302).set("Location",`${m}/`).send("");}),g.healthCheck&&k.get(`${u}/:repoName/health`,async(w,b)=>{let m=ye(w,u),y=h.find(C=>C.name===w.params.repoName);if(!y){te(b,ee("Not Found",m,`<p>Unknown repo: ${w.params.repoName}</p>`),404);return}if(!y.schema){te(b,ee("Health Check",m,`<p class="badge badge-warn">No Zod schema attached to "${y.name}"</p>`));return}let i=Le(y.schema,t.dialect,{primaryKey:y.documentKey,exclude:y.repoCfg?.exclude,columnMap:y.repoCfg?.columnMap}),d=[],c=false,a=null;try{c=await t.tableExists(y.tableName),c&&(d=await t.getTableColumns(y.tableName));}catch(C){a=C?.message??String(C);}let f=new Set(d),v=new Set(i.map(C=>C.name)),S=i.filter(C=>!f.has(C.name)),O=d.filter(C=>!v.has(C)),A=i.filter(C=>f.has(C.name)),x=c&&S.length===0&&!a;if(Ie(w)){ge(b,{repo:y.name,table:y.tableName,tableExists:c,healthy:x,error:a,columns:{expected:i.map(C=>({name:C.name,type:C.sqlType,nullable:C.nullable,isPrimaryKey:C.isPrimaryKey})),actual:d,matched:A.map(C=>C.name),missing:S.map(C=>({name:C.name,type:C.sqlType})),extra:O}});return}let R=x?'<span class="badge badge-ok">Healthy</span>':'<span class="badge badge-err">Unhealthy</span>',$=i.map(C=>{let B=f.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
|
+
`),T=O.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
|
+
`),P=ee(`Health: ${y.name}`,m,`<div class="card">
|
|
48
|
+
<p>Table: <code>${y.tableName}</code> ${c?R:'<span class="badge badge-err">NOT FOUND</span>'}</p>
|
|
49
|
+
${a?`<p class="badge badge-err">Error: ${a}</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>${
|
|
53
|
+
<tbody>${$}${T}</tbody>
|
|
54
54
|
</table>
|
|
55
|
-
</div>`);
|
|
56
|
-
<p>This will read <strong>all</strong> documents from the <code>${
|
|
57
|
-
and upsert them into the <code>${
|
|
55
|
+
</div>`);te(b,P);}),g.manualSync&&(k.get(`${u}/:repoName/force-sync`,(w,b)=>{let m=ye(w,u),y=h.find(d=>d.name===w.params.repoName);if(!y){te(b,ee("Not Found",m,`<p>Unknown repo: ${w.params.repoName}</p>`),404);return}let i=ee(`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="${
|
|
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>`);
|
|
63
|
-
<p class="badge badge-err">Error: ${
|
|
64
|
-
<p>Synced ${
|
|
65
|
-
</div>`),500);return}if(
|
|
66
|
-
<pre style="white-space:pre-wrap">${
|
|
67
|
-
|
|
68
|
-
`)}</pre></details>`:"",
|
|
69
|
-
<p class="badge ${
|
|
70
|
-
<p>Synced <strong>${
|
|
71
|
-
${
|
|
72
|
-
${
|
|
73
|
-
</div>`);
|
|
74
|
-
`),
|
|
62
|
+
</div>`);te(b,i);}),k.post(`${u}/:repoName/force-sync`,async(w,b)=>{let m=ye(w,u),y=h.find(x=>x.name===w.params.repoName);if(!y){ge(b,{error:`Unknown repo: ${w.params.repoName}`},404);return}let i=y.repo.ref;if(!i){ge(b,{error:`No collection reference for "${y.name}"`},400);return}let d=0,c=0,a=[],f=500,v=i.limit(f),S=null;try{for(;;){let $=await(S?v.startAfter(S):v).get();if($.empty)break;for(let T of $.docs){let P=T.data(),C=String(P[y.documentKey]??T.id),B=Ne(P,{exclude:y.repoCfg?.exclude,columnMap:y.repoCfg?.columnMap});try{await r({operation:"UPSERT",repoName:y.name,docId:C,data:B,timestamp:new Date().toISOString()}),d++;}catch(I){c++;let D=I?.message??String(I);console.error(`[ForceSync:${y.name}] doc=${C} failed:`,I),a.length<5&&a.push(`${C}: ${D}`);}}if(S=$.docs[$.docs.length-1],$.docs.length<f)break}let x=n.get(y.name);x&&await x.flush();}catch(x){if(Ie(w)){ge(b,{error:x?.message??String(x),synced:d,errors:c},500);return}te(b,ee(`Force Sync: ${y.name}`,m,`<div class="card">
|
|
63
|
+
<p class="badge badge-err">Error: ${x?.message??String(x)}</p>
|
|
64
|
+
<p>Synced ${d} docs before failure (${c} errors).</p>
|
|
65
|
+
</div>`),500);return}if(Ie(w)){ge(b,{repo:y.name,table:y.tableName,synced:d,errors:c,...a.length>0&&{errorSamples:a}});return}let O=a.length>0?`<details style="margin-top:1rem"><summary>First ${a.length} error(s)</summary>
|
|
66
|
+
<pre style="white-space:pre-wrap">${a.map(x=>x.replace(/[<>&]/g,R=>`&#${R.charCodeAt(0)};`)).join(`
|
|
67
|
+
|
|
68
|
+
`)}</pre></details>`:"",A=ee(`Force Sync: ${y.name}`,m,`<div class="card">
|
|
69
|
+
<p class="badge ${c>0?"badge-warn":"badge-ok"}">${c>0?"Completed with errors":"Complete"}</p>
|
|
70
|
+
<p>Synced <strong>${d}</strong> documents to <code>${y.tableName}</code>.</p>
|
|
71
|
+
${c>0?`<p class="badge badge-warn">${c} error(s)</p>`:""}
|
|
72
|
+
${O}
|
|
73
|
+
</div>`);te(b,A);})),g.viewQueue&&k.get(`${u}/queues`,(w,b)=>{let m=ye(w,u),y=[];for(let c of h){let a=n.get(c.name);y.push({repo:c.name,table:c.tableName,pending:a?a.size:0});}if(Ie(w)){ge(b,{queues:y});return}let i=y.map(c=>`<tr><td>${c.repo}</td><td>${c.table}</td><td>${c.pending===0?'<span class="badge badge-ok">0</span>':`<span class="badge badge-warn">${c.pending}</span>`}</td></tr>`).join(`
|
|
74
|
+
`),d=ee("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>${
|
|
77
|
+
<tbody>${i}</tbody>
|
|
78
78
|
</table>
|
|
79
|
-
</div>`);
|
|
80
|
-
`),console:`${
|
|
81
|
-
<td>${
|
|
82
|
-
<td><strong>${T.name}</strong><br><span class="muted">${T.message}</span>${
|
|
79
|
+
</div>`);te(b,d);}),g.configCheck&&k.get(`${u}/config-check`,async(w,b)=>{let m=ye(w,u),y=process.env.GCLOUD_PROJECT??process.env.GOOGLE_CLOUD_PROJECT??process.env.GCP_PROJECT??"unknown",i="https://console.cloud.google.com",d=p??"firestore-sync",c=[];try{await t.tableExists("__nonexistent_health_check__"),c.push({name:"BigQuery API",category:"bigquery",status:"ok",message:"BigQuery API is reachable"});}catch(x){let R=x?.message??String(x),$=R.toLowerCase(),T=$.includes("disabled")||$.includes("has not been used")||$.includes("accessnotconfigured"),P=$.includes("permission")||R.includes("403")||$.includes("access denied"),C=$.includes("project")&&$.includes("not found"),B=$.includes("not found")||R.includes("404");T?c.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:`${i}/apis/library/bigquery.googleapis.com?project=${y}`}}):C?c.push({name:"BigQuery Project",category:"bigquery",status:"error",message:R,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:`${i}/home/dashboard`}}):P?c.push({name:"BigQuery API",category:"bigquery",status:"error",message:`Permission denied: ${R}`,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:`${i}/iam-admin/iam?project=${y}`}}):B?c.push({name:"BigQuery Dataset",category:"bigquery",status:"error",message:`Dataset not found: ${R}`,fix:{hint:"Create the dataset first",gcloud:`bq mk --dataset ${y}:YOUR_DATASET_ID`,console:`${i}/bigquery?project=${y}`}}):c.push({name:"BigQuery API",category:"bigquery",status:"ok",message:"BigQuery API is reachable (table lookup returned expected error)"});}for(let x of h)try{let R=await t.tableExists(x.tableName);c.push({name:`Table: ${x.tableName}`,category:"bigquery",status:R?"ok":"warn",message:R?`Table \`${x.tableName}\` exists`:`Table \`${x.tableName}\` does not exist yet`,...!R&&{fix:{hint:"Table will be auto-created on first sync if autoMigrate is enabled. Or create it manually."}}});}catch(R){c.push({name:`Table: ${x.tableName}`,category:"bigquery",status:"error",message:R?.message??String(R)});}if(l)for(let x of h){let R=`${d}-${x.name}`;try{let $=l.topic(R);if(typeof $.exists=="function"){let[T]=await $.exists();c.push({name:`Topic: ${R}`,category:"pubsub",status:T?"ok":"error",message:T?`Topic \`${R}\` exists`:`Topic \`${R}\` does not exist`,...!T&&{fix:{gcloud:`gcloud pubsub topics create ${R} --project=${y}`,console:`${i}/cloudpubsub/topic/list?project=${y}`}}});}else c.push({name:`Topic: ${R}`,category:"pubsub",status:"warn",message:"Cannot verify topic existence (PubSub client doesn't expose .exists())",fix:{gcloud:`gcloud pubsub topics create ${R} --project=${y}`,console:`${i}/cloudpubsub/topic/list?project=${y}`,hint:"Ensure the topic exists. It is auto-created by the Firebase emulator but must exist in production."}});}catch($){let T=$?.message??String($),P=T.includes("disabled")||T.includes("has not been used");if(c.push({name:P?"Pub/Sub API":`Topic: ${R}`,category:"pubsub",status:"error",message:P?"Pub/Sub API is not enabled":T,fix:P?{gcloud:`gcloud services enable pubsub.googleapis.com --project=${y}`,console:`${i}/apis/library/pubsub.googleapis.com?project=${y}`}:{gcloud:`gcloud pubsub topics create ${R} --project=${y}`,console:`${i}/cloudpubsub/topic/list?project=${y}`}}),P)break}}else c.push({name:"Pub/Sub Client",category:"pubsub",status:"warn",message:"PubSub client not available for config check"});if(Ie(w)){let x=c.every(R=>R.status==="ok");ge(b,{project:y,healthy:x,checks:c});return}let a=x=>x==="ok"?'<span class="badge badge-ok">OK</span>':x==="warn"?'<span class="badge badge-warn">WARN</span>':'<span class="badge badge-err">ERROR</span>',f={bigquery:c.filter(x=>x.category==="bigquery"),pubsub:c.filter(x=>x.category==="pubsub"),firestore:c.filter(x=>x.category==="firestore")},v=(x,R)=>{if(R.length===0)return "";let $=R.map(T=>{let P="";if(T.fix){let C=[];T.fix.hint&&C.push(`<p class="muted">${T.fix.hint}</p>`),T.fix.gcloud&&C.push(`<pre>$ ${T.fix.gcloud}</pre>`),T.fix.console&&C.push(`<p><a href="${T.fix.console}" target="_blank">Open GCP Console \u2192</a></p>`),P=`<div style="margin-top:.5rem">${C.join("")}</div>`;}return `<tr>
|
|
81
|
+
<td>${a(T.status)}</td>
|
|
82
|
+
<td><strong>${T.name}</strong><br><span class="muted">${T.message}</span>${P}</td>
|
|
83
83
|
</tr>`}).join(`
|
|
84
|
-
`);return `<h2>${
|
|
84
|
+
`);return `<h2>${x}</h2>
|
|
85
85
|
<table><thead><tr><th style="width:80px">Status</th><th>Check</th></tr></thead>
|
|
86
|
-
<tbody>${
|
|
87
|
-
<p>Project: <code>${
|
|
88
|
-
${
|
|
89
|
-
${
|
|
90
|
-
${
|
|
91
|
-
</div>`);
|
|
92
|
-
<input type="hidden" id="${r}__isnull" name="${
|
|
86
|
+
<tbody>${$}</tbody></table>`},O=c.every(x=>x.status==="ok")?'<span class="badge badge-ok">All checks passed</span>':'<span class="badge badge-warn">Some issues found</span>',A=ee("Config Check",m,`<div class="card">
|
|
87
|
+
<p>Project: <code>${y}</code> ${O}</p>
|
|
88
|
+
${v("BigQuery",f.bigquery)}
|
|
89
|
+
${v("Pub/Sub",f.pubsub)}
|
|
90
|
+
${v("Firestore",f.firestore)}
|
|
91
|
+
</div>`);te(b,A);}),async(w,b)=>{await k.handle(w,b);}}var zn="firestore-sync";function Zn(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 _t(e,t){let{onDocumentCreated:n,onDocumentUpdated:r,onDocumentDeleted:s}=t.deps.firestoreTriggers,o=t.deps.pubsub,l=t?.topicPrefix??zn,p={},u=new Map;function g(k){let w=u.get(k);return w||(w=o.topic(k),u.set(k,w),w)}async function h(k,w){await g(k).publishMessage({json:w});}for(let[k,w]of Object.entries(e)){let b=t?.repos?.[k],m;if(w._isGroup){if(!b?.triggerPath){console.warn(`[SyncTriggers] Skipping collection-group repo "${k}". Provide a triggerPath in the sync repos config for group collections.`);continue}m=b.triggerPath;}else m=b?.triggerPath??Zn(k,w);if(!m)continue;let y=w._systemKeys?.[0]??"docId",i=`${l}-${k}`;p[`${k}_onCreate`]=n(m,async d=>{let c=d.data;if(!c)return;let a=c.data();if(!a)return;let f=String(a[y]??c.id),v=Ne(a,{exclude:b?.exclude,columnMap:b?.columnMap}),S={operation:"INSERT",repoName:k,docId:f,data:v,timestamp:new Date().toISOString(),version:Date.now()};await h(i,S);}),p[`${k}_onUpdate`]=r(m,async d=>{let c=d.data?.after;if(!c)return;let a=c.data();if(!a)return;let f=String(a[y]??c.id),v=Ne(a,{exclude:b?.exclude,columnMap:b?.columnMap}),S={operation:"UPSERT",repoName:k,docId:f,data:v,timestamp:new Date().toISOString(),version:Date.now()};await h(i,S);}),p[`${k}_onDelete`]=s(m,async d=>{let c=d.data;if(!c)return;let a=c.data(),f=String(a?.[y]??c.id),v={operation:"DELETE",repoName:k,docId:f,data:null,timestamp:new Date().toISOString(),version:Date.now()};await h(i,v);});}return p}var He=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 l=n.get(o.docId);if(!l)n.set(o.docId,o.data);else {let p=Number(l[ae]??0);Number(o.data[ae]??0)>=p&&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 Dt=new Set;async function Fn(e,t,n,r,s,o,l){if(Dt.has(e))return;let p=Le(n,t.dialect,{primaryKey:s,exclude:o,columnMap:l});if(!await t.tableExists(r))await t.createTable({tableName:r,columns:p});else {let g=new Set(await t.getTableColumns(r)),h=p.filter(k=>!g.has(k.name));h.length>0&&await t.addColumns(r,h);}Dt.add(e);}function jt(e,t){let{deps:n,adapter:r,batchSize:s=100,flushIntervalMs:o=5e3,autoMigrate:l=false,topicPrefix:p="firestore-sync",workerOptions:u,repos:g={}}=t,h=new Map;function k(m,y){let i=h.get(m);if(i)return i;let c=g[m]?.tableName??m,a=async(f,v)=>{console.error(`[SyncWorker] Flush failed for "${m}" (${f.length} events):`,v);try{let S=`${p}-${m}-dlq`,O=n.pubsub.topic(S),[A]=await O.exists();A||(await O.create(),console.info(`[SyncWorker] Created DLQ topic "${S}"`));for(let x of f)await O.publishMessage({json:x});}catch(S){console.error(`[SyncWorker] Dead-letter publish also failed for ${m}:`,S);}};return i=new He({adapter:r,tableName:c,primaryKey:y,batchSize:s,flushIntervalMs:o,onFlushError:a}),h.set(m,i),i}async function w(m){let{repoName:y}=m,i=e[y];if(!i){console.warn(`[SyncWorker] Unknown repo "${y}", skipping event`);return}let d=i._systemKeys?.[0]??i.documentKey??"docId",c=g[y],a=c?.columnMap,f=a?.[d]??d;if(l){let S=i.schema??void 0;if(S){let O=c?.tableName??y;await Fn(y,r,S,O,d,c?.exclude,a);}}let v=k(y,f);m.data&&(m.data[ae]=m.version??Date.now()),v.enqueue(m);}function b(m){let y=async i=>{let d=i.data?.message?.json??i.data?.json;if(!d){console.warn("[SyncWorker] Received empty PubSub message");return}await w(d);let c=h.get(d.repoName);c&&await c.flush();};return u?n.pubsubHandler.onMessagePublished({topic:m,...u},y):n.pubsubHandler.onMessagePublished(m,y)}return {handleMessage:w,createHandler:b,queues:h,async shutdown(){let m=[];for(let y of h.values())m.push(y.shutdown());await Promise.all(m);}}}var qn="firestore-sync";function zt(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 Zt(e,t){let{deps:n,adapter:r,topicPrefix:s=qn,batchSize:o,flushIntervalMs:l,autoMigrate:p,admin:u,workerOptions:g,repos:h}=t,k=zt(n.pubsub),w=zt(r),b=_t(e,{deps:{firestoreTriggers:n.firestoreTriggers,pubsub:k},topicPrefix:s,repos:h}),m=jt(e,{deps:{pubsubHandler:n.pubsubHandler,pubsub:k},adapter:w,batchSize:o,flushIntervalMs:l,autoMigrate:p,topicPrefix:s,workerOptions:g,repos:h}),y={};for(let c of Object.keys(e))y[`sync_${c}`]=m.createHandler(`${s}-${c}`);let i=null;u&&(i=It(e,w,m.queues,m.handleMessage,u,h??{},k,s),y.adminsync=u.onRequest?u.httpsOptions?u.onRequest(u.httpsOptions,i):u.onRequest(i):i);let d={functions:{...b,...y},adminHandler:i,handleMessage:m.handleMessage,queues:m.queues,shutdown:m.shutdown};for(let c of ["adminHandler","handleMessage","queues","shutdown"])Object.defineProperty(d,c,{enumerable:false});return d}Pe();Pe();function Bn(e){return e.replace(/([A-Z])/g," $1").replace(/_/g," ").replace(/^\s/,"").replace(/^./,t=>t.toUpperCase())}function Ft(e){let t=e,n=true,r=false,s;for(;;){let o=q(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=Tt(t),t=U(t);else break}return {inner:t,required:n,nullable:r,defaultValue:s}}function ne(e,t=""){if(q(e)==="ZodObject"){let r=Q(e);return Object.entries(r).map(([s,o])=>qt(t?`${t}.${s}`:s,s,o))}return [qt(t||"value",t||"value",e)]}function qt(e,t,n){let{inner:r,required:s,nullable:o,defaultValue:l}=Ft(n),p=q(r),u=Bn(t.split(".").pop()??t);switch(p){case "ZodString":{let g=At(r),h=g.includes("email"),k=g.includes("url");return {name:e,label:u,type:"text",required:s,nullable:o,defaultValue:l,hint:h?"email":k?"url":void 0}}case "ZodNumber":case "ZodBigInt":return {name:e,label:u,type:"number",required:s,nullable:o,defaultValue:l};case "ZodBoolean":return {name:e,label:u,type:"checkbox",required:s,nullable:o,defaultValue:l};case "ZodDate":case "ZodCoerce":return {name:e,label:u,type:"datetime-local",required:s,nullable:o,defaultValue:l};case "ZodEnum":{let g=Ae(r);return {name:e,label:u,type:"select",required:s,nullable:o,defaultValue:l,options:g}}case "ZodNativeEnum":{let g=Ee(r),h=Object.values(g).filter(k=>typeof k=="string");return {name:e,label:u,type:"select",required:s,nullable:o,defaultValue:l,options:h}}case "ZodLiteral":{let g=String(Ue(r)??"");return {name:e,label:u,type:"select",required:s,nullable:o,defaultValue:l,options:[g]}}case "ZodObject":{let g=ne(r,e);return {name:e,label:u,type:"textarea",required:s,nullable:o,defaultValue:l,nested:g,hint:"JSON object"}}case "ZodArray":{let g=Ot(r);if(!g)return {name:e,label:u,type:"textarea",required:s,nullable:o,defaultValue:l,hint:"JSON array"};let{inner:h}=Ft(g),k=q(h),w,b,m;switch(k){case "ZodString":w="text";break;case "ZodNumber":case "ZodBigInt":w="number";break;case "ZodBoolean":w="checkbox";break;case "ZodDate":w="datetime-local";break;case "ZodEnum":w="select",b=Ae(h);break;case "ZodNativeEnum":w="select",b=Object.values(Ee(h)).filter(y=>typeof y=="string");break;case "ZodObject":w="object",m=ne(h);break;default:return {name:e,label:u,type:"textarea",required:s,nullable:o,defaultValue:l,hint:"JSON array"}}return {name:e,label:u,type:"textarea",required:s,nullable:o,defaultValue:l,arrayElementType:w,arrayElementOptions:b,arrayElementFields:m}}default:return {name:e,label:u,type:"textarea",required:s,nullable:o,defaultValue:l,hint:"JSON"}}}function We(e,t=0){let n=t>0?`ml-${t*4}`:"",r=`field_${e.name.replace(/\./g,"__")}`,s=e.name,o=e.required?" required":"",l=e.defaultValue==="__null__",p=!l&&e.defaultValue!=null?String(e.defaultValue):"",u=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="${l?"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
|
-
<input type="checkbox" class="checkbox checkbox-xs" ${
|
|
94
|
+
<input type="checkbox" class="checkbox checkbox-xs" ${l?"checked":""}
|
|
95
95
|
onchange="(function(cb){
|
|
96
96
|
var inp = document.getElementById('${r}');
|
|
97
97
|
var h = document.getElementById('${r}__isnull');
|
|
@@ -100,61 +100,61 @@ ${n}
|
|
|
100
100
|
})(this)">
|
|
101
101
|
<span>null</span>
|
|
102
102
|
</label>
|
|
103
|
-
</span>`:"",
|
|
103
|
+
</span>`:"",g;switch(e.type){case "checkbox":if(e.nullable){let h=l?"__null__":p==="true"?"true":p==="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
|
-
${
|
|
107
|
+
${N(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="${
|
|
112
|
-
<option value="__null__"${
|
|
113
|
-
<option value="true"${
|
|
114
|
-
<option value="false"${
|
|
111
|
+
<select id="${r}" name="${s}" class="select select-bordered select-sm w-full">
|
|
112
|
+
<option value="__null__"${h==="__null__"?" selected":""}>\u2014 null \u2014</option>
|
|
113
|
+
<option value="true"${h==="true"?" selected":""}>\u2713 true</option>
|
|
114
|
+
<option value="false"${h==="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="${
|
|
119
|
+
<input type="checkbox" id="${r}" name="${s}" value="true"${p==="true"?" checked":""} class="checkbox checkbox-primary checkbox-sm">
|
|
120
120
|
<span class="label-text font-medium">
|
|
121
|
-
${
|
|
121
|
+
${N(e.label)}${e.required?' <span class="text-error">*</span>':""}
|
|
122
122
|
</span>
|
|
123
123
|
</label>
|
|
124
|
-
</div>`;case "select":
|
|
124
|
+
</div>`;case "select":g=`<select id="${r}" name="${s}"${o}${l?' 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(
|
|
126
|
+
${(e.options??[]).map(h=>`<option value="${N(h)}"${p===h?" selected":""}>${N(h)}</option>`).join(`
|
|
127
127
|
`)}
|
|
128
|
-
</select>`;break;case "textarea":if(e.arrayElementType)return
|
|
128
|
+
</select>`;break;case "textarea":if(e.arrayElementType)return Mn(e,t);if(e.nested&&e.nested.length>0){let h=e.nested.map(k=>We(k,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
|
-
${
|
|
132
|
+
${N(e.label)}${e.required?' <span class="text-error">*</span>':""}
|
|
133
133
|
</legend>
|
|
134
|
-
${
|
|
135
|
-
</fieldset>`}
|
|
134
|
+
${h}
|
|
135
|
+
</fieldset>`}g=`<textarea id="${r}" name="${s}"${o} rows="3"${l?' 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="${
|
|
139
|
-
value="${
|
|
138
|
+
placeholder="${N(e.hint??"JSON")}">${N(p)}</textarea>`;break;default:g=`<input type="${e.type}" id="${r}" name="${s}"${o}${l?' disabled style="opacity:0.35"':""}
|
|
139
|
+
value="${N(p)}"
|
|
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
|
-
${
|
|
145
|
-
${e.hint?`<span class="text-base-content/40 text-xs ml-1">(${
|
|
144
|
+
${N(e.label)}${e.required?' <span class="text-error">*</span>':""}
|
|
145
|
+
${e.hint?`<span class="text-base-content/40 text-xs ml-1">(${N(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">${
|
|
150
|
-
${
|
|
149
|
+
<div class="flex-1 min-w-0">${g}</div>
|
|
150
|
+
${u}
|
|
151
151
|
</div>
|
|
152
|
-
</div>`}function
|
|
153
|
-
`):
|
|
154
|
-
`),
|
|
155
|
-
<input type="hidden" id="${r}__isnull" name="${
|
|
152
|
+
</div>`}function N(e){return e.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""").replace(/'/g,"'")}function Mn(e,t){let n=t>0?`ml-${t*4}`:"",r=`field_${e.name.replace(/\./g,"__")}`,s=e.defaultValue==="__null__",o=e.arrayElementType==="object",l=[];if(e.defaultValue!=null&&e.defaultValue!==""&&e.defaultValue!=="__null__")try{l=JSON.parse(String(e.defaultValue));}catch{}Array.isArray(l)||(l=[]);let p=o?l.map(h=>Mt(e,h??{})).join(`
|
|
153
|
+
`):l.map(h=>Bt(e,h)).join(`
|
|
154
|
+
`),u=o?Mt(e,{}):Bt(e,""),g=e.nullable?`<span class="flex items-center gap-1 mt-2">
|
|
155
|
+
<input type="hidden" id="${r}__isnull" name="${N(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" ${
|
|
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,549 @@ ${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="${
|
|
169
|
+
data-frs-array="${N(e.name)}" data-frs-array-type="${N(e.arrayElementType??"text")}">
|
|
170
170
|
<legend class="fieldset-legend text-xs font-semibold text-base-content/60 px-1">
|
|
171
|
-
${
|
|
171
|
+
${N(e.label)}${e.required?' <span class="text-error">*</span>':""}
|
|
172
172
|
</legend>
|
|
173
|
-
<input type="hidden" id="${r}" name="${
|
|
174
|
-
<div data-frs-array-items${
|
|
175
|
-
${
|
|
173
|
+
<input type="hidden" id="${r}" name="${N(e.name)}" value="${N(JSON.stringify(l))}"${s?" disabled":""}>
|
|
174
|
+
<div data-frs-array-items${s?' style="opacity:0.35"':""}>
|
|
175
|
+
${p}
|
|
176
176
|
</div>
|
|
177
|
-
<template data-frs-array-tpl>${
|
|
177
|
+
<template data-frs-array-tpl>${u}</template>
|
|
178
178
|
<button type="button" class="btn btn-xs btn-outline mt-1" data-frs-array-add>+ Add</button>
|
|
179
|
-
${
|
|
180
|
-
</fieldset>`}function
|
|
179
|
+
${g}
|
|
180
|
+
</fieldset>`}function Bt(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(
|
|
182
|
+
${(e.arrayElementOptions??[]).map(s=>`<option value="${N(s)}"${n===s?" selected":""}>${N(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="${
|
|
186
|
+
</label>`;break;case "number":r=`<input type="number" data-frs-val value="${N(n)}" class="input input-bordered input-sm flex-1">`;break;case "datetime-local":r=`<input type="datetime-local" data-frs-val value="${N(n)}" class="input input-bordered input-sm flex-1">`;break;default:r=`<input type="text" data-frs-val value="${N(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>×</button>
|
|
189
|
-
</div>`}function
|
|
189
|
+
</div>`}function Mt(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>×</button>
|
|
192
192
|
</div>
|
|
193
|
-
${(e.arrayElementFields??[]).map(
|
|
193
|
+
${(e.arrayElementFields??[]).map(s=>{let o=t[s.name],l=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="${
|
|
196
|
-
<span class="label-text text-sm">${
|
|
195
|
+
<input type="checkbox" data-frs-key="${N(s.name)}" class="checkbox checkbox-sm checkbox-primary"${l==="true"?" checked":""}>
|
|
196
|
+
<span class="label-text text-sm">${N(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">${
|
|
200
|
-
<select data-frs-key="${
|
|
201
|
-
${
|
|
202
|
-
${(
|
|
199
|
+
<label class="label pb-1"><span class="label-text text-sm">${N(s.label)}</span></label>
|
|
200
|
+
<select data-frs-key="${N(s.name)}" class="select select-bordered select-sm w-full">
|
|
201
|
+
${s.required?"":'<option value="">\u2014</option>'}
|
|
202
|
+
${(s.options??[]).map(p=>`<option value="${N(p)}"${l===p?" selected":""}>${N(p)}</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">${
|
|
206
|
-
<input type="number" data-frs-key="${
|
|
205
|
+
<label class="label pb-1"><span class="label-text text-sm">${N(s.label)}</span></label>
|
|
206
|
+
<input type="number" data-frs-key="${N(s.name)}" value="${N(l)}" 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">${
|
|
209
|
-
<input type="datetime-local" data-frs-key="${
|
|
208
|
+
<label class="label pb-1"><span class="label-text text-sm">${N(s.label)}</span></label>
|
|
209
|
+
<input type="datetime-local" data-frs-key="${N(s.name)}" value="${N(l)}" 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">${
|
|
212
|
-
<textarea data-frs-key="${
|
|
211
|
+
<label class="label pb-1"><span class="label-text text-sm">${N(s.label)}</span></label>
|
|
212
|
+
<textarea data-frs-key="${N(s.name)}" rows="2" class="textarea textarea-bordered textarea-sm w-full font-mono text-xs" placeholder="JSON">${N(l)}</textarea>
|
|
213
213
|
</div>`;default:return `<div class="form-control mb-2">
|
|
214
|
-
<label class="label pb-1"><span class="label-text text-sm">${
|
|
215
|
-
<input type="text" data-frs-key="${
|
|
214
|
+
<label class="label pb-1"><span class="label-text text-sm">${N(s.label)}</span></label>
|
|
215
|
+
<input type="text" data-frs-key="${N(s.name)}" value="${N(l)}" class="input input-bordered input-sm w-full">
|
|
216
216
|
</div>`}}).join(`
|
|
217
217
|
`)}
|
|
218
|
-
</div>`}function
|
|
218
|
+
</div>`}function ie(e,t,n,r="Save"){let s=e.map(o=>We(o)).join(`
|
|
219
219
|
`);return `
|
|
220
|
-
<form action="${
|
|
221
|
-
${
|
|
220
|
+
<form action="${N(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">${
|
|
223
|
+
<button type="submit" class="btn btn-primary btn-sm">${N(r)}</button>
|
|
224
224
|
<button type="button" class="btn btn-ghost btn-sm" onclick="history.back()">Cancel</button>
|
|
225
225
|
</div>
|
|
226
|
-
</form>`}
|
|
226
|
+
</form>`}Ke();var Lt=`// \u2500\u2500 Shared helpers \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
|
|
227
|
+
function frsGetBasePath() {
|
|
228
|
+
var root = document.querySelector("[data-frs-panel-root]");
|
|
229
|
+
var bp = root && root.getAttribute("data-frs-base-path");
|
|
230
|
+
if (typeof bp === "string") return bp.replace(/\\/$/, "");
|
|
231
|
+
return window.location.pathname.replace(/\\/[^/]*\\/?$/, "");
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// \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
|
|
235
|
+
(function () {
|
|
236
|
+
function panelEls() {
|
|
237
|
+
return {
|
|
238
|
+
root: document.querySelector("[data-frs-panel-root]"),
|
|
239
|
+
backdrop: document.querySelector("[data-frs-panel-backdrop]"),
|
|
240
|
+
panel: document.querySelector("[data-frs-panel]"),
|
|
241
|
+
title: document.querySelector("[data-frs-panel-title]"),
|
|
242
|
+
body: document.querySelector("[data-frs-panel-body]"),
|
|
243
|
+
};
|
|
244
|
+
}
|
|
245
|
+
function openPanel(label) {
|
|
246
|
+
var els = panelEls();
|
|
247
|
+
if (!els.root || !els.panel) return;
|
|
248
|
+
els.root.classList.remove("hidden");
|
|
249
|
+
els.root.setAttribute("aria-hidden", "false");
|
|
250
|
+
requestAnimationFrame(function () {
|
|
251
|
+
if (els.backdrop) els.backdrop.style.opacity = "1";
|
|
252
|
+
if (els.panel) {
|
|
253
|
+
els.panel.classList.remove("translate-x-full");
|
|
254
|
+
els.panel.style.transform = "translateX(0)";
|
|
255
|
+
els.panel.style.translate = "0 0";
|
|
256
|
+
}
|
|
257
|
+
});
|
|
258
|
+
if (els.title) els.title.textContent = label || "Relation";
|
|
259
|
+
if (els.body) {
|
|
260
|
+
els.body.innerHTML =
|
|
261
|
+
'<div class="flex items-center justify-center py-12 text-base-content/40"><span class="loading loading-spinner loading-md"></span></div>';
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
function closePanel() {
|
|
265
|
+
var els = panelEls();
|
|
266
|
+
if (!els.root || !els.panel) return;
|
|
267
|
+
if (els.backdrop) els.backdrop.style.opacity = "0";
|
|
268
|
+
if (els.panel) {
|
|
269
|
+
els.panel.style.transform = "translateX(100%)";
|
|
270
|
+
els.panel.style.translate = "100% 0";
|
|
271
|
+
els.panel.classList.add("translate-x-full");
|
|
272
|
+
}
|
|
273
|
+
setTimeout(function () {
|
|
274
|
+
els.root.classList.add("hidden");
|
|
275
|
+
els.root.setAttribute("aria-hidden", "true");
|
|
276
|
+
}, 200);
|
|
277
|
+
}
|
|
278
|
+
function getBasePath() {
|
|
279
|
+
return frsGetBasePath();
|
|
280
|
+
}
|
|
281
|
+
function fetchPanel(url, label) {
|
|
282
|
+
openPanel(label);
|
|
283
|
+
fetch(url, { credentials: "same-origin" })
|
|
284
|
+
.then(function (r) {
|
|
285
|
+
return r.text();
|
|
286
|
+
})
|
|
287
|
+
.then(function (html) {
|
|
288
|
+
var els = panelEls();
|
|
289
|
+
if (els.body) els.body.innerHTML = html;
|
|
290
|
+
})
|
|
291
|
+
.catch(function (err) {
|
|
292
|
+
var els = panelEls();
|
|
293
|
+
if (els.body) {
|
|
294
|
+
els.body.innerHTML =
|
|
295
|
+
'<div class="p-6 text-error text-sm">Error: ' +
|
|
296
|
+
(err && err.message ? err.message : String(err)) +
|
|
297
|
+
"</div>";
|
|
298
|
+
}
|
|
299
|
+
});
|
|
300
|
+
}
|
|
301
|
+
document.addEventListener("click", function (e) {
|
|
302
|
+
var trigger = e.target.closest("[data-frs-relation]");
|
|
303
|
+
if (trigger) {
|
|
304
|
+
e.preventDefault();
|
|
305
|
+
var type = trigger.getAttribute("data-frs-rel-type");
|
|
306
|
+
var repo = trigger.getAttribute("data-frs-rel-repo");
|
|
307
|
+
var fk = trigger.getAttribute("data-frs-rel-fk");
|
|
308
|
+
var val = trigger.getAttribute("data-frs-rel-val");
|
|
309
|
+
var label = trigger.getAttribute("data-frs-rel-label") || "Relation";
|
|
310
|
+
var bp = getBasePath();
|
|
311
|
+
var url;
|
|
312
|
+
if (type === "one") {
|
|
313
|
+
url =
|
|
314
|
+
bp +
|
|
315
|
+
"/" +
|
|
316
|
+
encodeURIComponent(repo) +
|
|
317
|
+
"/_panel?type=one&id=" +
|
|
318
|
+
encodeURIComponent(val);
|
|
319
|
+
} else {
|
|
320
|
+
url =
|
|
321
|
+
bp +
|
|
322
|
+
"/" +
|
|
323
|
+
encodeURIComponent(repo) +
|
|
324
|
+
"/_panel?type=many&fk=" +
|
|
325
|
+
encodeURIComponent(fk) +
|
|
326
|
+
"&fv=" +
|
|
327
|
+
encodeURIComponent(val);
|
|
328
|
+
}
|
|
329
|
+
fetchPanel(url, repo + " \xB7 " + label);
|
|
330
|
+
return;
|
|
331
|
+
}
|
|
332
|
+
if (e.target.closest("[data-frs-panel-close]")) {
|
|
333
|
+
closePanel();
|
|
334
|
+
return;
|
|
335
|
+
}
|
|
336
|
+
if (e.target.closest("[data-frs-panel-backdrop]")) {
|
|
337
|
+
closePanel();
|
|
338
|
+
return;
|
|
339
|
+
}
|
|
340
|
+
var pageBtn = e.target.closest("[data-frs-panel-page]");
|
|
341
|
+
if (pageBtn) {
|
|
342
|
+
// Recompute URL by inspecting the current open panel context \u2014 encoded in the body's first anchor "Full view \u2192"
|
|
343
|
+
// Simpler: rebuild via the previous URL stored in data-attr.
|
|
344
|
+
var fullViewLink = document.querySelector(
|
|
345
|
+
"[data-frs-panel-body] a.btn-outline[href]",
|
|
346
|
+
);
|
|
347
|
+
if (!fullViewLink) return;
|
|
348
|
+
var dir = pageBtn.getAttribute("data-frs-panel-page");
|
|
349
|
+
var cursor = pageBtn.getAttribute("data-cursor") || "";
|
|
350
|
+
var fullViewUrl = new URL(fullViewLink.href, window.location.href);
|
|
351
|
+
var repo = fullViewUrl.pathname.split("/").filter(Boolean).pop();
|
|
352
|
+
var bp = fullViewUrl.pathname.replace(/\\/[^/]+\\/?$/, "");
|
|
353
|
+
var fk = "";
|
|
354
|
+
var fv = "";
|
|
355
|
+
fullViewUrl.searchParams.forEach(function (v, k) {
|
|
356
|
+
if (k.indexOf("fv_") === 0) {
|
|
357
|
+
fk = k.slice(3);
|
|
358
|
+
fv = v;
|
|
359
|
+
}
|
|
360
|
+
});
|
|
361
|
+
var url =
|
|
362
|
+
bp +
|
|
363
|
+
"/" +
|
|
364
|
+
encodeURIComponent(repo) +
|
|
365
|
+
"/_panel?type=many&fk=" +
|
|
366
|
+
encodeURIComponent(fk) +
|
|
367
|
+
"&fv=" +
|
|
368
|
+
encodeURIComponent(fv) +
|
|
369
|
+
"&cursor=" +
|
|
370
|
+
encodeURIComponent(cursor) +
|
|
371
|
+
"&dir=" +
|
|
372
|
+
encodeURIComponent(dir);
|
|
373
|
+
fetchPanel(url, repo);
|
|
374
|
+
}
|
|
375
|
+
});
|
|
376
|
+
document.addEventListener("keydown", function (e) {
|
|
377
|
+
if (e.key === "Escape") closePanel();
|
|
378
|
+
});
|
|
379
|
+
})();
|
|
380
|
+
|
|
381
|
+
// \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
|
|
382
|
+
(function () {
|
|
383
|
+
var state = {
|
|
384
|
+
selectAllAcrossQuery: false,
|
|
385
|
+
// Keep a Set of selected ids on the current page (cleared on navigation).
|
|
386
|
+
selected: new Set(),
|
|
387
|
+
};
|
|
388
|
+
|
|
389
|
+
function bar() {
|
|
390
|
+
return document.querySelector("[data-frs-bulk-bar]");
|
|
391
|
+
}
|
|
392
|
+
function summarize() {
|
|
393
|
+
var b = bar();
|
|
394
|
+
if (!b) return;
|
|
395
|
+
var total = parseInt(b.getAttribute("data-frs-total") || "0", 10);
|
|
396
|
+
var summary = b.querySelector("[data-frs-bulk-summary]");
|
|
397
|
+
var selectAllBtn = b.querySelector("[data-frs-bulk-select-all]");
|
|
398
|
+
var allActive = b.querySelector("[data-frs-bulk-all-active]");
|
|
399
|
+
var pageSize = parseInt(b.getAttribute("data-frs-page-size") || "0", 10);
|
|
400
|
+
var n = state.selected.size;
|
|
401
|
+
if (state.selectAllAcrossQuery) {
|
|
402
|
+
if (allActive) allActive.classList.remove("hidden");
|
|
403
|
+
if (selectAllBtn) selectAllBtn.classList.add("hidden");
|
|
404
|
+
if (summary) summary.textContent = "";
|
|
405
|
+
b.classList.remove("hidden");
|
|
406
|
+
return;
|
|
407
|
+
}
|
|
408
|
+
if (n === 0) {
|
|
409
|
+
b.classList.add("hidden");
|
|
410
|
+
if (selectAllBtn) selectAllBtn.classList.add("hidden");
|
|
411
|
+
if (allActive) allActive.classList.add("hidden");
|
|
412
|
+
return;
|
|
413
|
+
}
|
|
414
|
+
b.classList.remove("hidden");
|
|
415
|
+
if (allActive) allActive.classList.add("hidden");
|
|
416
|
+
if (summary) {
|
|
417
|
+
summary.textContent =
|
|
418
|
+
n + " selected" + (pageSize ? " on this page" : "");
|
|
419
|
+
}
|
|
420
|
+
if (selectAllBtn && total > n) {
|
|
421
|
+
selectAllBtn.classList.remove("hidden");
|
|
422
|
+
} else if (selectAllBtn) {
|
|
423
|
+
selectAllBtn.classList.add("hidden");
|
|
424
|
+
}
|
|
425
|
+
}
|
|
426
|
+
function syncHeader() {
|
|
427
|
+
var head = document.querySelector("[data-frs-select-page]");
|
|
428
|
+
if (!head) return;
|
|
429
|
+
var rows = document.querySelectorAll("[data-frs-select-row]");
|
|
430
|
+
var checked = 0;
|
|
431
|
+
rows.forEach(function (r) {
|
|
432
|
+
if (r.checked) checked++;
|
|
433
|
+
});
|
|
434
|
+
head.indeterminate = checked > 0 && checked < rows.length;
|
|
435
|
+
head.checked = rows.length > 0 && checked === rows.length;
|
|
436
|
+
}
|
|
437
|
+
document.addEventListener("change", function (e) {
|
|
438
|
+
if (e.target.matches("[data-frs-select-row]")) {
|
|
439
|
+
var id = e.target.value;
|
|
440
|
+
if (e.target.checked) state.selected.add(id);
|
|
441
|
+
else state.selected.delete(id);
|
|
442
|
+
state.selectAllAcrossQuery = false;
|
|
443
|
+
syncHeader();
|
|
444
|
+
summarize();
|
|
445
|
+
} else if (e.target.matches("[data-frs-select-page]")) {
|
|
446
|
+
var rows = document.querySelectorAll("[data-frs-select-row]");
|
|
447
|
+
rows.forEach(function (r) {
|
|
448
|
+
r.checked = e.target.checked;
|
|
449
|
+
if (e.target.checked) state.selected.add(r.value);
|
|
450
|
+
else state.selected.delete(r.value);
|
|
451
|
+
});
|
|
452
|
+
state.selectAllAcrossQuery = false;
|
|
453
|
+
summarize();
|
|
454
|
+
}
|
|
455
|
+
});
|
|
456
|
+
document.addEventListener("click", function (e) {
|
|
457
|
+
var sa = e.target.closest("[data-frs-bulk-select-all]");
|
|
458
|
+
if (sa) {
|
|
459
|
+
state.selectAllAcrossQuery = true;
|
|
460
|
+
summarize();
|
|
461
|
+
return;
|
|
462
|
+
}
|
|
463
|
+
var clear = e.target.closest("[data-frs-bulk-clear]");
|
|
464
|
+
if (clear) {
|
|
465
|
+
state.selectAllAcrossQuery = false;
|
|
466
|
+
state.selected.clear();
|
|
467
|
+
document
|
|
468
|
+
.querySelectorAll("[data-frs-select-row]")
|
|
469
|
+
.forEach(function (r) {
|
|
470
|
+
r.checked = false;
|
|
471
|
+
});
|
|
472
|
+
syncHeader();
|
|
473
|
+
summarize();
|
|
474
|
+
return;
|
|
475
|
+
}
|
|
476
|
+
var actBtn = e.target.closest("[data-frs-bulk-action]");
|
|
477
|
+
if (actBtn) {
|
|
478
|
+
var action = actBtn.getAttribute("data-frs-bulk-action");
|
|
479
|
+
if (action === "delete") doBulkDelete();
|
|
480
|
+
else if (action === "update") openBulkUpdateModal();
|
|
481
|
+
}
|
|
482
|
+
});
|
|
483
|
+
|
|
484
|
+
function buildPayload() {
|
|
485
|
+
var b = bar();
|
|
486
|
+
if (!b) return null;
|
|
487
|
+
if (state.selectAllAcrossQuery) {
|
|
488
|
+
var filters = [];
|
|
489
|
+
try {
|
|
490
|
+
filters = JSON.parse(b.getAttribute("data-frs-filters") || "[]");
|
|
491
|
+
} catch (_) {}
|
|
492
|
+
return { selectAll: true, filters: filters };
|
|
493
|
+
}
|
|
494
|
+
return { ids: Array.from(state.selected) };
|
|
495
|
+
}
|
|
496
|
+
|
|
497
|
+
function repoEndpoint(action) {
|
|
498
|
+
var b = bar();
|
|
499
|
+
if (!b) return null;
|
|
500
|
+
var repo = b.getAttribute("data-frs-repo");
|
|
501
|
+
var bp = frsGetBasePath();
|
|
502
|
+
return bp + "/" + encodeURIComponent(repo) + "/_bulk/" + action;
|
|
503
|
+
}
|
|
504
|
+
|
|
505
|
+
function doBulkDelete() {
|
|
506
|
+
var payload = buildPayload();
|
|
507
|
+
if (!payload) return;
|
|
508
|
+
var n = state.selectAllAcrossQuery
|
|
509
|
+
? "all matching"
|
|
510
|
+
: payload.ids.length + "";
|
|
511
|
+
if (!confirm("Delete " + n + " documents? This cannot be undone.")) return;
|
|
512
|
+
var url = repoEndpoint("delete");
|
|
513
|
+
if (!url) return;
|
|
514
|
+
fetch(url, {
|
|
515
|
+
method: "POST",
|
|
516
|
+
credentials: "same-origin",
|
|
517
|
+
headers: { "Content-Type": "application/json" },
|
|
518
|
+
body: JSON.stringify(payload),
|
|
519
|
+
})
|
|
520
|
+
.then(function (r) {
|
|
521
|
+
return r.json().then(function (j) {
|
|
522
|
+
return { ok: r.ok, body: j };
|
|
523
|
+
});
|
|
524
|
+
})
|
|
525
|
+
.then(function (res) {
|
|
526
|
+
if (!res.ok) {
|
|
527
|
+
alert("Bulk delete failed: " + (res.body && res.body.error));
|
|
528
|
+
return;
|
|
529
|
+
}
|
|
530
|
+
window.location.reload();
|
|
531
|
+
})
|
|
532
|
+
.catch(function (err) {
|
|
533
|
+
alert("Bulk delete failed: " + err.message);
|
|
534
|
+
});
|
|
535
|
+
}
|
|
536
|
+
|
|
537
|
+
function openBulkUpdateModal() {
|
|
538
|
+
var b = bar();
|
|
539
|
+
var dialog = document.getElementById("frs-bulk-update-modal");
|
|
540
|
+
if (!b || !dialog) return;
|
|
541
|
+
var fields = [];
|
|
542
|
+
try {
|
|
543
|
+
fields = JSON.parse(b.getAttribute("data-frs-fields") || "[]");
|
|
544
|
+
} catch (_) {}
|
|
545
|
+
if (fields.length === 0) return;
|
|
546
|
+
var summary = dialog.querySelector(
|
|
547
|
+
"[data-frs-bulk-update-summary]",
|
|
548
|
+
);
|
|
549
|
+
if (summary) {
|
|
550
|
+
summary.textContent = state.selectAllAcrossQuery
|
|
551
|
+
? "Update one field on all matching documents."
|
|
552
|
+
: "Update one field on " +
|
|
553
|
+
state.selected.size +
|
|
554
|
+
" selected document" +
|
|
555
|
+
(state.selected.size !== 1 ? "s" : "") +
|
|
556
|
+
".";
|
|
557
|
+
}
|
|
558
|
+
var select = dialog.querySelector("[data-frs-bulk-field-select]");
|
|
559
|
+
var valueContainer = dialog.querySelector(
|
|
560
|
+
"[data-frs-bulk-value-container]",
|
|
561
|
+
);
|
|
562
|
+
var renderValueInput = function () {
|
|
563
|
+
if (!valueContainer || !select) return;
|
|
564
|
+
var name = select.value;
|
|
565
|
+
var meta = fields.filter(function (f) {
|
|
566
|
+
return f.name === name;
|
|
567
|
+
})[0];
|
|
568
|
+
if (!meta) {
|
|
569
|
+
valueContainer.innerHTML = "";
|
|
570
|
+
return;
|
|
571
|
+
}
|
|
572
|
+
var inputHtml = "";
|
|
573
|
+
var label =
|
|
574
|
+
'<div class="label"><span class="label-text text-xs uppercase tracking-wide">Value</span></div>';
|
|
575
|
+
if (meta.type === "boolean") {
|
|
576
|
+
inputHtml =
|
|
577
|
+
'<select name="value" class="select select-bordered select-sm w-full">' +
|
|
578
|
+
'<option value="true">true</option><option value="false">false</option>' +
|
|
579
|
+
"</select>";
|
|
580
|
+
} else if (
|
|
581
|
+
(meta.type === "enum" || meta.type === "literal") &&
|
|
582
|
+
meta.enumValues &&
|
|
583
|
+
meta.enumValues.length
|
|
584
|
+
) {
|
|
585
|
+
inputHtml =
|
|
586
|
+
'<select name="value" class="select select-bordered select-sm w-full">';
|
|
587
|
+
meta.enumValues.forEach(function (v) {
|
|
588
|
+
inputHtml +=
|
|
589
|
+
'<option value="' + escapeAttr(v) + '">' + escapeHtml(v) + "</option>";
|
|
590
|
+
});
|
|
591
|
+
inputHtml += "</select>";
|
|
592
|
+
} else if (meta.type === "number" || meta.type === "bigint") {
|
|
593
|
+
inputHtml =
|
|
594
|
+
'<input type="number" step="any" name="value" class="input input-bordered input-sm w-full" />';
|
|
595
|
+
} else {
|
|
596
|
+
inputHtml =
|
|
597
|
+
'<input type="text" name="value" class="input input-bordered input-sm w-full" />';
|
|
598
|
+
}
|
|
599
|
+
valueContainer.innerHTML =
|
|
600
|
+
'<label class="form-control w-full">' + label + inputHtml + "</label>";
|
|
601
|
+
};
|
|
602
|
+
if (select) {
|
|
603
|
+
select.value = "";
|
|
604
|
+
select.onchange = renderValueInput;
|
|
605
|
+
}
|
|
606
|
+
if (valueContainer) valueContainer.innerHTML = "";
|
|
607
|
+
var form = dialog.querySelector("[data-frs-bulk-update-form]");
|
|
608
|
+
if (form) {
|
|
609
|
+
form.onsubmit = function (e) {
|
|
610
|
+
e.preventDefault();
|
|
611
|
+
if (!select || !select.value) return;
|
|
612
|
+
var meta = fields.filter(function (f) {
|
|
613
|
+
return f.name === select.value;
|
|
614
|
+
})[0];
|
|
615
|
+
if (!meta) return;
|
|
616
|
+
var input = valueContainer.querySelector(
|
|
617
|
+
'input[name="value"], select[name="value"]',
|
|
618
|
+
);
|
|
619
|
+
if (!input) return;
|
|
620
|
+
var raw = input.value;
|
|
621
|
+
var typed;
|
|
622
|
+
if (meta.type === "boolean") typed = raw === "true";
|
|
623
|
+
else if (meta.type === "number") typed = raw === "" ? null : Number(raw);
|
|
624
|
+
else if (meta.type === "bigint") typed = raw === "" ? null : Number(raw);
|
|
625
|
+
else typed = raw;
|
|
626
|
+
var payload = buildPayload();
|
|
627
|
+
if (!payload) return;
|
|
628
|
+
payload.field = select.value;
|
|
629
|
+
payload.value = typed;
|
|
630
|
+
var url = repoEndpoint("update");
|
|
631
|
+
if (!url) return;
|
|
632
|
+
fetch(url, {
|
|
633
|
+
method: "POST",
|
|
634
|
+
credentials: "same-origin",
|
|
635
|
+
headers: { "Content-Type": "application/json" },
|
|
636
|
+
body: JSON.stringify(payload),
|
|
637
|
+
})
|
|
638
|
+
.then(function (r) {
|
|
639
|
+
return r.json().then(function (j) {
|
|
640
|
+
return { ok: r.ok, body: j };
|
|
641
|
+
});
|
|
642
|
+
})
|
|
643
|
+
.then(function (res) {
|
|
644
|
+
if (!res.ok) {
|
|
645
|
+
alert("Bulk update failed: " + (res.body && res.body.error));
|
|
646
|
+
return;
|
|
647
|
+
}
|
|
648
|
+
dialog.close();
|
|
649
|
+
window.location.reload();
|
|
650
|
+
})
|
|
651
|
+
.catch(function (err) {
|
|
652
|
+
alert("Bulk update failed: " + err.message);
|
|
653
|
+
});
|
|
654
|
+
};
|
|
655
|
+
}
|
|
656
|
+
var cancel = dialog.querySelector("[data-frs-bulk-update-cancel]");
|
|
657
|
+
if (cancel)
|
|
658
|
+
cancel.onclick = function () {
|
|
659
|
+
dialog.close();
|
|
660
|
+
};
|
|
661
|
+
if (typeof dialog.showModal === "function") dialog.showModal();
|
|
662
|
+
else dialog.setAttribute("open", "");
|
|
663
|
+
}
|
|
664
|
+
|
|
665
|
+
function escapeHtml(s) {
|
|
666
|
+
return String(s).replace(/[&<>"']/g, function (c) {
|
|
667
|
+
return (
|
|
668
|
+
{ "&": "&", "<": "<", ">": ">", '"': """, "'": "'" }[
|
|
669
|
+
c
|
|
670
|
+
] || c
|
|
671
|
+
);
|
|
672
|
+
});
|
|
673
|
+
}
|
|
674
|
+
function escapeAttr(s) {
|
|
675
|
+
return escapeHtml(s);
|
|
676
|
+
}
|
|
677
|
+
})();
|
|
678
|
+
|
|
679
|
+
// \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
|
|
680
|
+
(function () {
|
|
681
|
+
function syncThemeUI() {
|
|
682
|
+
var current =
|
|
683
|
+
document.documentElement.getAttribute("data-theme") || "corporate";
|
|
684
|
+
var label = document.querySelector("[data-frs-theme-current]");
|
|
685
|
+
if (label) label.textContent = current;
|
|
686
|
+
document.querySelectorAll("[data-frs-theme-check]").forEach(function (el) {
|
|
687
|
+
el.classList.toggle(
|
|
688
|
+
"hidden",
|
|
689
|
+
el.getAttribute("data-frs-theme-check") !== current,
|
|
690
|
+
);
|
|
691
|
+
});
|
|
692
|
+
}
|
|
693
|
+
document.addEventListener("DOMContentLoaded", syncThemeUI);
|
|
694
|
+
document.addEventListener("click", function (e) {
|
|
695
|
+
var btn = e.target.closest("[data-frs-theme]");
|
|
696
|
+
if (!btn) return;
|
|
697
|
+
var theme = btn.getAttribute("data-frs-theme");
|
|
698
|
+
if (!theme) return;
|
|
699
|
+
document.documentElement.setAttribute("data-theme", theme);
|
|
700
|
+
try {
|
|
701
|
+
localStorage.setItem("frs-admin-theme", theme);
|
|
702
|
+
} catch (_) {}
|
|
703
|
+
syncThemeUI();
|
|
704
|
+
// Close the dropdown by blurring focus (DaisyUI dropdown closes on focusout)
|
|
705
|
+
if (document.activeElement && document.activeElement.blur) {
|
|
706
|
+
document.activeElement.blur();
|
|
707
|
+
}
|
|
708
|
+
});
|
|
709
|
+
})();
|
|
710
|
+
|
|
711
|
+
// \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
712
|
document.addEventListener("submit", function (e) {
|
|
228
713
|
var form = e.target;
|
|
229
714
|
if (!form.hasAttribute("data-frs-form")) return;
|
|
@@ -453,7 +938,12 @@ function initColumnVisibility(table) {
|
|
|
453
938
|
if (dataThs.length === 0) return;
|
|
454
939
|
|
|
455
940
|
var wrap = table.closest("[data-frs-table-wrap]");
|
|
456
|
-
var toolbar =
|
|
941
|
+
var toolbar =
|
|
942
|
+
(wrap &&
|
|
943
|
+
(wrap.parentElement
|
|
944
|
+
? wrap.parentElement.querySelector("[data-frs-toolbar]")
|
|
945
|
+
: null)) ||
|
|
946
|
+
(wrap && wrap.previousElementSibling);
|
|
457
947
|
if (!toolbar) return;
|
|
458
948
|
|
|
459
949
|
var repo = table.getAttribute("data-frs-repo") || "default";
|
|
@@ -642,7 +1132,7 @@ function initColumnReorder(table) {
|
|
|
642
1132
|
});
|
|
643
1133
|
});
|
|
644
1134
|
}
|
|
645
|
-
`;function we(){return jsxRuntime.jsx("script",{dangerouslySetInnerHTML:{__html:$t}})}function re(e){return "<!DOCTYPE html>"+server.renderToString(e)}var oe=({opts:e,children:t})=>{let{title:n,breadcrumb:r,flash:o,basePath:s="/"}=e;return jsxRuntime.jsxs("html",{lang:"en","data-theme":"corporate",children:[jsxRuntime.jsxs("head",{children:[jsxRuntime.jsx("meta",{charset:"UTF-8"}),jsxRuntime.jsx("meta",{name:"viewport",content:"width=device-width, initial-scale=1"}),jsxRuntime.jsxs("title",{children:[n," \u2014 FRS Admin"]}),jsxRuntime.jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/themes.css",rel:"stylesheet",type:"text/css"}),jsxRuntime.jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/daisyui.css",rel:"stylesheet",type:"text/css"}),jsxRuntime.jsx("script",{src:"https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4"})]}),jsxRuntime.jsxs("body",{class:"bg-base-200/50 min-h-screen flex flex-col",children:[jsxRuntime.jsx("div",{class:"navbar bg-neutral text-neutral-content shadow-sm sticky top-0 z-50 px-6",children:jsxRuntime.jsx("div",{class:"flex-1",children:jsxRuntime.jsx("a",{href:s,class:"font-bold text-lg tracking-tight hover:opacity-80 transition-opacity",children:"FRS Admin"})})}),jsxRuntime.jsxs("main",{class:"px-6 py-8 w-full flex-1",children:[r&&r.length>0&&jsxRuntime.jsx("div",{class:"text-sm breadcrumbs mb-4",children:jsxRuntime.jsx("ul",{children:r.map((a,h)=>a.href?jsxRuntime.jsx("li",{children:jsxRuntime.jsx("a",{href:a.href,children:a.label})},h):jsxRuntime.jsx("li",{class:"text-base-content/60",children:a.label},h))})}),jsxRuntime.jsx("h1",{class:"text-2xl font-bold mb-6",children:n}),o&&jsxRuntime.jsxs("div",{role:"alert",class:`alert ${o.type==="success"?"alert-success":o.type==="warning"?"alert-warning":"alert-error"} mb-6`,children:[jsxRuntime.jsx("span",{class:"flex-1",children:o.message}),o.action&&jsxRuntime.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]}),jsxRuntime.jsx(we,{})]})]})};function Fe(e,t){return re(jsxRuntime.jsx(oe,{opts:t,children:jsxRuntime.jsx("div",{dangerouslySetInnerHTML:{__html:e}})}))}function Ze(e,t){return re(jsxRuntime.jsx(oe,{opts:{title:"Repositories",basePath:t},children:e.length===0?jsxRuntime.jsxs("div",{class:"text-center py-20 text-base-content/50",children:[jsxRuntime.jsx("p",{class:"text-lg font-medium mb-1",children:"No repositories configured"}),jsxRuntime.jsx("p",{class:"text-sm",children:"Add a repository to your FRS config to get started."})]}):jsxRuntime.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=>jsxRuntime.jsx("a",{href:`${t}/${n.name}`,class:"card bg-base-100 border border-base-300 hover:shadow-md no-underline transition-shadow",children:jsxRuntime.jsxs("div",{class:"card-body p-5",children:[jsxRuntime.jsx("h2",{class:"card-title text-sm font-semibold",children:n.name}),jsxRuntime.jsx("p",{class:"text-xs text-base-content/50 font-mono",children:n.path})]})},n.name))})}))}var Ot=[{value:"==",label:"="},{value:"!=",label:"\u2260"},{value:"in",label:"in"},{value:"not-in",label:"not in"}],Rn=[{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"}],Sn=[{value:"array-contains",label:"contains"},{value:"array-contains-any",label:"contains any"}];function wn(e){switch(e){case "ZodNumber":case "ZodBigInt":case "ZodDate":return Rn;case "ZodBoolean":return Ot;case "ZodArray":return Sn;default:return Ot}}var se="__null__";function $n(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="${se}"]');if(!o){o=document.createElement('option');o.value='${se}';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='${se}';i.readOnly=true;}i.style.opacity='0.55';}else{i.style.opacity='';if(i.tagName==='SELECT'){var o2=i.querySelector('option[value="${se}"][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!=='${se}')?i.dataset._prev:'';}}})(this)`}function Cn(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 $e({inputId:e,active:t}){return jsxRuntime.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:[jsxRuntime.jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",checked:t,onchange:$n(e)}),jsxRuntime.jsx("span",{children:"\u2205"})]})}function On({col:e,active:t}){let n=t?.value??"",r=n===se,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 h=new Set(n.split(",").map(m=>m.trim()).filter(Boolean)),d=`eg_${e.name.replace(/\./g,"__")}`;return jsxRuntime.jsxs("div",{class:"flex flex-wrap items-center gap-1 w-full",children:[jsxRuntime.jsx("input",{type:"hidden",id:o,name:`fv_${e.name}`,value:n}),e.enumValues.map(m=>jsxRuntime.jsxs("label",{class:"flex items-center gap-1 text-xs border border-base-300 rounded px-2 cursor-pointer hover:bg-base-200",children:[jsxRuntime.jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",value:m,checked:h.has(m),"data-enum-group":d,onchange:Cn(o,d)}),jsxRuntime.jsx("span",{children:m})]},m))]})}return jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.jsxs("select",{id:o,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsxRuntime.jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),e.enumValues.map(h=>jsxRuntime.jsx("option",{value:h,selected:n===h,children:h},h)),e.nullable&&jsxRuntime.jsx("option",{value:se,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsxRuntime.jsx($e,{inputId:o,active:r})]})}if(e.zodType==="ZodBoolean")return jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.jsxs("select",{id:o,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsxRuntime.jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),jsxRuntime.jsx("option",{value:"true",selected:n==="true",children:"true"}),jsxRuntime.jsx("option",{value:"false",selected:n==="false",children:"false"}),e.nullable&&jsxRuntime.jsx("option",{value:se,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsxRuntime.jsx($e,{inputId:o,active:r})]});if(e.zodType==="ZodArray"){let h=t?.op==="array-contains-any";return jsxRuntime.jsx("input",{id:o,type:"text",name:`fv_${e.name}`,value:n,placeholder:h?"val1, val2, \u2026":"value",class:"input input-sm input-bordered w-full"})}return e.zodType==="ZodNumber"||e.zodType==="ZodBigInt"?jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.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&&jsxRuntime.jsx($e,{inputId:o,active:r})]}):e.zodType==="ZodDate"?jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.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&&jsxRuntime.jsx($e,{inputId:o,active:r})]}):jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.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&&jsxRuntime.jsx($e,{inputId:o,active:r})]})}function ze({action:e,columnMeta:t,activeFilters:n,isGroup:r}){let o=Object.fromEntries(n.map(d=>[d.field,d])),s=n.length>0,a=n.length>=2||r&&s,h=t.filter(d=>d.zodType!=="ZodObject"&&d.zodType!=="ZodRecord");return jsxRuntime.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:[jsxRuntime.jsxs("summary",{class:"collapse-title text-sm font-medium py-2 min-h-0",children:["Filters",s&&jsxRuntime.jsxs("span",{class:"badge badge-primary badge-sm ml-2",children:[n.length," active"]})]}),jsxRuntime.jsx("div",{class:"collapse-content pb-4 pt-2",children:jsxRuntime.jsxs("form",{method:"get",action:e,children:[jsxRuntime.jsx("div",{class:"grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4",children:h.map(d=>{let m=wn(d.zodType),y=o[d.name],x=y?.op??m[0].value;return jsxRuntime.jsxs("div",{class:"flex flex-col gap-1.5",children:[jsxRuntime.jsx("label",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:d.name}),jsxRuntime.jsxs("div",{class:"flex gap-1.5",children:[m.length>1?jsxRuntime.jsx("select",{name:`fo_${d.name}`,class:"select select-sm select-bordered w-20 shrink-0",children:m.map(i=>jsxRuntime.jsx("option",{value:i.value,selected:i.value===x,children:i.label},i.value))}):jsxRuntime.jsx("input",{type:"hidden",name:`fo_${d.name}`,value:m[0].value}),jsxRuntime.jsx(On,{col:d,active:y})]})]},d.name)})}),jsxRuntime.jsxs("div",{class:"flex flex-wrap gap-2 mt-4 pt-3 border-t border-base-200 items-center",children:[jsxRuntime.jsx("button",{type:"submit",class:"btn btn-sm btn-primary",children:"Apply"}),s&&jsxRuntime.jsx("a",{href:e,class:"btn btn-sm btn-ghost",children:"Clear"}),a&&jsxRuntime.jsxs("span",{class:"text-xs text-warning ml-auto flex items-center gap-1",children:[jsxRuntime.jsx("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:jsxRuntime.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 qe(e,t,n,r,o,s){let a=n==="create"?`Create ${e}`:`Edit ${e} / ${r??""}`,h=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 re(jsxRuntime.jsx(oe,{opts:{title:a,breadcrumb:h,basePath:o,flash:s},children:jsxRuntime.jsx("div",{class:"card bg-base-100 border border-base-300",children:jsxRuntime.jsx("div",{class:"card-body p-6",children:jsxRuntime.jsx("div",{dangerouslySetInnerHTML:{__html:t}})})})}))}function Be(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 kt(e,t,n,r,o){let s=Be(e,r,o);return s.set("cursor",t),s.set("dir",n),`?${s.toString()}`}function kn(e,t,n,r){let o=Be(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 Tn(e,t,n){return `?${Be(t,n,e).toString()}`}function Me(e,t,n,r,o,s,a=[],h=[],d=false,m=[],y,x,i,l){let f=`${r}/${e}`,p=`${f}/create`;return re(jsxRuntime.jsxs(oe,{opts:{title:e,breadcrumb:[{label:"Repositories",href:r},{label:e}],basePath:r,flash:s},children:[a.length>0&&jsxRuntime.jsx(ze,{action:f,columnMeta:a,activeFilters:h,isGroup:l}),i&&jsxRuntime.jsxs("div",{role:"alert",class:`alert ${i.type==="index"?"alert-warning":"alert-error"} mb-6 shadow-sm`,children:[jsxRuntime.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:i.type==="index"?jsxRuntime.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"}):jsxRuntime.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"})}),jsxRuntime.jsxs("div",{class:"flex-1",children:[jsxRuntime.jsx("h3",{class:"font-bold",children:i.type==="index"?"Composite index required":"Query failed"}),jsxRuntime.jsx("div",{class:"text-sm",children:i.message})]}),i.indexUrl&&jsxRuntime.jsx("a",{href:i.indexUrl,target:"_blank",rel:"noopener noreferrer",class:"btn btn-sm btn-outline",children:"Create Index \u2192"})]}),jsxRuntime.jsxs("div",{class:"flex flex-wrap justify-between items-center mb-4 gap-3",children:[jsxRuntime.jsxs("div",{class:"flex items-center gap-3",children:[jsxRuntime.jsxs("span",{class:"text-sm text-base-content/60",children:[t.length," document",t.length!==1&&"s"]}),jsxRuntime.jsxs("div",{class:"flex items-center gap-1.5 text-sm text-base-content/60",children:[jsxRuntime.jsx("span",{children:"Rows"}),jsxRuntime.jsx("div",{class:"join",children:[10,25,50,100].map(c=>jsxRuntime.jsx("a",{href:Tn(c,h,y),class:`join-item btn btn-xs ${x===c?"btn-active btn-primary":"btn-outline"}`,children:c},c))})]})]}),jsxRuntime.jsx("a",{href:p,class:"btn btn-primary btn-sm",children:"+ New"})]}),jsxRuntime.jsx("div",{class:"overflow-x-auto rounded-box border border-base-300 bg-base-100","data-frs-table-wrap":true,children:jsxRuntime.jsxs("table",{class:"table table-sm w-full","data-frs-table":true,"data-frs-repo":e,"data-frs-colcount":n.length,children:[jsxRuntime.jsx("thead",{children:jsxRuntime.jsxs("tr",{class:"bg-base-200/50",children:[[...n].map((c,b)=>{let u=y?.field===c,g=u?y.dir==="asc"?" \u25B2":" \u25BC":"";return jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:jsxRuntime.jsxs("a",{href:kn(c,y,h,x),class:`hover:text-base-content inline-flex items-center gap-0.5${u?" text-primary font-bold":""}`,children:[c,g]})},b)}),m.map((c,b)=>jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:c.column},`rel-${b}`)),jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide text-right",children:"Actions"})]})}),jsxRuntime.jsx("tbody",{children:t.length===0?jsxRuntime.jsx("tr",{children:jsxRuntime.jsx("td",{colspan:n.length+m.length+1,class:"text-center py-16 text-base-content/40",children:"No documents found"})}):t.map((c,b)=>{let u=String(c.docId??c.id??""),g=`${r}/${e}/${encodeURIComponent(u)}/edit`,v=`${r}/${e}/${encodeURIComponent(u)}/delete`;return jsxRuntime.jsxs("tr",{class:"hover",children:[n.map((R,w)=>jsxRuntime.jsx("td",{class:"align-top py-2",children:jsxRuntime.jsx(je,{val:c[R]})},w)),m.map((R,w)=>{let C=c[R.key];if(C==null||C==="")return jsxRuntime.jsx("td",{class:"py-2"},`rel-${w}`);let E=`${r}/${R.targetRepo}?fv_${R.targetKey}=${encodeURIComponent(String(C))}`;return jsxRuntime.jsx("td",{class:"align-middle py-2",children:jsxRuntime.jsx("a",{href:E,class:"btn btn-xs btn-ghost btn-outline",children:R.column})},`rel-${w}`)}),jsxRuntime.jsx("td",{class:"align-middle text-right whitespace-nowrap py-2",children:jsxRuntime.jsxs("div",{class:"flex gap-1 justify-end",children:[jsxRuntime.jsx("a",{href:g,class:"btn btn-xs btn-outline",children:"Edit"}),d&&jsxRuntime.jsx("form",{method:"post",action:v,onsubmit:"return confirm('Delete this document?')",children:jsxRuntime.jsx("button",{type:"submit",class:"btn btn-xs btn-error btn-outline",children:"Delete"})})]})})]},b)})})]})}),(o.hasPrev||o.hasNext)&&jsxRuntime.jsxs("div",{class:"flex justify-center items-center mt-6 gap-2",children:[o.hasPrev?jsxRuntime.jsx("a",{href:kt(h,o.prevCursor,"prev",y,x),class:"btn btn-sm btn-outline",children:"\u2190 Previous"}):jsxRuntime.jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"\u2190 Previous"}),o.hasNext?jsxRuntime.jsx("a",{href:kt(h,o.nextCursor,"next",y,x),class:"btn btn-sm btn-outline",children:"Next \u2192"}):jsxRuntime.jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"Next \u2192"})]})]}))}var An="";function Pe(e,t){return Fe(e,t)}function Ue(e,t){return Ze(e,t)}function Le(e,t,n,r,o,s,a,h,d,m,y,x,i,l){return Me(e,t,n,r,o,s,a,h,d,m,y,x,i,l)}function ae(e,t,n,r,o,s){return qe(e,t,n,r,o,s)}var En=new Set(["<","<=",">",">=","!="]),_n=new Set(["array-contains","array-contains-any"]);function Ke(e){return e==="desc"?"DESCENDING":"ASCENDING"}function Pn(e){let t=e.split("/").filter(Boolean);return t[t.length-1]??e}function Nn(e,t,n,r,o){let s=[],a=new Set;for(let d of r)if(d.op==="=="||d.op==="in"||d.op==="not-in"){if(a.has(d.field))continue;a.add(d.field),s.push({fieldPath:d.field,order:"ASCENDING"});}for(let d of r)if(_n.has(d.op)){if(a.has(d.field))continue;a.add(d.field),s.push({fieldPath:d.field,arrayConfig:"CONTAINS"});}for(let d of r)if(En.has(d.op)){if(a.has(d.field))continue;a.add(d.field);let m=o?.field===d.field?Ke(o.dir):"ASCENDING";s.push({fieldPath:d.field,order:m});}if(o&&!a.has(o.field)&&s.push({fieldPath:o.field,order:Ke(o.dir)}),s.length===1&&n)return jn(e,t,s[0]);let h=o&&s.some(d=>d.fieldPath===o.field)?Ke(o.dir):"ASCENDING";return s.push({fieldPath:"__name__",order:h}),In(e,t,n,s)}function In(e,t,n,r,o="(default)"){let s=`projects/${e}/databases/${o}/collectionGroups/${t}/indexes/_`,a=[...Ve(1,s),...Ne(2,n?2:1)];for(let m of r)a.push(...Tt(3,At(m)));let h=o==="(default)"?"-default-":o,d=encodeURIComponent(Et(a));return `https://console.firebase.google.com/project/${e}/firestore/databases/${h}/indexes?create_composite=${d}`}function Dn(e){return e.match(/https:\/\/console\.firebase\.google\.com[^\s)"]*/)?.[0]}function He(e){let t=[],n=e>>>0;for(;n>=128;)t.push(n&127|128),n>>>=7;return t.push(n&127),t}function Ge(e,t){return e<<3|t}function Ve(e,t){let n=Array.from(new TextEncoder().encode(t));return [Ge(e,2),...He(n.length),...n]}function Ne(e,t){return [Ge(e,0),...He(t)]}function Tt(e,t){return [Ge(e,2),...He(t.length),...t]}function At(e){let t=[...Ve(1,e.fieldPath)];return e.arrayConfig==="CONTAINS"?t.push(...Ne(3,1)):t.push(...Ne(2,e.order==="DESCENDING"?2:1)),t}function Et(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 jn(e,t,n,r="(default)"){let o=`projects/${e}/databases/${r}/collectionGroups/${t}/fields/${n.fieldPath}`,s=[...Ve(1,o),...Ne(2,2),...Tt(3,At(n))],a=r==="(default)"?"-default-":r,h=encodeURIComponent(Et(s));return `https://console.firebase.google.com/project/${e}/firestore/databases/${a}/indexes/automatic?create_exemption=${h}`}function Fn(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 ue(e){let t=e;return t?t.code===9?true:typeof t.message=="string"?t.message.includes("requires an index"):false:false}function Ce(e,t){let n=e??{},r=ue(e),o;if(r&&(o=n.message?Dn(n.message):void 0,!o)){let s=Fn(t.ref);if(s){let a=Pn(t.path);o=Nn(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 _t="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function zn(){let e="";for(let t=0;t<20;t++)e+=_t.charAt(Math.floor(Math.random()*_t.length));return e}function Pt(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 Je(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 We(e,t,n){let r=e.documentKey??"docId",o=Ce(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 I(e,t,n=200){e.status(n).set("Content-Type","text/html; charset=utf-8").send(t);}function Qe(e,t){e.status(302).set("Location",t).send("");}function Xe(e,t){let n=t.shape,r={};for(let[o,s]of Object.entries(n)){let a=et(s);if(a==="ZodObject"){if(e[o+"__isnull"]==="1"){r[o]=null;continue}let m={},y=false;for(let[l,f]of Object.entries(e))l.startsWith(`${o}.`)&&(m[l.slice(o.length+1)]=f,y=true);if(y){let l=s;for(;;){let f=j(l);if(f==="ZodOptional"||f==="ZodNullable"||f==="ZodDefault")l=M(l);else break}r[o]=Xe(m,l);continue}let x=e[o],i=Array.isArray(x)?x[x.length-1]:x;if(i)try{r[o]=JSON.parse(i);}catch{r[o]=i;}continue}let h=e[o],d=Array.isArray(h)?h[h.length-1]:h;if(e[o+"__isnull"]==="1"){r[o]=null;continue}if(d===void 0||d===""){a==="ZodBoolean"&&(r[o]=false);continue}switch(a){case "ZodBoolean":d==="__null__"?r[o]=null:r[o]=d==="true"||d==="on"||d==="1";break;case "ZodNumber":case "ZodBigInt":r[o]=Number(d);break;case "ZodDate":r[o]=new Date(d);break;case "ZodArray":try{r[o]=JSON.parse(d);}catch{r[o]=d;}break;default:if(d.startsWith("{")||d.startsWith("["))try{r[o]=JSON.parse(d);break}catch{}r[o]=d;}}return r}function Nt(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 et(e){let t=e;for(;;){let n=j(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=M(t);else return n}}function qn(e){let t=e;for(;;){let n=j(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=M(t);else return t}}function It(e){let t=e;for(;;){let n=j(t);if(n==="ZodOptional"||n==="ZodNullable")return true;if(n==="ZodDefault"){t=M(t);continue}return false}}function Dt(e){let t=qn(e),n=j(t);if(n==="ZodEnum"){let r=be(t);return r.length>0?r:void 0}if(n==="ZodNativeEnum"){let r=ve(t),o=Object.values(r).filter(s=>typeof s=="string");return o.length>0?o:void 0}if(n==="ZodLiteral"){let r=Te(t);return typeof r=="string"?[r]:void 0}}function jt(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 h=t.shape[o];for(;;){let m=j(h);if(m==="ZodOptional"||m==="ZodNullable"||m==="ZodDefault")h=M(h);else break}let d=j(h);if(d==="ZodObject"&&typeof a=="object"&&a!==null&&!Array.isArray(a)){let m=jt(a,h,s);Object.assign(r,m);}else if(d==="ZodDate"){let m=Nt(a);m!==null&&(r[s]=m);}else if(typeof a=="object"&&a!==null&&!Array.isArray(a)&&("_seconds"in a||typeof a.toDate=="function")){let m=Nt(a);r[s]=m??JSON.stringify(a,null,2);}else typeof a=="object"?r[s]=JSON.stringify(a,null,2):r[s]=String(a);}return r}function Ye(e,t){return e.map(n=>({...n,defaultValue:t[n.name]??n.defaultValue,nested:n.nested?Ye(n.nested,t):void 0}))}function Bn(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 h=(s??"").trim();if(!h)continue;let d=e[`fo_${a}`]??"==",m=n.has(d)?d:"==";r.push({field:a,op:m,value:h});}return r}function Mn(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 Ft(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 h=et(a);if(h==="ZodObject"){let d=a;for(;;){let y=j(d);if(y==="ZodOptional"||y==="ZodNullable"||y==="ZodDefault")d=M(d);else break}let m=ee(d);r.push(...Ft(Object.keys(m),d,s));}else r.push({name:s,zodType:h,nullable:It(a),enumValues:Dt(a)});}return r}function Un(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=M(r);else break}let s=ee(r);if(!(o in s))return null;r=s[o];}return r}function fe(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 h=s.slice(0,a),d=s.slice(a+1);r.has(h)||r.set(h,[]),r.get(h).push(d);}}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 h=e.shape[s];for(;;){let d=j(h);if(d==="ZodOptional"||d==="ZodNullable"||d==="ZodDefault")h=M(h);else break}if(j(h)!=="ZodObject"){o[s]=e.shape[s];continue}o[s]=fe(h,a);}return zod.z.object(o)}function pe(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",h=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${s}/${a}/${h}${n}`}let r=process.env.K_SERVICE,o=e.hostname??e.headers?.host??"";return r&&o.includes("cloudfunctions.net")?`/${r.toLowerCase()}${n}`:n}function Zt(e,t){return {handleDashboard:(m,y)=>{let x=pe(m,t),i=Object.values(e).map(l=>({name:l.name,path:l.path}));I(y,Ue(i,x));},handleList:async(m,y)=>{let x=m.params.repoName;if(!x){I(y,"Bad request",400);return}let i=e[x];if(!i){I(y,"Repository not found",404);return}let l=i.pageSize??25,f=m.query??{},p=f.cursor,c=f.dir==="prev"?"prev":"next",b=f.ob??"",u=f.od==="desc"?"desc":"asc",g=b?{field:b,dir:u}:void 0,v=parseInt(f.ps??""),R=Number.isFinite(v)&&v>0?Math.min(v,200):l,w=i.listColumns??Object.keys(i.schema.shape),C=i.documentKey??"docId",E=[C,...w.filter(q=>q!==C)],S=i.filterableFields?(()=>{let q=[];for(let X of i.filterableFields)(X.includes(".")||w.includes(X))&&q.push(X);return q})():w,$=(()=>{let q=[];for(let X of S)if(X.includes(".")){let me=Un(i.schema,X);q.push({name:X,zodType:me?et(me):"ZodString",nullable:me?It(me):false,enumValues:me?Dt(me):void 0});}else q.push(...Ft([X],i.schema));return q})(),k=new Set($.map(q=>q.name)),T=Bn(f,k),N=Mn(T),O;if(p)try{let q=i.repo.ref;typeof q.doc=="function"&&(O=await q.doc(p).get());}catch{}let B=await i.repo.query.paginate({pageSize:R,cursor:O,direction:c,...N.length>0?{where:N}:{},...g?{orderBy:[{field:g.field,direction:g.dir}]}:{}}).catch(q=>({queryError:Ce(q,{ref:i.repo.ref,path:i.path,isGroup:!!i.isGroup,filters:T,sort:g})})),D="queryError"in B,Y=D?[]:B.data,en=D?"":B.nextCursor?.id??"",tn=D?"":B.prevCursor?.id??"",nn=D?B.queryError:void 0,rn=pe(m,t);I(y,Le(i.name,Y,E,rn,{hasPrev:D?false:B.hasPrevPage,hasNext:D?false:B.hasNextPage,prevCursor:tn,nextCursor:en},void 0,$,T,i.allowDelete??false,i.relationalMeta,g,R,nn,i.isGroup));},handleCreateForm:(m,y)=>{let x=m.params.repoName;if(!x){I(y,"Bad request",400);return}let i=e[x];if(!i){I(y,"Repository not found",404);return}let l=pe(m,t),f=fe(i.schema,i.createFields),p=W(f),c=`${l}/${i.name}/create`,b=ne(p,c,"POST","Create document");I(y,ae(i.name,b,"create",null,l));},handleCreateSubmit:async(m,y)=>{let x=m.params.repoName;if(!x){I(y,"Bad request",400);return}let i=e[x];if(!i){I(y,"Repository not found",404);return}let l=pe(m,t),f=m.body??{},p=Xe(f,i.schema),c=fe(i.schema,i.createFields),b=c.safeParse(p);if(!b.success){let u=W(c),g=`${l}/${i.name}/create`,v=ne(u,g,"POST","Create document"),R=b.error.issues.map(w=>`${w.path.join(".")}: ${w.message}`).join(", ");I(y,ae(i.name,v,"create",null,l,{type:"error",message:`Validation error: ${R}`}),422);return}try{if(i.isGroup&&i.parentKeys&&i.parentKeys.length>0){let u={...b.data};i.createdKey&&(u[i.createdKey]=new Date);let g=i.parentKeys.filter(C=>!u[C]);if(g.length>0)throw new Error(`Missing parent key(s) for subcollection create: ${g.join(", ")}`);let v=i.parentKeys.map(C=>u[C]),R=i.documentKey??"docId",w=u[R]||zn();await i.repo.set(...v,w,u);}else await i.repo.create(b.data);Qe(y,`${l}/${i.name}?flash=created`);}catch(u){let g=fe(i.schema,i.createFields),v=W(g),R=`${l}/${i.name}/create`,w=ne(v,R,"POST","Create document");I(y,ae(i.name,w,"create",null,l,{type:"error",message:`Save error: ${u.message}`}),500);}},handleEditForm:async(m,y)=>{let x=m.params.repoName,i=m.params.id;if(!x||!i){I(y,"Bad request",400);return}let l=e[x];if(!l){I(y,"Repository not found",404);return}let f=pe(m,t),p=null;try{p=await Je(l,i);}catch(R){let w=We(l,i,R),C=ue(R)?424:500;I(y,Pe("",{title:`Edit ${l.name} / ${i}`,basePath:f,breadcrumb:[{label:"Repositories",href:f},{label:l.name,href:`${f}/${l.name}`},{label:`Edit ${i}`}],flash:w}),C);return}if(!p){I(y,"Document not found",404);return}let c=jt(p,l.schema),b=fe(l.schema,l.mutableFields),u=Ye(W(b),c),g=`${f}/${l.name}/${encodeURIComponent(i)}/edit`,v=ne(u,g,"POST","Save changes");I(y,ae(l.name,v,"edit",i,f));},handleEditSubmit:async(m,y)=>{let x=m.params.repoName,i=m.params.id;if(!x||!i){I(y,"Bad request",400);return}let l=e[x];if(!l){I(y,"Repository not found",404);return}let f=pe(m,t),p=m.body??{},c=Xe(p,l.schema),b=fe(l.schema,l.mutableFields),g=b.partial().safeParse(c);if(!g.success){let v=Object.fromEntries(Object.entries(p).map(([S,$])=>[S,Array.isArray($)?$.join(","):$??""])),R=Ye(W(b),v),w=`${f}/${l.name}/${encodeURIComponent(i)}/edit`,C=ne(R,w,"POST","Save changes"),E=g.error.issues.map(S=>`${S.path.join(".")}: ${S.message}`).join(", ");I(y,ae(l.name,C,"edit",i,f,{type:"error",message:`Validation error: ${E}`}),422);return}try{let v=await Je(l,i),R=(v&&Pt(v,l.pathKey))??[i];await l.repo.update(...R,g.data),Qe(y,`${f}/${l.name}?flash=updated`);}catch(v){let R=fe(l.schema,l.mutableFields),w=W(R),C=`${f}/${l.name}/${encodeURIComponent(i)}/edit`,E=ne(w,C,"POST","Save changes"),S=ue(v)?We(l,i,v):{type:"error",message:`Save error: ${v.message}`},$=ue(v)?424:500;I(y,ae(l.name,E,"edit",i,f,S),$);}},handleDelete:async(m,y)=>{let x=m.params.repoName,i=m.params.id;if(!x||!i){I(y,"Bad request",400);return}let l=e[x];if(!l){I(y,"Repository not found",404);return}if(!l.allowDelete){I(y,"Delete is not allowed for this repository",403);return}let f=pe(m,t);try{let p=await Je(l,i),c=(p&&Pt(p,l.pathKey))??[i];await l.repo.delete(...c),Qe(y,`${f}/${l.name}?flash=deleted`);}catch(p){let c=ue(p)?We(l,i,p):{type:"error",message:`Delete error: ${p.message}`},b=ue(p)?424:500;I(y,Pe("",{title:`Delete ${l.name} / ${i}`,basePath:f,breadcrumb:[{label:"Repositories",href:f},{label:l.name,href:`${f}/${l.name}`},{label:`Delete ${i}`}],flash:c}),b);}}}}async function Ln(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function Kn(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 zt(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:o,middleware:s=[],httpsOptions:a}=e,h=t==="/"?"":t.replace(/\/$/,""),d={};for(let[i,l]of Object.entries(n)){let f=l.schema??l.repo.schema??null;if(!f)throw new Error(`[createAdminServer] Repository "${i}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let p,c,b;if(l.fieldsConfig){let v=l.fieldsConfig;p=[],c=[],b=[];for(let[R,w]of Object.entries(v))for(let C of w)C==="filterable"?p.push(R):C==="mutable"?c.push(R):C==="create"&&b.push(R);p.length===0&&(p=void 0),c.length===0&&(c=void 0),b.length===0&&(b=void 0);}let u=(()=>{let v=l.repo._parentKeys;return v&&v.length>0?v:void 0})();if(u&&b)for(let v of u)b.includes(v)||b.push(v);let g={name:i,path:l.path,repo:l.repo,schema:f,documentKey:l.documentKey??"docId",pathKey:l.repo._pathKey??void 0,isGroup:!!l.repo._isGroup,parentKeys:u,createdKey:l.repo._createdKey??void 0,listColumns:l.listColumns,pageSize:l.pageSize,filterableFields:p,mutableFields:c,createFields:b,allowDelete:l.allowDelete??false,relationalMeta:(()=>{if(!l.relationalFields||l.relationalFields.length===0)return;let v=l.repo.relationalKeys??{},R=[];for(let w of l.relationalFields){let C=v[w.key];C&&R.push({key:w.key,column:w.column,targetRepo:String(C.repo),targetKey:String(C.key),type:C.type});}return R.length>0?R:void 0})()};d[i]=g;}let m=Zt(d,h),y=new G;if(r&&y.use(async(i,l,f)=>{let p=i,c=String(p.headers?.["content-type"]??"");if(c.includes("application/x-www-form-urlencoded")){let b=await Ln(p);i.body=Kn(b);}else if(c.includes("application/json")&&typeof p.body=="string")try{i.body=JSON.parse(p.body);}catch{}await f();}),o)if(typeof o=="function")y.use(o);else {let i=o.realm??"Admin",l="Basic "+Buffer.from(`${o.username}:${o.password}`).toString("base64");y.use((f,p,c)=>{if((f.headers?.authorization??"")!==l){p.status(401).set("WWW-Authenticate",`Basic realm="${i}"`).set("Content-Type","text/plain").send("Unauthorized");return}c();});}for(let i of s)y.use(i);y.get(`${h}/`,m.handleDashboard),y.get(`${h}`,m.handleDashboard),y.get(`${h}/:repoName`,m.handleList),y.get(`${h}/:repoName/create`,m.handleCreateForm),y.post(`${h}/:repoName/create`,m.handleCreateSubmit),y.get(`${h}/:repoName/:id/edit`,m.handleEditForm),y.post(`${h}/:repoName/:id/edit`,m.handleEditSubmit),y.post(`${h}/:repoName/:id/delete`,m.handleDelete);let x=async(i,l)=>{await y.handle(i,l);};return a&&(x.httpsOptions=a),x}var Bt="preserve";function Mt(){return Bt}function Hn(e){return typeof e=="object"&&e!==null&&typeof e._seconds=="number"&&typeof e._nanoseconds=="number"}function Ut(e){if(e==null)return null;if(e instanceof Date)return Number.isNaN(e.getTime())?null:e;if(e instanceof firestore.Timestamp)return e.toDate();if(Hn(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 Lt(e){return e}function rt(e,t,n=200){let r=Lt(t);e.status(n).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(r));}function ge(e,t,n,r=200){rt(e,{success:true,data:t,meta:n},r);}function z(e,t,n=400){rt(e,{success:false,error:t},n);}function nt(e,t,n,r,o){let s=Ce(t,n),a=s.type==="index",h=a?424:500,m={success:false,error:a?s.message:o&&t instanceof Error?t.message:r};a&&(m.errorType="index",s.indexUrl&&(m.indexUrl=s.indexUrl)),rt(e,m,h);}var Kt="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function Vn(){let e="";for(let t=0;t<20;t++)e+=Kt.charAt(Math.floor(Math.random()*Kt.length));return e}function he(e){let t=e._def??e.def;if(!t)return e;let n=t.typeName??t.type;if(n==="ZodDate"||n==="date")return zod.z.preprocess(r=>Ut(r)??r,e);if(n==="ZodObject"||n==="object"){let r=e.shape,o={};for(let[s,a]of Object.entries(r))o[s]=he(a);return zod.z.object(o)}if(n==="ZodArray"||n==="array"){let r=t.element??t.type;if(r)return zod.z.array(he(r))}if(n==="ZodOptional"||n==="optional"){let r=t.innerType;if(r)return he(r).optional()}if(n==="ZodNullable"||n==="nullable"){let r=t.innerType;if(r)return he(r).nullable()}if(n==="ZodDefault"||n==="default"){let r=t.innerType,o=t.defaultValue;if(r){let s=he(r);return typeof o=="function"?s.default(o()):s.default(o)}}return e}function Jn(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 h=a.split(".")[0];h&&r[h]&&(o[h]=r[h]);}return zod.z.object(o)}function Ht(e,t,n,r=false,o=[]){try{let s=Jn(e,n,o),a=r?s.partial():s;return {success:!0,data:(Mt()==="normalize"?he(a):a).parse(t)}}catch(s){return s instanceof zod.z.ZodError?{success:false,error:`Validation failed: ${s.issues.map(h=>`${h.path.join(".")}: ${h.message}`).join(", ")}`}:{success:false,error:"Validation failed"}}}function Wn(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 h=Array.isArray(a)?a[0]:a;if(h===void 0||h==="")continue;let d=s.match(/^(\w+)__(\w+)$/),m,y="==";if(d&&d[1]&&d[2]){m=d[1];let i=d[2];if(o[i])y=o[i];else continue}else if(!d)m=s;else continue;if(r&&!r.has(m))continue;let x=h;y==="in"||y==="not-in"||y==="array-contains-any"?x=h.split(",").map(i=>Gt(i.trim())):x=Gt(h),n.push({field:m,op:y,value:x});}return n}function Gt(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 Ie(e){return e?{docId:e.id}:null}async function Vt(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 Jt(e,t,n){function r(l,f){return !l||!e[l]?(z(f,`Repository "${l}" not found`,404),null):e[l]}function o(l,f){if(!f)return;let p=l[f];if(typeof p!="string"||!p)return;let c=p.split("/").filter(Boolean),b=[];for(let u=1;u<c.length;u+=2)b.push(c[u]);return b.length>0?b:void 0}async function s(l,f){let p=`by${l.documentKey.charAt(0).toUpperCase()}${l.documentKey.slice(1)}`,c=l.repo.get[p];if(typeof c=="function")try{let u=await c(f);if(u)return u}catch{}return (await l.repo.query.by({where:[[l.documentKey,"==",f]],limit:1}))[0]??null}async function a(l,f){let p=l.params||{},c=r(p.repoName,f);if(!c)return;let b=[],u;try{let g=l.query??{},v=Math.min(Number(g.pageSize)||c.pageSize,100),R=g.cursor,w=g.direction?.toLowerCase()==="prev"?"prev":"next",C=g.orderBy,E=g.orderDir?.toLowerCase()==="desc"?"desc":"asc",S=g.select,$=S?S.split(",").map(D=>D.trim()):void 0,k;c.allowedIncludes&&g.includes&&(k=(typeof g.includes=="string"?g.includes.split(",").map(Y=>Y.trim()):Array.isArray(g.includes)?g.includes:[]).filter(Y=>typeof Y=="string"&&c.allowedIncludes.includes(Y)),k?.length===0&&(k=void 0));let T=Wn(g,c.filterableFields);b=T.map(D=>({field:D.field,op:D.op,value:String(D.value??"")})),C&&(u={field:C,dir:E});let N={pageSize:v,direction:w};if(R)try{let D=typeof R=="string"?JSON.parse(R):R;N.cursor=await Vt(c,D);}catch{}C&&(N.orderBy=[{field:C,direction:E}]),T.length>0&&(N.where=T.map(D=>[D.field,D.op,D.value])),$&&(N.select=$),k&&(N.include=k);let O=await c.repo.query.paginate(N),B={items:O.data,hasNextPage:O.hasNextPage,hasPrevPage:O.hasPrevPage,nextCursor:Ie(O.nextCursor),prevCursor:Ie(O.prevCursor)};ge(f,B,{pageSize:v,hasMore:O.hasNextPage});}catch(g){nt(f,g,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:b,sort:u},"Failed to fetch documents",n);}}async function h(l,f){let p=l.params||{},c=r(p.repoName,f);if(!c)return;let b=[],u;try{let g=l.body??{},v=Math.min(g.pageSize||c.pageSize,100),R=g.direction==="prev"?"prev":"next";g.where&&(b=g.where.map(S=>({field:String(S[0]),op:S[1],value:String(S[2]??"")}))),g.orderBy&&g.orderBy[0]&&(u={field:g.orderBy[0].field,dir:g.orderBy[0].direction==="desc"?"desc":"asc"});let w={pageSize:v,direction:R};if(g.cursor)try{let S=typeof g.cursor=="string"?JSON.parse(g.cursor):g.cursor;w.cursor=await Vt(c,S);}catch{}if(c.allowedIncludes&&g.includes&&g.includes.length>0){let S=g.includes.filter($=>typeof $=="string"?c.allowedIncludes.includes($):typeof $=="object"&&$!==null&&"relation"in $&&typeof $.relation=="string"?c.allowedIncludes.includes($.relation):!1);S.length>0&&(w.include=S);}if(g.where&&g.where.length>0){if(c.filterableFields){let S=new Set(c.filterableFields),$=g.where.filter(k=>!S.has(k[0]));if($.length>0){z(f,`Fields not filterable: ${$.map(k=>k[0]).join(", ")}`,400);return}}w.where=g.where;}if(g.orWhere&&g.orWhere.length>0){if(c.filterableFields){let S=new Set(c.filterableFields),$=g.orWhere.filter(k=>!S.has(k[0]));if($.length>0){z(f,`Fields not filterable: ${$.map(k=>k[0]).join(", ")}`,400);return}}w.orWhere=g.orWhere;}if(g.orWhereGroups&&g.orWhereGroups.length>0){if(c.filterableFields){let S=new Set(c.filterableFields);for(let $ of g.orWhereGroups){let k=$.filter(T=>!S.has(T[0]));if(k.length>0){z(f,`Fields not filterable: ${k.map(T=>T[0]).join(", ")}`,400);return}}}w.orWhereGroups=g.orWhereGroups;}g.orderBy&&g.orderBy.length>0&&(w.orderBy=g.orderBy),g.select&&g.select.length>0&&(w.select=g.select);let C=await c.repo.query.paginate(w),E={items:C.data,hasNextPage:C.hasNextPage,hasPrevPage:C.hasPrevPage,nextCursor:Ie(C.nextCursor),prevCursor:Ie(C.prevCursor)};ge(f,E,{pageSize:v,hasMore:C.hasNextPage});}catch(g){nt(f,g,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:b,sort:u},"Failed to query documents",n);}}async function d(l,f){let p=l.params||{},c=r(p.repoName,f);if(!c)return;let b=p.id;if(!b){z(f,"Document ID required",400);return}try{let u=await s(c,b);if(!u){z(f,"Document not found",404);return}ge(f,u);}catch(u){nt(f,u,{ref:c.repo.ref,path:c.path,isGroup:!!c.isGroup,filters:[{field:c.documentKey,op:"==",value:b}]},"Failed to fetch document",n);}}async function m(l,f){let p=l.params||{},c=r(p.repoName,f);if(c)try{let b=l.body??{},u=Ht(c.schema,b,c.createFields,!1,c.systemKeys);if(!u.success){z(f,u.error,400);return}if(c.validate){let v=await c.validate(u.data,"create");if(v){z(f,v,400);return}}let g;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(E=>!v[E]);if(R.length>0){z(f,`Missing parent key(s) for subcollection create: ${R.join(", ")}`,400);return}let w=c.parentKeys.map(E=>v[E]),C=v[c.documentKey]||Vn();g=await c.repo.set(...w,C,v);}else g=await c.repo.create(u.data);ge(f,g,void 0,201);}catch(b){let u=n&&b instanceof Error?b.message:"Failed to create document";z(f,u,500);}}async function y(l,f,p){let c=l.params||{},b=r(c.repoName,f);if(!b)return;let u=c.id;if(!u){z(f,"Document ID required",400);return}try{let g=l.body??{},v=Ht(b.schema,g,b.mutableFields,p,b.systemKeys);if(!v.success){z(f,v.error,400);return}if(b.validate){let E=await b.validate(v.data,"update");if(E){z(f,E,400);return}}let R=await s(b,u),w=(R&&o(R,b.pathKey))??[u],C=await b.repo.update(...w,v.data);ge(f,C);}catch(g){let v=n&&g instanceof Error?g.message:"Failed to update document";z(f,v,500);}}async function x(l,f){let p=l.params||{},c=r(p.repoName,f);if(!c)return;if(!c.allowDelete){z(f,"Delete not allowed for this repository",403);return}let b=p.id;if(!b){z(f,"Document ID required",400);return}try{let u=await s(c,b),g=(u&&o(u,c.pathKey))??[b];await c.repo.delete(...g),ge(f,{deleted:!0});}catch(u){let g=n&&u instanceof Error?u.message:"Failed to delete document";z(f,g,500);}}function i(l,f){f.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:h,handleGet:d,handleCreate:m,handleUpdate:y,handleDelete:x,handleOptions:i}}function ot(e){try{return zod.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 U(e){return {description:e,content:{"application/json":{schema:H("ErrorResponse")}}}}function ke(e,t){return {description:e,content:{"application/json":{schema:{type:"object",properties:{success:{type:"boolean",enum:[true]},data:t},required:["success","data"]}}}}}function Wt(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 Qn(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 Xn(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 Yn(){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 er(e,t,n,r,o){let s={},a=e.name,h=`${t}/${e.name}`,d=`${h}/{${e.documentKey}}`,m={name:e.documentKey,in:"path",required:true,schema:{type:"string"},description:"Unique document identifier"};s[h]={get:{operationId:`list${ie(e.name)}`,summary:`List ${e.name} (paginated)`,tags:[a],parameters:[...Qn(e),...Xn(e)],responses:{200:Wt(H(n)),500:U("Internal server error")}},post:{operationId:`create${ie(e.name)}`,summary:`Create a ${Q(e.name)}`,tags:[a],requestBody:{required:true,content:{"application/json":{schema:H(r??n)}}},responses:{201:ke("Document created",H(n)),400:U("Validation error"),500:U("Internal server error")}}},s[`${h}/query`]={post:{operationId:`query${ie(e.name)}`,summary:`Query ${e.name} with advanced filters`,tags:[a],requestBody:{required:true,content:{"application/json":{schema:H("QueryRequestBody")}}},responses:{200:Wt(H(n)),400:U("Invalid query"),500:U("Internal server error")}}};let y={};return y.get={operationId:`get${ie(Q(e.name))}`,summary:`Get a single ${Q(e.name)}`,tags:[a],parameters:[m],responses:{200:ke("Document found",H(n)),404:U("Document not found"),500:U("Internal server error")}},y.put={operationId:`update${ie(Q(e.name))}`,summary:`Update a ${Q(e.name)} (full replace)`,tags:[a],parameters:[m],requestBody:{required:true,content:{"application/json":{schema:H(o??n)}}},responses:{200:ke("Document updated",H(n)),400:U("Validation error"),404:U("Document not found"),500:U("Internal server error")}},y.patch={operationId:`patch${ie(Q(e.name))}`,summary:`Partially update a ${Q(e.name)}`,tags:[a],parameters:[m],requestBody:{required:true,content:{"application/json":{schema:{allOf:[H(o??n)],description:"All fields are optional for partial updates"}}}},responses:{200:ke("Document patched",H(n)),400:U("Validation error"),404:U("Document not found"),500:U("Internal server error")}},e.allowDelete&&(y.delete={operationId:`delete${ie(Q(e.name))}`,summary:`Delete a ${Q(e.name)}`,tags:[a],parameters:[m],responses:{200:ke("Document deleted",{type:"object",properties:{id:{type:"string"}}}),404:U("Document not found"),500:U("Internal server error")}}),s[d]=y,s}function at(e,t,n={}){let{title:r="CRUD API",version:o="1.0.0",description:s,servers:a,auth:h=false}=n,d=t==="/"?"":t.replace(/\/$/,""),m={},y={},x=[];m.ErrorResponse={type:"object",properties:{success:{type:"boolean",enum:[false]},error:{type:"string"}},required:["success","error"]},m.QueryRequestBody=Yn();for(let[p,c]of Object.entries(e)){let b=ie(Q(p)),u=`${b}Create`,g=`${b}Update`;m[b]=ot(c.schema);let v=$=>{let k=$&&$.length>0?$:Object.keys(c.schema.shape),T={};for(let N of k){let O=N.split(".")[0];O&&c.schema.shape[O]&&!c.systemKeys.includes(O)&&(T[O]=c.schema.shape[O]);}return T},R=null,w=v(c.createFields);Object.keys(w).length>0&&(m[u]=ot(zod.z.object(w)),R=u);let C=null,E=v(c.mutableFields);Object.keys(E).length>0&&(m[g]=ot(zod.z.object(E)),C=g);let S=er(c,d,b,R,C);Object.assign(y,S),x.push({name:p,description:`Operations on ${p} (collection: ${c.path})`});}let i={},l;return h==="basic"?(i.basicAuth={type:"http",scheme:"basic"},l=[{basicAuth:[]}]):h==="bearer"&&(i.bearerAuth={type:"http",scheme:"bearer",bearerFormat:"JWT"},l=[{bearerAuth:[]}]),{openapi:"3.1.0",info:{title:r,version:o,...s?{description:s}:{}},...a&&a.length>0?{servers:a}:{},paths:y,components:{schemas:m,...Object.keys(i).length>0?{securitySchemes:i}:{}},...l?{security:l}:{},tags:x}}function ie(e){return e.charAt(0).toUpperCase()+e.slice(1)}function Q(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 tr(e,t){return `<!DOCTYPE html>
|
|
1135
|
+
`;function _e(){return jsxRuntime.jsx("script",{dangerouslySetInnerHTML:{__html:Lt}})}tt();function le(e){return "<!DOCTYPE html>"+server.renderToString(e)}var Yn=["corporate","silk","dark"],er=()=>jsxRuntime.jsxs("div",{class:"dropdown dropdown-end","data-frs-theme-switcher":true,children:[jsxRuntime.jsxs("button",{type:"button",tabIndex:0,class:"btn btn-sm btn-ghost text-neutral-content gap-1.5","aria-label":"Switch theme",children:[jsxRuntime.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:[jsxRuntime.jsx("circle",{cx:"12",cy:"12",r:"4"}),jsxRuntime.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"})]}),jsxRuntime.jsx("span",{class:"text-xs hidden sm:inline","data-frs-theme-current":true,children:"Theme"})]}),jsxRuntime.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:Yn.map(e=>jsxRuntime.jsx("li",{children:jsxRuntime.jsxs("button",{type:"button","data-frs-theme":e,class:"capitalize justify-between",children:[jsxRuntime.jsx("span",{children:e}),jsxRuntime.jsx("span",{class:"hidden text-primary","data-frs-theme-check":e,children:"\u2713"})]})},e))})]}),ce=({opts:e,children:t})=>{let{title:n,breadcrumb:r,flash:s,basePath:o="/"}=e;return jsxRuntime.jsxs("html",{lang:"en","data-theme":"corporate",children:[jsxRuntime.jsxs("head",{children:[jsxRuntime.jsx("meta",{charset:"UTF-8"}),jsxRuntime.jsx("meta",{name:"viewport",content:"width=device-width, initial-scale=1"}),jsxRuntime.jsxs("title",{children:[n," \u2014 FRS Admin"]}),jsxRuntime.jsx("script",{dangerouslySetInnerHTML:{__html:"(function(){try{var t=localStorage.getItem('frs-admin-theme');if(t){document.documentElement.setAttribute('data-theme',t);}}catch(_){}})();"}}),jsxRuntime.jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/themes.css",rel:"stylesheet",type:"text/css"}),jsxRuntime.jsx("link",{href:"https://cdn.jsdelivr.net/npm/daisyui@5/daisyui.css",rel:"stylesheet",type:"text/css"}),jsxRuntime.jsx("script",{src:"https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4"})]}),jsxRuntime.jsxs("body",{class:"bg-base-200/50 min-h-screen flex flex-col",children:[jsxRuntime.jsxs("div",{class:"navbar bg-neutral text-neutral-content shadow-sm sticky top-0 z-50 px-6",children:[jsxRuntime.jsx("div",{class:"flex-1",children:jsxRuntime.jsx("a",{href:o,class:"font-bold text-lg tracking-tight hover:opacity-80 transition-opacity",children:"FRS Admin"})}),jsxRuntime.jsx("div",{class:"flex-none",children:jsxRuntime.jsx(er,{})})]}),jsxRuntime.jsxs("main",{class:"px-6 py-8 w-full flex-1",children:[r&&r.length>0&&jsxRuntime.jsx("div",{class:"text-sm breadcrumbs mb-4",children:jsxRuntime.jsx("ul",{children:r.map((l,p)=>l.href?jsxRuntime.jsx("li",{children:jsxRuntime.jsx("a",{href:l.href,children:l.label})},p):jsxRuntime.jsx("li",{class:"text-base-content/60",children:l.label},p))})}),jsxRuntime.jsx("h1",{class:"text-2xl font-bold mb-6",children:n}),s&&jsxRuntime.jsxs("div",{role:"alert",class:`alert ${s.type==="success"?"alert-success":s.type==="warning"?"alert-warning":"alert-error"} mb-6`,children:[jsxRuntime.jsx("span",{class:"flex-1",children:s.message}),s.action&&jsxRuntime.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]}),jsxRuntime.jsx(et,{basePath:o}),jsxRuntime.jsx(_e,{})]})]})};function nt(e,t){return le(jsxRuntime.jsx(ce,{opts:t,children:jsxRuntime.jsx("div",{dangerouslySetInnerHTML:{__html:e}})}))}function rt(e,t){return le(jsxRuntime.jsx(ce,{opts:{title:"Repositories",basePath:t},children:e.length===0?jsxRuntime.jsxs("div",{class:"text-center py-20 text-base-content/50",children:[jsxRuntime.jsx("p",{class:"text-lg font-medium mb-1",children:"No repositories configured"}),jsxRuntime.jsx("p",{class:"text-sm",children:"Add a repository to your FRS config to get started."})]}):jsxRuntime.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=>jsxRuntime.jsx("a",{href:`${t}/${n.name}`,class:"card bg-base-100 border border-base-300 hover:shadow-md no-underline transition-shadow",children:jsxRuntime.jsxs("div",{class:"card-body p-5",children:[jsxRuntime.jsx("h2",{class:"card-title text-sm font-semibold",children:n.name}),jsxRuntime.jsx("p",{class:"text-xs text-base-content/50 font-mono",children:n.path})]})},n.name))})}))}var Gt=[{value:"==",label:"="},{value:"!=",label:"\u2260"},{value:"in",label:"in"},{value:"not-in",label:"not in"}],tr=[{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"}],nr=[{value:"array-contains",label:"contains"},{value:"array-contains-any",label:"contains any"}];function rr(e){switch(e){case "ZodNumber":case "ZodBigInt":case "ZodDate":return tr;case "ZodBoolean":return Gt;case "ZodArray":return nr;default:return Gt}}var de="__null__";function sr(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="${de}"]');if(!o){o=document.createElement('option');o.value='${de}';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='${de}';i.readOnly=true;}i.style.opacity='0.55';}else{i.style.opacity='';if(i.tagName==='SELECT'){var o2=i.querySelector('option[value="${de}"][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!=='${de}')?i.dataset._prev:'';}}})(this)`}function or(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 je({inputId:e,active:t}){return jsxRuntime.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:[jsxRuntime.jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",checked:t,onchange:sr(e)}),jsxRuntime.jsx("span",{children:"\u2205"})]})}function ar({col:e,active:t}){let n=t?.value??"",r=n===de,s=`fv_input_${e.name.replace(/\./g,"__")}`,o=t?.op,l=o==="in"||o==="not-in";if(e.enumValues&&e.enumValues.length>0){if(l){let p=new Set(n.split(",").map(g=>g.trim()).filter(Boolean)),u=`eg_${e.name.replace(/\./g,"__")}`;return jsxRuntime.jsxs("div",{class:"flex flex-wrap items-center gap-1 w-full",children:[jsxRuntime.jsx("input",{type:"hidden",id:s,name:`fv_${e.name}`,value:n}),e.enumValues.map(g=>jsxRuntime.jsxs("label",{class:"flex items-center gap-1 text-xs border border-base-300 rounded px-2 cursor-pointer hover:bg-base-200",children:[jsxRuntime.jsx("input",{type:"checkbox",class:"checkbox checkbox-xs",value:g,checked:p.has(g),"data-enum-group":u,onchange:or(s,u)}),jsxRuntime.jsx("span",{children:g})]},g))]})}return jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.jsxs("select",{id:s,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsxRuntime.jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),e.enumValues.map(p=>jsxRuntime.jsx("option",{value:p,selected:n===p,children:p},p)),e.nullable&&jsxRuntime.jsx("option",{value:de,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsxRuntime.jsx(je,{inputId:s,active:r})]})}if(e.zodType==="ZodBoolean")return jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.jsxs("select",{id:s,name:`fv_${e.name}`,class:"select select-sm select-bordered w-full",style:r?"opacity:0.55":void 0,children:[jsxRuntime.jsx("option",{value:"",selected:n===""&&!r,children:"\u2014"}),jsxRuntime.jsx("option",{value:"true",selected:n==="true",children:"true"}),jsxRuntime.jsx("option",{value:"false",selected:n==="false",children:"false"}),e.nullable&&jsxRuntime.jsx("option",{value:de,"data-_auto":"1",selected:r,children:"\u2205 null"})]}),e.nullable&&jsxRuntime.jsx(je,{inputId:s,active:r})]});if(e.zodType==="ZodArray"){let p=t?.op==="array-contains-any";return jsxRuntime.jsx("input",{id:s,type:"text",name:`fv_${e.name}`,value:n,placeholder:p?"val1, val2, \u2026":"value",class:"input input-sm input-bordered w-full"})}return e.zodType==="ZodNumber"||e.zodType==="ZodBigInt"?jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.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&&jsxRuntime.jsx(je,{inputId:s,active:r})]}):e.zodType==="ZodDate"?jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.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&&jsxRuntime.jsx(je,{inputId:s,active:r})]}):jsxRuntime.jsxs("div",{class:"flex items-center gap-1 w-full",children:[jsxRuntime.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&&jsxRuntime.jsx(je,{inputId:s,active:r})]})}function st({action:e,columnMeta:t,activeFilters:n,isGroup:r}){let s=Object.fromEntries(n.map(u=>[u.field,u])),o=n.length>0,l=n.length>=2||r&&o,p=t.filter(u=>u.zodType!=="ZodObject"&&u.zodType!=="ZodRecord");return jsxRuntime.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:[jsxRuntime.jsxs("summary",{class:"collapse-title text-sm font-medium py-2 min-h-0",children:["Filters",o&&jsxRuntime.jsxs("span",{class:"badge badge-primary badge-sm ml-2",children:[n.length," active"]})]}),jsxRuntime.jsx("div",{class:"collapse-content pb-4 pt-2",children:jsxRuntime.jsxs("form",{method:"get",action:e,children:[jsxRuntime.jsx("div",{class:"grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4",children:p.map(u=>{let g=rr(u.zodType),h=s[u.name],k=h?.op??g[0].value;return jsxRuntime.jsxs("div",{class:"flex flex-col gap-1.5",children:[jsxRuntime.jsx("label",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:u.name}),jsxRuntime.jsxs("div",{class:"flex gap-1.5",children:[g.length>1?jsxRuntime.jsx("select",{name:`fo_${u.name}`,class:"select select-sm select-bordered w-20 shrink-0",children:g.map(w=>jsxRuntime.jsx("option",{value:w.value,selected:w.value===k,children:w.label},w.value))}):jsxRuntime.jsx("input",{type:"hidden",name:`fo_${u.name}`,value:g[0].value}),jsxRuntime.jsx(ar,{col:u,active:h})]})]},u.name)})}),jsxRuntime.jsxs("div",{class:"flex flex-wrap gap-2 mt-4 pt-3 border-t border-base-200 items-center",children:[jsxRuntime.jsx("button",{type:"submit",class:"btn btn-sm btn-primary",children:"Apply"}),o&&jsxRuntime.jsx("a",{href:e,class:"btn btn-sm btn-ghost",children:"Clear"}),l&&jsxRuntime.jsxs("span",{class:"text-xs text-warning ml-auto flex items-center gap-1",children:[jsxRuntime.jsx("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:jsxRuntime.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 ot(e,t,n,r,s,o){let l=n==="create"?`Create ${e}`:`Edit ${e} / ${r??""}`,p=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 le(jsxRuntime.jsx(ce,{opts:{title:l,breadcrumb:p,basePath:s,flash:o},children:jsxRuntime.jsx("div",{class:"card bg-base-100 border border-base-300",children:jsxRuntime.jsx("div",{class:"card-body p-6",children:jsxRuntime.jsx("div",{dangerouslySetInnerHTML:{__html:t}})})})}))}Ke();Ye();function at(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 Jt(e,t,n,r,s){let o=at(e,r,s);return o.set("cursor",t),o.set("dir",n),`?${o.toString()}`}function ir(e,t,n,r){let s=at(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 lr(e,t,n){return `?${at(t,n,e).toString()}`}function it(e,t,n,r,s,o,l=[],p=[],u=false,g=[],h,k,w,b,m,y,i){let d=`${r}/${e}`,c=`${d}/create`,a={};if(i)for(let x of n)a[x]=ke($e(i,x));let f=(y??[]).filter(x=>{let R=a[x]??ke($e(i,x));return R==="string"||R==="number"||R==="bigint"||R==="boolean"||R==="enum"||R==="literal"}),v=f.length>0,S=u,O=S||v,A=f.map(x=>{let R=$e(i,x),$=ke(R),T=l.find(P=>P.name===x);return {name:x,type:$,enumValues:T?.enumValues??null,nullable:T?.nullable??false}});return le(jsxRuntime.jsxs(ce,{opts:{title:e,breadcrumb:[{label:"Repositories",href:r},{label:e}],basePath:r,flash:o},children:[l.length>0&&jsxRuntime.jsx(st,{action:d,columnMeta:l,activeFilters:p,isGroup:b}),w&&jsxRuntime.jsxs("div",{role:"alert",class:`alert ${w.type==="index"?"alert-warning":"alert-error"} mb-6 shadow-sm`,children:[jsxRuntime.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:w.type==="index"?jsxRuntime.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"}):jsxRuntime.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"})}),jsxRuntime.jsxs("div",{class:"flex-1",children:[jsxRuntime.jsx("h3",{class:"font-bold",children:w.type==="index"?"Composite index required":"Query failed"}),jsxRuntime.jsx("div",{class:"text-sm",children:w.message})]}),w.indexUrl&&jsxRuntime.jsx("a",{href:w.indexUrl,target:"_blank",rel:"noopener noreferrer",class:"btn btn-sm btn-outline",children:"Create Index \u2192"})]}),jsxRuntime.jsxs("div",{class:"flex flex-wrap justify-between items-center mb-4 gap-3","data-frs-toolbar":true,children:[jsxRuntime.jsxs("div",{class:"flex items-center gap-3",children:[jsxRuntime.jsxs("span",{class:"text-sm text-base-content/60",children:[t.length,typeof m=="number"&&jsxRuntime.jsxs(jsxRuntime.Fragment,{children:[" ","of ",jsxRuntime.jsx("span",{class:"font-semibold text-base-content/80",children:m})]})," ","document",(typeof m=="number"?m:t.length)!==1&&"s"]}),jsxRuntime.jsxs("div",{class:"flex items-center gap-1.5 text-sm text-base-content/60",children:[jsxRuntime.jsx("span",{children:"Rows"}),jsxRuntime.jsx("div",{class:"join",children:[10,25,50,100].map(x=>jsxRuntime.jsx("a",{href:lr(x,p,h),class:`join-item btn btn-xs ${k===x?"btn-active btn-primary":"btn-outline"}`,children:x},x))})]})]}),jsxRuntime.jsx("a",{href:c,class:"btn btn-primary btn-sm",children:"+ New"})]}),O&&jsxRuntime.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(k??t.length),"data-frs-allow-delete":S?"1":"0","data-frs-allow-update":v?"1":"0","data-frs-fields":JSON.stringify(A),"data-frs-filters":JSON.stringify(p),children:[jsxRuntime.jsxs("div",{class:"flex-1 text-sm",children:[jsxRuntime.jsx("span",{"data-frs-bulk-summary":true,children:"0 selected"}),typeof m=="number"&&m>t.length&&jsxRuntime.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"]}),jsxRuntime.jsxs("span",{class:"hidden ml-3 italic","data-frs-bulk-all-active":true,children:["All ",m??"?"," matching documents are selected."," ",jsxRuntime.jsx("button",{type:"button",class:"link","data-frs-bulk-clear":true,children:"Clear selection"})]})]}),jsxRuntime.jsxs("div",{class:"flex gap-2",children:[v&&jsxRuntime.jsx("button",{type:"button",class:"btn btn-sm btn-outline","data-frs-bulk-action":"update",children:"Update field\u2026"}),S&&jsxRuntime.jsx("button",{type:"button",class:"btn btn-sm btn-error btn-outline","data-frs-bulk-action":"delete",children:"Delete"})]})]}),jsxRuntime.jsx("div",{class:"overflow-x-auto rounded-box border border-base-300 bg-base-100","data-frs-table-wrap":true,children:jsxRuntime.jsxs("table",{class:"table table-sm w-full","data-frs-table":true,"data-frs-repo":e,"data-frs-colcount":n.length,children:[jsxRuntime.jsx("thead",{children:jsxRuntime.jsxs("tr",{class:"bg-base-200/50",children:[O&&jsxRuntime.jsx("th",{class:"w-8",children:jsxRuntime.jsx("input",{type:"checkbox",class:"checkbox checkbox-xs checkbox-primary","data-frs-select-page":true,"aria-label":"Select all on this page"})}),[...n].map((x,R)=>{let $=h?.field===x,T=$?h.dir==="asc"?" \u25B2":" \u25BC":"";return jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:jsxRuntime.jsxs("a",{href:ir(x,h,p,k),class:`hover:text-base-content inline-flex items-center gap-0.5${$?" text-primary font-bold":""}`,children:[x,T]})},R)}),g.map((x,R)=>jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide",children:x.column},`rel-${R}`)),jsxRuntime.jsx("th",{class:"text-xs font-semibold text-base-content/60 uppercase tracking-wide text-right",children:"Actions"})]})}),jsxRuntime.jsx("tbody",{children:t.length===0?jsxRuntime.jsx("tr",{children:jsxRuntime.jsx("td",{colspan:n.length+g.length+1+(O?1:0),class:"text-center py-16 text-base-content/40",children:"No documents found"})}):t.map((x,R)=>{let $=String(x.docId??x.id??""),T=`${r}/${e}/${encodeURIComponent($)}/edit`,P=`${r}/${e}/${encodeURIComponent($)}/delete`;return jsxRuntime.jsxs("tr",{class:"hover","data-frs-row-id":$,children:[O&&jsxRuntime.jsx("td",{class:"align-middle py-2",children:jsxRuntime.jsx("input",{type:"checkbox",class:"checkbox checkbox-xs checkbox-primary","data-frs-select-row":true,value:$,"aria-label":`Select ${$}`})}),n.map((C,B)=>{let I=x[C],D=a[C],se=D?De(D,I):null;return jsxRuntime.jsx("td",{class:"align-top py-2",children:jsxRuntime.jsx(Se,{val:I,mismatch:se})},B)}),g.map((C,B)=>{let I=x[C.key];if(I==null||I==="")return jsxRuntime.jsx("td",{class:"py-2"},`rel-${B}`);let D=C.type==="one"?`${r}/${C.targetRepo}/${encodeURIComponent(String(I))}/edit`:`${r}/${C.targetRepo}?fv_${C.targetKey}=${encodeURIComponent(String(I))}`;return jsxRuntime.jsx("td",{class:"align-middle py-2",children:jsxRuntime.jsx("a",{href:D,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(I),"data-frs-rel-label":C.column,children:C.column})},`rel-${B}`)}),jsxRuntime.jsx("td",{class:"align-middle text-right whitespace-nowrap py-2",children:jsxRuntime.jsxs("div",{class:"flex gap-1 justify-end",children:[jsxRuntime.jsx("a",{href:T,class:"btn btn-xs btn-outline",children:"Edit"}),u&&jsxRuntime.jsx("form",{method:"post",action:P,onsubmit:"return confirm('Delete this document?')",children:jsxRuntime.jsx("button",{type:"submit",class:"btn btn-xs btn-error btn-outline",children:"Delete"})})]})})]},R)})})]})}),(s.hasPrev||s.hasNext)&&jsxRuntime.jsxs("div",{class:"flex flex-col items-center mt-6 gap-2",children:[jsxRuntime.jsxs("div",{class:"flex justify-center items-center gap-2",children:[s.hasPrev?jsxRuntime.jsx("a",{href:Jt(p,s.prevCursor,"prev",h,k),class:"btn btn-sm btn-outline",children:"\u2190 Previous"}):jsxRuntime.jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"\u2190 Previous"}),s.hasNext?jsxRuntime.jsx("a",{href:Jt(p,s.nextCursor,"next",h,k),class:"btn btn-sm btn-outline",children:"Next \u2192"}):jsxRuntime.jsx("button",{class:"btn btn-sm btn-outline",disabled:true,children:"Next \u2192"})]}),typeof m=="number"&&jsxRuntime.jsxs("div",{class:"text-xs text-base-content/50",children:[m," total document",m!==1?"s":"",p.length>0?" matching filters":""]})]}),O&&v&&jsxRuntime.jsxs("dialog",{id:"frs-bulk-update-modal",class:"modal",children:[jsxRuntime.jsxs("div",{class:"modal-box",children:[jsxRuntime.jsx("h3",{class:"font-bold text-lg mb-3",children:"Bulk update field"}),jsxRuntime.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."}),jsxRuntime.jsxs("form",{method:"dialog","data-frs-bulk-update-form":true,children:[jsxRuntime.jsxs("label",{class:"form-control w-full mb-3",children:[jsxRuntime.jsx("div",{class:"label",children:jsxRuntime.jsx("span",{class:"label-text text-xs uppercase tracking-wide",children:"Field"})}),jsxRuntime.jsxs("select",{class:"select select-bordered select-sm w-full",name:"field",required:true,"data-frs-bulk-field-select":true,children:[jsxRuntime.jsx("option",{value:"",children:"\u2014 Select a field \u2014"}),f.map(x=>jsxRuntime.jsx("option",{value:x,children:x},x))]})]}),jsxRuntime.jsx("div",{class:"mb-4","data-frs-bulk-value-container":true}),jsxRuntime.jsxs("div",{class:"modal-action",children:[jsxRuntime.jsx("button",{type:"button",class:"btn btn-sm btn-ghost","data-frs-bulk-update-cancel":true,children:"Cancel"}),jsxRuntime.jsx("button",{type:"submit",class:"btn btn-sm btn-primary","data-frs-bulk-update-submit":true,children:"Apply"})]})]})]}),jsxRuntime.jsx("form",{method:"dialog",class:"modal-backdrop",children:jsxRuntime.jsx("button",{children:"close"})})]})]}))}var dr="";function Ge(e,t){return nt(e,t)}function lt(e,t){return rt(e,t)}function ct(e,t,n,r,s,o,l,p,u,g,h,k,w,b,m,y,i){return it(e,t,n,r,s,o,l,p,u,g,h,k,w,b,m,y,i)}function ue(e,t,n,r,s,o){return ot(e,t,n,r,s,o)}var ur=new Set(["<","<=",">",">=","!="]),pr=new Set(["array-contains","array-contains-any"]);function dt(e){return e==="desc"?"DESCENDING":"ASCENDING"}function fr(e){let t=e.split("/").filter(Boolean);return t[t.length-1]??e}function mr(e,t,n,r,s){let o=[],l=new Set;for(let u of r)if(u.op==="=="||u.op==="in"||u.op==="not-in"){if(l.has(u.field))continue;l.add(u.field),o.push({fieldPath:u.field,order:"ASCENDING"});}for(let u of r)if(pr.has(u.op)){if(l.has(u.field))continue;l.add(u.field),o.push({fieldPath:u.field,arrayConfig:"CONTAINS"});}for(let u of r)if(ur.has(u.op)){if(l.has(u.field))continue;l.add(u.field);let g=s?.field===u.field?dt(s.dir):"ASCENDING";o.push({fieldPath:u.field,order:g});}if(s&&!l.has(s.field)&&o.push({fieldPath:s.field,order:dt(s.dir)}),o.length===1&&n)return br(e,t,o[0]);let p=s&&o.some(u=>u.fieldPath===s.field)?dt(s.dir):"ASCENDING";return o.push({fieldPath:"__name__",order:p}),yr(e,t,n,o)}function yr(e,t,n,r,s="(default)"){let o=`projects/${e}/databases/${s}/collectionGroups/${t}/indexes/_`,l=[...ft(1,o),...Je(2,n?2:1)];for(let g of r)l.push(...Qt(3,Wt(g)));let p=s==="(default)"?"-default-":s,u=encodeURIComponent(Xt(l));return `https://console.firebase.google.com/project/${e}/firestore/databases/${p}/indexes?create_composite=${u}`}function gr(e){return e.match(/https:\/\/console\.firebase\.google\.com[^\s)"]*/)?.[0]}function ut(e){let t=[],n=e>>>0;for(;n>=128;)t.push(n&127|128),n>>>=7;return t.push(n&127),t}function pt(e,t){return e<<3|t}function ft(e,t){let n=Array.from(new TextEncoder().encode(t));return [pt(e,2),...ut(n.length),...n]}function Je(e,t){return [pt(e,0),...ut(t)]}function Qt(e,t){return [pt(e,2),...ut(t.length),...t]}function Wt(e){let t=[...ft(1,e.fieldPath)];return e.arrayConfig==="CONTAINS"?t.push(...Je(3,1)):t.push(...Je(2,e.order==="DESCENDING"?2:1)),t}function Xt(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 br(e,t,n,r="(default)"){let s=`projects/${e}/databases/${r}/collectionGroups/${t}/fields/${n.fieldPath}`,o=[...ft(1,s),...Je(2,2),...Qt(3,Wt(n))],l=r==="(default)"?"-default-":r,p=encodeURIComponent(Xt(o));return `https://console.firebase.google.com/project/${e}/firestore/databases/${l}/indexes/automatic?create_exemption=${p}`}function hr(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 he(e){let t=e;return t?t.code===9?true:typeof t.message=="string"?t.message.includes("requires an index"):false:false}function ze(e,t){let n=e??{},r=he(e),s;if(r&&(s=n.message?gr(n.message):void 0,!s)){let o=hr(t.ref);if(o){let l=fr(t.path);s=mr(o,l,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 Yt="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function xr(){let e="";for(let t=0;t<20;t++)e+=Yt.charAt(Math.floor(Math.random()*Yt.length));return e}function mt(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 Ze(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 yt(e,t,n){let r=e.documentKey??"docId",s=ze(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 gt(e,t){e.status(302).set("Location",t).send("");}function bt(e,t){let n=t.shape,r={};for(let[s,o]of Object.entries(n)){let l=vt(o);if(l==="ZodObject"){if(e[s+"__isnull"]==="1"){r[s]=null;continue}let g={},h=false;for(let[b,m]of Object.entries(e))b.startsWith(`${s}.`)&&(g[b.slice(s.length+1)]=m,h=true);if(h){let b=o;for(;;){let m=q(b);if(m==="ZodOptional"||m==="ZodNullable"||m==="ZodDefault")b=U(b);else break}r[s]=bt(g,b);continue}let k=e[s],w=Array.isArray(k)?k[k.length-1]:k;if(w)try{r[s]=JSON.parse(w);}catch{r[s]=w;}continue}let p=e[s],u=Array.isArray(p)?p[p.length-1]:p;if(e[s+"__isnull"]==="1"){r[s]=null;continue}if(u===void 0||u===""){l==="ZodBoolean"&&(r[s]=false);continue}switch(l){case "ZodBoolean":u==="__null__"?r[s]=null:r[s]=u==="true"||u==="on"||u==="1";break;case "ZodNumber":case "ZodBigInt":r[s]=Number(u);break;case "ZodDate":r[s]=new Date(u);break;case "ZodArray":try{r[s]=JSON.parse(u);}catch{r[s]=u;}break;default:if(u.startsWith("{")||u.startsWith("["))try{r[s]=JSON.parse(u);break}catch{}r[s]=u;}}return r}function en(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 vt(e){let t=e;for(;;){let n=q(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=U(t);else return n}}function wr(e){let t=e;for(;;){let n=q(t);if(n==="ZodOptional"||n==="ZodNullable"||n==="ZodDefault")t=U(t);else return t}}function nn(e){let t=e;for(;;){let n=q(t);if(n==="ZodOptional"||n==="ZodNullable")return true;if(n==="ZodDefault"){t=U(t);continue}return false}}function rn(e){let t=wr(e),n=q(t);if(n==="ZodEnum"){let r=Ae(t);return r.length>0?r:void 0}if(n==="ZodNativeEnum"){let r=Ee(t),s=Object.values(r).filter(o=>typeof o=="string");return s.length>0?s:void 0}if(n==="ZodLiteral"){let r=Ue(t);return typeof r=="string"?[r]:void 0}}function sn(e,t,n=""){let r={};for(let s of Object.keys(t.shape)){let o=n?`${n}.${s}`:s,l=e[s];if(l===null){r[o]="__null__";continue}if(l===void 0)continue;let p=t.shape[s];for(;;){let g=q(p);if(g==="ZodOptional"||g==="ZodNullable"||g==="ZodDefault")p=U(p);else break}let u=q(p);if(u==="ZodObject"&&typeof l=="object"&&l!==null&&!Array.isArray(l)){let g=sn(l,p,o);Object.assign(r,g);}else if(u==="ZodDate"){let g=en(l);g!==null&&(r[o]=g);}else if(typeof l=="object"&&l!==null&&!Array.isArray(l)&&("_seconds"in l||typeof l.toDate=="function")){let g=en(l);r[o]=g??JSON.stringify(l,null,2);}else typeof l=="object"?r[o]=JSON.stringify(l,null,2):r[o]=String(l);}return r}function ht(e,t){return e.map(n=>({...n,defaultValue:t[n.name]??n.defaultValue,nested:n.nested?ht(n.nested,t):void 0}))}function Rr(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 l=s.slice(3);if(!t.has(l))continue;let p=(o??"").trim();if(!p)continue;let u=e[`fo_${l}`]??"==",g=n.has(u)?u:"==";r.push({field:l,op:g,value:p});}return r}function tn(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 on(e,t,n=""){let r=[];for(let s of e){let o=n?`${n}.${s}`:s,l=t.shape[s];if(!l){r.push({name:o,zodType:"ZodString"});continue}let p=vt(l);if(p==="ZodObject"){let u=l;for(;;){let h=q(u);if(h==="ZodOptional"||h==="ZodNullable"||h==="ZodDefault")u=U(u);else break}let g=Q(u);r.push(...on(Object.keys(g),u,o));}else r.push({name:o,zodType:p,nullable:nn(l),enumValues:rn(l)});}return r}function Sr(e,t){let n=t.split("."),r=e;for(let s of n){for(;;){let l=q(r);if(l==="ZodOptional"||l==="ZodNullable"||l==="ZodDefault")r=U(r);else break}let o=Q(r);if(!(s in o))return null;r=o[s];}return r}function ve(e,t){if(!t||t.length===0)return e;let n=[],r=new Map;for(let o of t){let l=o.indexOf(".");if(l===-1)n.push(o);else {let p=o.slice(0,l),u=o.slice(l+1);r.has(p)||r.set(p,[]),r.get(p).push(u);}}let s={};for(let o of n)o in e.shape&&(s[o]=e.shape[o]);for(let[o,l]of r){if(!(o in e.shape))continue;let p=e.shape[o];for(;;){let u=q(p);if(u==="ZodOptional"||u==="ZodNullable"||u==="ZodDefault")p=U(p);else break}if(q(p)!=="ZodObject"){s[o]=e.shape[o];continue}s[o]=ve(p,l);}return zod.z.object(s)}function pe(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",l=process.env.FUNCTION_REGION??"us-central1",p=(process.env.FUNCTION_TARGET??"").replace(/\./g,"-");return `/${o}/${l}/${p}${n}`}let r=process.env.K_SERVICE,s=e.hostname??e.headers?.host??"";return r&&s.includes("cloudfunctions.net")?`/${r.toLowerCase()}${n}`:n}function an(e,t){let n=(i,d)=>{let c=pe(i,t),a=Object.values(e).map(f=>({name:f.name,path:f.path}));z(d,lt(a,c));},r=async(i,d)=>{let c=i.params.repoName;if(!c){z(d,"Bad request",400);return}let a=e[c];if(!a){z(d,"Repository not found",404);return}let f=a.pageSize??25,v=i.query??{},S=v.cursor,O=v.dir==="prev"?"prev":"next",A=v.ob??"",x=v.od==="desc"?"desc":"asc",R=A?{field:A,dir:x}:void 0,$=parseInt(v.ps??""),T=Number.isFinite($)&&$>0?Math.min($,200):f,P=a.listColumns??Object.keys(a.schema.shape),C=a.documentKey??"docId",B=[C,...P.filter(K=>K!==C)],I=a.filterableFields?(()=>{let K=[];for(let oe of a.filterableFields)(oe.includes(".")||P.includes(oe))&&K.push(oe);return K})():P,D=(()=>{let K=[];for(let oe of I)if(oe.includes(".")){let we=Sr(a.schema,oe);K.push({name:oe,zodType:we?vt(we):"ZodString",nullable:we?nn(we):false,enumValues:we?rn(we):void 0});}else K.push(...on([oe],a.schema));return K})(),se=new Set(D.map(K=>K.name)),Te=Rr(v,se),Be=tn(Te),Ct;if(S)try{let K=a.repo.ref;typeof K.doc=="function"&&(Ct=await K.doc(S).get());}catch{}let[me,Sn]=await Promise.all([a.repo.query.paginate({pageSize:T,cursor:Ct,direction:O,...Be.length>0?{where:Be}:{},...R?{orderBy:[{field:R.field,direction:R.dir}]}:{}}).catch(K=>({queryError:ze(K,{ref:a.repo.ref,path:a.path,isGroup:!!a.isGroup,filters:Te,sort:R})})),a.repo.aggregate.count(Be.length>0?{where:Be}:{}).catch(()=>{})]),xe="queryError"in me,kn=xe?[]:me.data,$n=xe?"":me.nextCursor?.id??"",Cn=xe?"":me.prevCursor?.id??"",On=xe?me.queryError:void 0,Tn=pe(i,t);z(d,ct(a.name,kn,B,Tn,{hasPrev:xe?false:me.hasPrevPage,hasNext:xe?false:me.hasNextPage,prevCursor:Cn,nextCursor:$n},void 0,D,Te,a.allowDelete??false,a.relationalMeta,R,T,On,a.isGroup,Sn,a.mutableFields,a.schema));},s=(i,d)=>{let c=i.params.repoName;if(!c){z(d,"Bad request",400);return}let a=e[c];if(!a){z(d,"Repository not found",404);return}let f=pe(i,t),v=ve(a.schema,a.createFields),S=ne(v),O=`${f}/${a.name}/create`,A=ie(S,O,"POST","Create document");z(d,ue(a.name,A,"create",null,f));},o=async(i,d)=>{let c=i.params.repoName;if(!c){z(d,"Bad request",400);return}let a=e[c];if(!a){z(d,"Repository not found",404);return}let f=pe(i,t),v=i.body??{},S=bt(v,a.schema),O=ve(a.schema,a.createFields),A=O.safeParse(S);if(!A.success){let x=ne(O),R=`${f}/${a.name}/create`,$=ie(x,R,"POST","Create document"),T=A.error.issues.map(P=>`${P.path.join(".")}: ${P.message}`).join(", ");z(d,ue(a.name,$,"create",null,f,{type:"error",message:`Validation error: ${T}`}),422);return}try{if(a.isGroup&&a.parentKeys&&a.parentKeys.length>0){let x={...A.data};a.createdKey&&(x[a.createdKey]=new Date);let R=a.parentKeys.filter(C=>!x[C]);if(R.length>0)throw new Error(`Missing parent key(s) for subcollection create: ${R.join(", ")}`);let $=a.parentKeys.map(C=>x[C]),T=a.documentKey??"docId",P=x[T]||xr();await a.repo.set(...$,P,x);}else await a.repo.create(A.data);gt(d,`${f}/${a.name}?flash=created`);}catch(x){let R=ve(a.schema,a.createFields),$=ne(R),T=`${f}/${a.name}/create`,P=ie($,T,"POST","Create document");z(d,ue(a.name,P,"create",null,f,{type:"error",message:`Save error: ${x.message}`}),500);}},l=async(i,d)=>{let c=i.params.repoName,a=i.params.id;if(!c||!a){z(d,"Bad request",400);return}let f=e[c];if(!f){z(d,"Repository not found",404);return}let v=pe(i,t),S=null;try{S=await Ze(f,a);}catch(T){let P=yt(f,a,T),C=he(T)?424:500;z(d,Ge("",{title:`Edit ${f.name} / ${a}`,basePath:v,breadcrumb:[{label:"Repositories",href:v},{label:f.name,href:`${v}/${f.name}`},{label:`Edit ${a}`}],flash:P}),C);return}if(!S){z(d,"Document not found",404);return}let O=sn(S,f.schema),A=ve(f.schema,f.mutableFields),x=ht(ne(A),O),R=`${v}/${f.name}/${encodeURIComponent(a)}/edit`,$=ie(x,R,"POST","Save changes");z(d,ue(f.name,$,"edit",a,v));},p=async(i,d)=>{let c=i.params.repoName,a=i.params.id;if(!c||!a){z(d,"Bad request",400);return}let f=e[c];if(!f){z(d,"Repository not found",404);return}let v=pe(i,t),S=i.body??{},O=bt(S,f.schema),A=ve(f.schema,f.mutableFields),R=A.partial().safeParse(O);if(!R.success){let $=Object.fromEntries(Object.entries(S).map(([I,D])=>[I,Array.isArray(D)?D.join(","):D??""])),T=ht(ne(A),$),P=`${v}/${f.name}/${encodeURIComponent(a)}/edit`,C=ie(T,P,"POST","Save changes"),B=R.error.issues.map(I=>`${I.path.join(".")}: ${I.message}`).join(", ");z(d,ue(f.name,C,"edit",a,v,{type:"error",message:`Validation error: ${B}`}),422);return}try{let $=await Ze(f,a),T=($&&mt($,f.pathKey))??[a];await f.repo.update(...T,R.data),gt(d,`${v}/${f.name}?flash=updated`);}catch($){let T=ve(f.schema,f.mutableFields),P=ne(T),C=`${v}/${f.name}/${encodeURIComponent(a)}/edit`,B=ie(P,C,"POST","Save changes"),I=he($)?yt(f,a,$):{type:"error",message:`Save error: ${$.message}`},D=he($)?424:500;z(d,ue(f.name,B,"edit",a,v,I),D);}},u=async(i,d)=>{let c=i.params.repoName,a=i.params.id;if(!c||!a){z(d,"Bad request",400);return}let f=e[c];if(!f){z(d,"Repository not found",404);return}if(!f.allowDelete){z(d,"Delete is not allowed for this repository",403);return}let v=pe(i,t);try{let S=await Ze(f,a),O=(S&&mt(S,f.pathKey))??[a];await f.repo.delete(...O),gt(d,`${v}/${f.name}?flash=deleted`);}catch(S){let O=he(S)?yt(f,a,S):{type:"error",message:`Delete error: ${S.message}`},A=he(S)?424:500;z(d,Ge("",{title:`Delete ${f.name} / ${a}`,basePath:v,breadcrumb:[{label:"Repositories",href:v},{label:f.name,href:`${v}/${f.name}`},{label:`Delete ${a}`}],flash:O}),A);}},g=async(i,d)=>{let c=i.params.repoName;if(!c){z(d,"Bad request",400);return}let a=e[c];if(!a){z(d,"Repository not found",404);return}let f=pe(i,t),v=i.query,S=v?.type==="many"?"many":"one",O=Math.max(1,Math.min(100,Number(v?.ps??25)||25)),A=a.listColumns??Object.keys(Q(a.schema)),{PanelOne:x,PanelMany:R}=await Promise.resolve().then(()=>(tt(),Kt)),{renderToString:$}=await import('hono/jsx/dom/server');if(S==="one"){let D=String(v?.id??"");if(!D){z(d,"<div class='p-6 text-error'>Missing id parameter.</div>",400);return}try{let se=await Ze(a,D),Te=$(x({doc:se,repoName:a.name,basePath:f,schema:a.schema,columns:A}));z(d,Te);}catch(se){z(d,`<div class='p-6 text-error text-sm'>Error: ${se.message}</div>`,500);}return}let T=String(v?.fk??""),P=String(v?.fv??"");if(!T||!P){z(d,"<div class='p-6 text-error'>Missing fk/fv parameters.</div>",400);return}let C=v?.cursor??"",B=v?.dir==="prev"?"prev":"next",I;if(C)try{let D=a.repo.ref;typeof D.doc=="function"&&(I=await D.doc(C).get());}catch{}try{let D=await a.repo.query.paginate({pageSize:O,cursor:I,direction:B,where:[[T,"==",kr(P)]]}),se=$(R({docs:D.data,repoName:a.name,basePath:f,fk:T,fv:P,columns:A,schema:a.schema,pagination:{hasPrev:D.hasPrevPage,hasNext:D.hasNextPage,prevCursor:D.prevCursor?.id??"",nextCursor:D.nextCursor?.id??"",pageSize:O}}));z(d,se);}catch(D){z(d,`<div class='p-6 text-error text-sm'>Error: ${D.message}</div>`,500);}},h=async(i,d)=>{let c=[];for(let a of d){let f;if(i.isGroup||i.parentKeys?.length){let v=await Ze(i,a);f=v?mt(v,i.pathKey):void 0;}f||(f=[a]);try{let v=i.repo.documentRef(...f);v&&c.push(v);}catch{}}return c},k=async(i,d)=>{let c=tn(d),a=i.documentKey??"docId",f=[],v;for(;;){let S=await i.repo.query.paginate({pageSize:500,cursor:v,direction:"next",...c.length>0?{where:c}:{}});for(let O of S.data){let A=String(O[a]??O.id??"");A&&f.push(A);}if(!S.hasNextPage||!S.nextCursor)break;v=S.nextCursor;}return f},w=async(i,d)=>{let c=i.params.repoName;if(!c){J(d,{error:"Bad request"},400);return}let a=e[c];if(!a){J(d,{error:"Repository not found"},404);return}if(!a.allowDelete){J(d,{error:"Delete is not allowed for this repository"},403);return}let f=i.body??{};try{let v=await m(a,f);if(v.length===0){J(d,{deleted:0});return}let S=await h(a,v);for(let O=0;O<S.length;O+=500)await a.repo.bulk.delete(S.slice(O,O+500));J(d,{deleted:S.length});}catch(v){J(d,{error:v.message},500);}},b=async(i,d)=>{let c=i.params.repoName;if(!c){J(d,{error:"Bad request"},400);return}let a=e[c];if(!a){J(d,{error:"Repository not found"},404);return}let f=i.body??{},v=String(f.field??"");if(!v){J(d,{error:"Missing 'field'"},400);return}if(!a.mutableFields||!a.mutableFields.includes(v)){J(d,{error:`Field '${v}' is not bulk-updatable`},403);return}let S=a.schema.shape?.[v],O=f.value;if(S){let A=S.safeParse(f.value);if(!A.success){J(d,{error:`Invalid value for '${v}': ${A.error.message}`},400);return}O=A.data;}try{let A=await m(a,f);if(A.length===0){J(d,{updated:0});return}let R=(await h(a,A)).map($=>({docRef:$,data:{[v]:O}}));for(let $=0;$<R.length;$+=500)await a.repo.bulk.update(R.slice($,$+500));J(d,{updated:R.length});}catch(A){J(d,{error:A.message},500);}};async function m(i,d){if(d.selectAll){let c=y(d.filters,i);return await k(i,c)}return Array.isArray(d.ids)?d.ids.filter(c=>typeof c=="string"&&!!c):[]}function y(i,d){if(!Array.isArray(i))return [];let c=new Set((d.filterableFields??Object.keys(Q(d.schema))).map(v=>String(v))),a=new Set(["==","!=","<","<=",">",">=","in","not-in","array-contains","array-contains-any"]),f=[];for(let v of i)v&&typeof v=="object"&&typeof v.field=="string"&&c.has(v.field)&&typeof v.value=="string"&&a.has(String(v.op))&&f.push({field:v.field,op:v.op,value:v.value});return f}return {handleDashboard:n,handleList:r,handleCreateForm:s,handleCreateSubmit:o,handleEditForm:l,handleEditSubmit:p,handleDelete:u,handlePanel:g,handleBulkDelete:w,handleBulkUpdate:b}}function J(e,t,n=200){e.status(n).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(t));}function kr(e){return e==="true"?true:e==="false"?false:e!==""&&!isNaN(Number(e))?Number(e):e}async function $r(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function Cr(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," ")),l=t[s];l===void 0?t[s]=o:Array.isArray(l)?l.push(o):t[s]=[l,o];}return t}function ln(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:s,middleware:o=[],httpsOptions:l}=e,p=t==="/"?"":t.replace(/\/$/,""),u={};for(let[w,b]of Object.entries(n)){let m=b.schema??b.repo.schema??null;if(!m)throw new Error(`[createAdminServer] Repository "${w}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let y,i,d;if(b.fieldsConfig){let f=b.fieldsConfig;y=[],i=[],d=[];for(let[v,S]of Object.entries(f))for(let O of S)O==="filterable"?y.push(v):O==="mutable"?i.push(v):O==="create"&&d.push(v);y.length===0&&(y=void 0),i.length===0&&(i=void 0),d.length===0&&(d=void 0);}let c=(()=>{let f=b.repo._parentKeys;return f&&f.length>0?f:void 0})();if(c&&d)for(let f of c)d.includes(f)||d.push(f);let a={name:w,path:b.path,repo:b.repo,schema:m,documentKey:b.documentKey??"docId",pathKey:b.repo._pathKey??void 0,isGroup:!!b.repo._isGroup,parentKeys:c,createdKey:b.repo._createdKey??void 0,listColumns:b.listColumns,pageSize:b.pageSize,filterableFields:y,mutableFields:i,createFields:d,allowDelete:b.allowDelete??false,relationalMeta:(()=>{if(!b.relationalFields||b.relationalFields.length===0)return;let f=b.repo.relationalKeys??{},v=[];for(let S of b.relationalFields){let O=f[S.key];O&&v.push({key:S.key,column:S.column,targetRepo:String(O.repo),targetKey:String(O.key),type:O.type});}return v.length>0?v:void 0})()};u[w]=a;}let g=an(u,p),h=new Y;if(r&&h.use(async(w,b,m)=>{let y=w,i=String(y.headers?.["content-type"]??"");if(i.includes("application/x-www-form-urlencoded")){let d=await $r(y);w.body=Cr(d);}else if(i.includes("application/json")&&typeof y.body=="string")try{w.body=JSON.parse(y.body);}catch{}await m();}),s)if(typeof s=="function")h.use(s);else {let w=s.realm??"Admin",b="Basic "+Buffer.from(`${s.username}:${s.password}`).toString("base64");h.use((m,y,i)=>{if((m.headers?.authorization??"")!==b){y.status(401).set("WWW-Authenticate",`Basic realm="${w}"`).set("Content-Type","text/plain").send("Unauthorized");return}i();});}for(let w of o)h.use(w);h.get(`${p}/`,g.handleDashboard),h.get(`${p}`,g.handleDashboard),h.get(`${p}/:repoName/_panel`,g.handlePanel),h.post(`${p}/:repoName/_bulk/delete`,g.handleBulkDelete),h.post(`${p}/:repoName/_bulk/update`,g.handleBulkUpdate),h.get(`${p}/:repoName`,g.handleList),h.get(`${p}/:repoName/create`,g.handleCreateForm),h.post(`${p}/:repoName/create`,g.handleCreateSubmit),h.get(`${p}/:repoName/:id/edit`,g.handleEditForm),h.post(`${p}/:repoName/:id/edit`,g.handleEditSubmit),h.post(`${p}/:repoName/:id/delete`,g.handleDelete);let k=async(w,b)=>{await h.handle(w,b);};return l&&(k.httpsOptions=l),k}var dn="preserve";function un(){return dn}function Or(e){return typeof e=="object"&&e!==null&&typeof e._seconds=="number"&&typeof e._nanoseconds=="number"}function pn(e){if(e==null)return null;if(e instanceof Date)return Number.isNaN(e.getTime())?null:e;if(e instanceof firestore.Timestamp)return e.toDate();if(Or(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 fn(e){return e}function Rt(e,t,n=200){let r=fn(t);e.status(n).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(r));}function Ce(e,t,n,r=200){Rt(e,{success:true,data:t,meta:n},r);}function H(e,t,n=400){Rt(e,{success:false,error:t},n);}function wt(e,t,n,r,s){let o=ze(t,n),l=o.type==="index",p=l?424:500,g={success:false,error:l?o.message:s&&t instanceof Error?t.message:r};l&&(g.errorType="index",o.indexUrl&&(g.indexUrl=o.indexUrl)),Rt(e,g,p);}var mn="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";function Ar(){let e="";for(let t=0;t<20;t++)e+=mn.charAt(Math.floor(Math.random()*mn.length));return e}function Oe(e){let t=e._def??e.def;if(!t)return e;let n=t.typeName??t.type;if(n==="ZodDate"||n==="date")return zod.z.preprocess(r=>pn(r)??r,e);if(n==="ZodObject"||n==="object"){let r=e.shape,s={};for(let[o,l]of Object.entries(r))s[o]=Oe(l);return zod.z.object(s)}if(n==="ZodArray"||n==="array"){let r=t.element??t.type;if(r)return zod.z.array(Oe(r))}if(n==="ZodOptional"||n==="optional"){let r=t.innerType;if(r)return Oe(r).optional()}if(n==="ZodNullable"||n==="nullable"){let r=t.innerType;if(r)return Oe(r).nullable()}if(n==="ZodDefault"||n==="default"){let r=t.innerType,s=t.defaultValue;if(r){let o=Oe(r);return typeof s=="function"?o.default(s()):o.default(s)}}return e}function Er(e,t,n=[]){let r=e.shape,s={},o=t&&t.length>0?t:Object.keys(r);for(let l of o){if(n.includes(l))continue;let p=l.split(".")[0];p&&r[p]&&(s[p]=r[p]);}return zod.z.object(s)}function yn(e,t,n,r=false,s=[]){try{let o=Er(e,n,s),l=r?o.partial():o;return {success:!0,data:(un()==="normalize"?Oe(l):l).parse(t)}}catch(o){return o instanceof zod.z.ZodError?{success:false,error:`Validation failed: ${o.issues.map(p=>`${p.path.join(".")}: ${p.message}`).join(", ")}`}:{success:false,error:"Validation failed"}}}function Pr(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,l]of Object.entries(e)){if(l===void 0||["cursor","limit","pageSize","orderBy","orderDir","select"].includes(o))continue;let p=Array.isArray(l)?l[0]:l;if(p===void 0||p==="")continue;let u=o.match(/^(\w+)__(\w+)$/),g,h="==";if(u&&u[1]&&u[2]){g=u[1];let w=u[2];if(s[w])h=s[w];else continue}else if(!u)g=o;else continue;if(r&&!r.has(g))continue;let k=p;h==="in"||h==="not-in"||h==="array-contains-any"?k=p.split(",").map(w=>gn(w.trim())):k=gn(p),n.push({field:g,op:h,value:k});}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 Qe(e){return e?{docId:e.id}:null}async function bn(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 hn(e,t,n){function r(b,m){return !b||!e[b]?(H(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 i=y.split("/").filter(Boolean),d=[];for(let c=1;c<i.length;c+=2)d.push(i[c]);return d.length>0?d:void 0}async function o(b,m){let y=`by${b.documentKey.charAt(0).toUpperCase()}${b.documentKey.slice(1)}`,i=b.repo.get[y];if(typeof i=="function")try{let c=await i(m);if(c)return c}catch{}return (await b.repo.query.by({where:[[b.documentKey,"==",m]],limit:1}))[0]??null}async function l(b,m){let y=b.params||{},i=r(y.repoName,m);if(!i)return;let d=[],c;try{let a=b.query??{},f=Math.min(Number(a.pageSize)||i.pageSize,100),v=a.cursor,S=a.direction?.toLowerCase()==="prev"?"prev":"next",O=a.orderBy,A=a.orderDir?.toLowerCase()==="desc"?"desc":"asc",x=a.select,R=x?x.split(",").map(I=>I.trim()):void 0,$;i.allowedIncludes&&a.includes&&($=(typeof a.includes=="string"?a.includes.split(",").map(D=>D.trim()):Array.isArray(a.includes)?a.includes:[]).filter(D=>typeof D=="string"&&i.allowedIncludes.includes(D)),$?.length===0&&($=void 0));let T=Pr(a,i.filterableFields);d=T.map(I=>({field:I.field,op:I.op,value:String(I.value??"")})),O&&(c={field:O,dir:A});let P={pageSize:f,direction:S};if(v)try{let I=typeof v=="string"?JSON.parse(v):v;P.cursor=await bn(i,I);}catch{}O&&(P.orderBy=[{field:O,direction:A}]),T.length>0&&(P.where=T.map(I=>[I.field,I.op,I.value])),R&&(P.select=R),$&&(P.include=$);let C=await i.repo.query.paginate(P),B={items:C.data,hasNextPage:C.hasNextPage,hasPrevPage:C.hasPrevPage,nextCursor:Qe(C.nextCursor),prevCursor:Qe(C.prevCursor)};Ce(m,B,{pageSize:f,hasMore:C.hasNextPage});}catch(a){wt(m,a,{ref:i.repo.ref,path:i.path,isGroup:!!i.isGroup,filters:d,sort:c},"Failed to fetch documents",n);}}async function p(b,m){let y=b.params||{},i=r(y.repoName,m);if(!i)return;let d=[],c;try{let a=b.body??{},f=Math.min(a.pageSize||i.pageSize,100),v=a.direction==="prev"?"prev":"next";a.where&&(d=a.where.map(x=>({field:String(x[0]),op:x[1],value:String(x[2]??"")}))),a.orderBy&&a.orderBy[0]&&(c={field:a.orderBy[0].field,dir:a.orderBy[0].direction==="desc"?"desc":"asc"});let S={pageSize:f,direction:v};if(a.cursor)try{let x=typeof a.cursor=="string"?JSON.parse(a.cursor):a.cursor;S.cursor=await bn(i,x);}catch{}if(i.allowedIncludes&&a.includes&&a.includes.length>0){let x=a.includes.filter(R=>typeof R=="string"?i.allowedIncludes.includes(R):typeof R=="object"&&R!==null&&"relation"in R&&typeof R.relation=="string"?i.allowedIncludes.includes(R.relation):!1);x.length>0&&(S.include=x);}if(a.where&&a.where.length>0){if(i.filterableFields){let x=new Set(i.filterableFields),R=a.where.filter($=>!x.has($[0]));if(R.length>0){H(m,`Fields not filterable: ${R.map($=>$[0]).join(", ")}`,400);return}}S.where=a.where;}if(a.orWhere&&a.orWhere.length>0){if(i.filterableFields){let x=new Set(i.filterableFields),R=a.orWhere.filter($=>!x.has($[0]));if(R.length>0){H(m,`Fields not filterable: ${R.map($=>$[0]).join(", ")}`,400);return}}S.orWhere=a.orWhere;}if(a.orWhereGroups&&a.orWhereGroups.length>0){if(i.filterableFields){let x=new Set(i.filterableFields);for(let R of a.orWhereGroups){let $=R.filter(T=>!x.has(T[0]));if($.length>0){H(m,`Fields not filterable: ${$.map(T=>T[0]).join(", ")}`,400);return}}}S.orWhereGroups=a.orWhereGroups;}a.orderBy&&a.orderBy.length>0&&(S.orderBy=a.orderBy),a.select&&a.select.length>0&&(S.select=a.select);let O=await i.repo.query.paginate(S),A={items:O.data,hasNextPage:O.hasNextPage,hasPrevPage:O.hasPrevPage,nextCursor:Qe(O.nextCursor),prevCursor:Qe(O.prevCursor)};Ce(m,A,{pageSize:f,hasMore:O.hasNextPage});}catch(a){wt(m,a,{ref:i.repo.ref,path:i.path,isGroup:!!i.isGroup,filters:d,sort:c},"Failed to query documents",n);}}async function u(b,m){let y=b.params||{},i=r(y.repoName,m);if(!i)return;let d=y.id;if(!d){H(m,"Document ID required",400);return}try{let c=await o(i,d);if(!c){H(m,"Document not found",404);return}Ce(m,c);}catch(c){wt(m,c,{ref:i.repo.ref,path:i.path,isGroup:!!i.isGroup,filters:[{field:i.documentKey,op:"==",value:d}]},"Failed to fetch document",n);}}async function g(b,m){let y=b.params||{},i=r(y.repoName,m);if(i)try{let d=b.body??{},c=yn(i.schema,d,i.createFields,!1,i.systemKeys);if(!c.success){H(m,c.error,400);return}if(i.validate){let f=await i.validate(c.data,"create");if(f){H(m,f,400);return}}let a;if(i.isGroup&&i.parentKeys&&i.parentKeys.length>0){let f={...c.data};i.createdKey&&(f[i.createdKey]=new Date);let v=i.parentKeys.filter(A=>!f[A]);if(v.length>0){H(m,`Missing parent key(s) for subcollection create: ${v.join(", ")}`,400);return}let S=i.parentKeys.map(A=>f[A]),O=f[i.documentKey]||Ar();a=await i.repo.set(...S,O,f);}else a=await i.repo.create(c.data);Ce(m,a,void 0,201);}catch(d){let c=n&&d instanceof Error?d.message:"Failed to create document";H(m,c,500);}}async function h(b,m,y){let i=b.params||{},d=r(i.repoName,m);if(!d)return;let c=i.id;if(!c){H(m,"Document ID required",400);return}try{let a=b.body??{},f=yn(d.schema,a,d.mutableFields,y,d.systemKeys);if(!f.success){H(m,f.error,400);return}if(d.validate){let A=await d.validate(f.data,"update");if(A){H(m,A,400);return}}let v=await o(d,c),S=(v&&s(v,d.pathKey))??[c],O=await d.repo.update(...S,f.data);Ce(m,O);}catch(a){let f=n&&a instanceof Error?a.message:"Failed to update document";H(m,f,500);}}async function k(b,m){let y=b.params||{},i=r(y.repoName,m);if(!i)return;if(!i.allowDelete){H(m,"Delete not allowed for this repository",403);return}let d=y.id;if(!d){H(m,"Document ID required",400);return}try{let c=await o(i,d),a=(c&&s(c,i.pathKey))??[d];await i.repo.delete(...a),Ce(m,{deleted:!0});}catch(c){let a=n&&c instanceof Error?c.message:"Failed to delete document";H(m,a,500);}}function w(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:l,handleQuery:p,handleGet:u,handleCreate:g,handleUpdate:h,handleDelete:k,handleOptions:w}}function St(e){try{return zod.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 X(e){return {$ref:`#/components/schemas/${e}`}}function G(e){return {description:e,content:{"application/json":{schema:X("ErrorResponse")}}}}function qe(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 Nr(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 Ir(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 _r(){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 Dr(e,t,n,r,s){let o={},l=e.name,p=`${t}/${e.name}`,u=`${p}/{${e.documentKey}}`,g={name:e.documentKey,in:"path",required:true,schema:{type:"string"},description:"Unique document identifier"};o[p]={get:{operationId:`list${fe(e.name)}`,summary:`List ${e.name} (paginated)`,tags:[l],parameters:[...Nr(e),...Ir(e)],responses:{200:vn(X(n)),500:G("Internal server error")}},post:{operationId:`create${fe(e.name)}`,summary:`Create a ${re(e.name)}`,tags:[l],requestBody:{required:true,content:{"application/json":{schema:X(r??n)}}},responses:{201:qe("Document created",X(n)),400:G("Validation error"),500:G("Internal server error")}}},o[`${p}/query`]={post:{operationId:`query${fe(e.name)}`,summary:`Query ${e.name} with advanced filters`,tags:[l],requestBody:{required:true,content:{"application/json":{schema:X("QueryRequestBody")}}},responses:{200:vn(X(n)),400:G("Invalid query"),500:G("Internal server error")}}};let h={};return h.get={operationId:`get${fe(re(e.name))}`,summary:`Get a single ${re(e.name)}`,tags:[l],parameters:[g],responses:{200:qe("Document found",X(n)),404:G("Document not found"),500:G("Internal server error")}},h.put={operationId:`update${fe(re(e.name))}`,summary:`Update a ${re(e.name)} (full replace)`,tags:[l],parameters:[g],requestBody:{required:true,content:{"application/json":{schema:X(s??n)}}},responses:{200:qe("Document updated",X(n)),400:G("Validation error"),404:G("Document not found"),500:G("Internal server error")}},h.patch={operationId:`patch${fe(re(e.name))}`,summary:`Partially update a ${re(e.name)}`,tags:[l],parameters:[g],requestBody:{required:true,content:{"application/json":{schema:{allOf:[X(s??n)],description:"All fields are optional for partial updates"}}}},responses:{200:qe("Document patched",X(n)),400:G("Validation error"),404:G("Document not found"),500:G("Internal server error")}},e.allowDelete&&(h.delete={operationId:`delete${fe(re(e.name))}`,summary:`Delete a ${re(e.name)}`,tags:[l],parameters:[g],responses:{200:qe("Document deleted",{type:"object",properties:{id:{type:"string"}}}),404:G("Document not found"),500:G("Internal server error")}}),o[u]=h,o}function $t(e,t,n={}){let{title:r="CRUD API",version:s="1.0.0",description:o,servers:l,auth:p=false}=n,u=t==="/"?"":t.replace(/\/$/,""),g={},h={},k=[];g.ErrorResponse={type:"object",properties:{success:{type:"boolean",enum:[false]},error:{type:"string"}},required:["success","error"]},g.QueryRequestBody=_r();for(let[y,i]of Object.entries(e)){let d=fe(re(y)),c=`${d}Create`,a=`${d}Update`;g[d]=St(i.schema);let f=R=>{let $=R&&R.length>0?R:Object.keys(i.schema.shape),T={};for(let P of $){let C=P.split(".")[0];C&&i.schema.shape[C]&&!i.systemKeys.includes(C)&&(T[C]=i.schema.shape[C]);}return T},v=null,S=f(i.createFields);Object.keys(S).length>0&&(g[c]=St(zod.z.object(S)),v=c);let O=null,A=f(i.mutableFields);Object.keys(A).length>0&&(g[a]=St(zod.z.object(A)),O=a);let x=Dr(i,u,d,v,O);Object.assign(h,x),k.push({name:y,description:`Operations on ${y} (collection: ${i.path})`});}let w={},b;return p==="basic"?(w.basicAuth={type:"http",scheme:"basic"},b=[{basicAuth:[]}]):p==="bearer"&&(w.bearerAuth={type:"http",scheme:"bearer",bearerFormat:"JWT"},b=[{bearerAuth:[]}]),{openapi:"3.1.0",info:{title:r,version:s,...o?{description:o}:{}},...l&&l.length>0?{servers:l}:{},paths:h,components:{schemas:g,...Object.keys(w).length>0?{securitySchemes:w}:{}},...b?{security:b}:{},tags:k}}function fe(e){return e.charAt(0).toUpperCase()+e.slice(1)}function re(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 jr(e,t){return `<!DOCTYPE html>
|
|
646
1136
|
<html lang="en">
|
|
647
1137
|
<head>
|
|
648
1138
|
<meta charset="utf-8" />
|
|
@@ -653,5 +1143,5 @@ function initColumnReorder(table) {
|
|
|
653
1143
|
<script id="api-reference" data-url="${t}"></script>
|
|
654
1144
|
<script src="https://cdn.jsdelivr.net/npm/@scalar/api-reference"></script>
|
|
655
1145
|
</body>
|
|
656
|
-
</html>`}function
|
|
1146
|
+
</html>`}function zr(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",l=process.env.FUNCTION_REGION??"us-central1",p=process.env.FUNCTION_TARGET??"";return `/${o}/${l}/${p}${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 Zr(e){return typeof e.rawBody=="string"?e.rawBody:Buffer.isBuffer(e.rawBody)?e.rawBody.toString("utf8"):""}function xn(e){let{basePath:t="/",repos:n,parseBody:r=true,auth:s,middleware:o=[],verbose:l=false,httpsOptions:p}=e,u=t==="/"?"":t.replace(/\/$/,""),g={};for(let[d,c]of Object.entries(n)){let a=c.schema??c.repo.schema??null;if(!a)throw new Error(`[createCrudServer] Repository "${d}" has no Zod schema. Either use createRepositoryConfig(schema)(config) or pass schema: explicitly.`);let f,v,S;if(c.fieldsConfig){let x=c.fieldsConfig;f=[],v=[],S=[];for(let[R,$]of Object.entries(x))for(let T of $)T==="filterable"?f.push(R):T==="mutable"?v.push(R):T==="create"&&S.push(R);f.length===0&&(f=void 0),v.length===0&&(v=void 0),S.length===0&&(S=void 0);}let O=(()=>{let x=c.repo._parentKeys;return x&&x.length>0?x:void 0})();if(O&&S)for(let x of O)S.includes(x)||S.push(x);let A={name:d,path:c.path,repo:c.repo,schema:a,systemKeys:c.repo._systemKeys??[c.documentKey??"docId"],documentKey:c.documentKey??"docId",pathKey:c.repo._pathKey??void 0,isGroup:!!c.repo._isGroup,parentKeys:O,createdKey:c.repo._createdKey??void 0,pageSize:c.pageSize??25,filterableFields:f,mutableFields:v,createFields:S,allowDelete:c.allowDelete??false,allowedIncludes:c.allowedIncludes,validate:c.validate};g[d]=A;}let h=hn(g,u,l),k=e.openapi,w=k&&typeof k=="object"?k:{},b=null;function m(){if(!b){let d=s&&typeof s!="function"?"basic":s?"bearer":false;b=$t(g,u,{...w,auth:w.auth??d});}return b}let y=new Y;if(y.use((d,c,a)=>{c.set("Access-Control-Allow-Origin","*"),c.set("Access-Control-Allow-Credentials","true"),a();}),r&&y.use(async(d,c,a)=>{let f=d;if(String(f.headers?.["content-type"]??"").includes("application/json")){if(typeof f.body=="string")try{d.body=JSON.parse(f.body);}catch{}else if(Buffer.isBuffer(d.rawBody))try{let S=await Zr(f);d.body=JSON.parse(S);}catch{}}await a();}),s)if(typeof s=="function")y.use(s);else {let d=s.realm??"API",c="Basic "+Buffer.from(`${s.username}:${s.password}`).toString("base64");y.use((a,f,v)=>{if((a.headers?.authorization??"")!==c){f.status(401).set("WWW-Authenticate",`Basic realm="${d}"`).set("Content-Type","application/json").send(JSON.stringify({success:false,error:"Unauthorized"}));return}v();});}for(let d of o)y.use(d);if(k!==false){let d=`${u}/__spec.json`,c=`${u}/__docs`;y.get(d,(a,f)=>{let v=m();f.status(200).set("Content-Type","application/json; charset=utf-8").send(JSON.stringify(v,null,2));}),y.get(c,(a,f)=>{let v=zr(a,u)+"/__spec.json",S=jr(w.title??"CRUD API",v);f.status(200).set("Content-Type","text/html; charset=utf-8").send(S);});}y.use((d,c,a)=>{if(d.method==="OPTIONS"){h.handleOptions(d,c);return}a();}),y.get(`${u}/:repoName`,h.handleList),y.post(`${u}/:repoName/query`,h.handleQuery),y.get(`${u}/:repoName/:id`,h.handleGet),y.post(`${u}/:repoName`,h.handleCreate),y.put(`${u}/:repoName/:id`,(d,c)=>h.handleUpdate(d,c,false)),y.patch(`${u}/:repoName/:id`,(d,c)=>h.handleUpdate(d,c,true)),y.delete(`${u}/:repoName/:id`,h.handleDelete);let i=async(d,c)=>{await y.handle(d,c);};return i.spec=m,p&&(i.httpsOptions=p),i}function wn(e,t){if(!t.onRequest)return e;let n=e.httpsOptions??t.httpsOptions;return n?t.onRequest(n,e):t.onRequest(e)}function Rn(e,t,n){let r={};for(let[s,o]of Object.entries(t)){if(!o)continue;let l=e[s];if(!l)throw new Error(`[createServers.${n}] Unknown repo "${s}" \u2014 not present in the registry passed to createServers().`);r[s]={...o,repo:l};}return r}function Fr(e,t={}){return {admin(n){let r=Rn(e,n.repos,"admin"),s=ln({...n,repos:r,httpsOptions:n.httpsOptions??t.httpsOptions});return wn(s,t)},crud(n){let r=Rn(e,n.repos,"crud"),s=xn({...n,repos:r,httpsOptions:n.httpsOptions??t.httpsOptions});return wn(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}),Zt(e,r)}}}exports.CSS=dr;exports.ClientScript=_e;exports.MiniRouter=Y;exports.createServers=Fr;exports.renderDashboard=lt;exports.renderField=We;exports.renderForm=ie;exports.renderFormPage=ue;exports.renderList=ct;exports.renderPage=Ge;exports.zodToFields=ne;//# sourceMappingURL=index.cjs.map
|
|
657
1147
|
//# sourceMappingURL=index.cjs.map
|