@oneclick.dev/cms-core-modules 0.0.114 → 0.0.116

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.
Files changed (87) hide show
  1. package/dist/{ContentEditor-MctMvN7D.js → ContentEditor-CsbOFg3a.js} +52 -52
  2. package/dist/{ContentEditor-C5yNNLeV.mjs → ContentEditor-Df5uWpVC.mjs} +10337 -8677
  3. package/dist/EditLayout.vue_vue_type_script_setup_true_lang-gozJKXvM.js +1 -0
  4. package/dist/EditLayout.vue_vue_type_script_setup_true_lang-pDO9b4Pv.mjs +77 -0
  5. package/dist/NewReservationDialog.vue_vue_type_script_setup_true_lang-BHeMJ6nr.mjs +1476 -0
  6. package/dist/NewReservationDialog.vue_vue_type_script_setup_true_lang-C2zwt5UF.js +1 -0
  7. package/dist/OrderDetailDialog.vue_vue_type_script_setup_true_lang-BYSeUW_V.js +925 -0
  8. package/dist/OrderDetailDialog.vue_vue_type_script_setup_true_lang-DwAUYR8p.mjs +5021 -0
  9. package/dist/Overview-CBahJviK.js +1 -0
  10. package/dist/Overview-CVQ1pkuD.mjs +527 -0
  11. package/dist/TableView-Csv5Lycy.mjs +6234 -0
  12. package/dist/TableView-Cwal0BPW.js +4 -0
  13. package/dist/agenda-BSdlrfxv.mjs +1253 -0
  14. package/dist/agenda-Bev1mO7E.js +1 -0
  15. package/dist/availability-ClBGVgE9.js +1 -0
  16. package/dist/availability-D8JdA4rP.mjs +274 -0
  17. package/dist/booking-data-Px7XCIfU.mjs +1024 -0
  18. package/dist/booking-data-aMS1p_3g.js +1 -0
  19. package/dist/cms-core-modules.css +1 -1
  20. package/dist/exceptions-CqityDo9.mjs +651 -0
  21. package/dist/exceptions-DXqc0Nza.js +1 -0
  22. package/dist/index-CM4eaK5T.mjs +1245 -0
  23. package/dist/index-DliTZzwI.js +35 -0
  24. package/dist/index.cjs.js +1 -1
  25. package/dist/index.mjs +12 -11
  26. package/dist/orders-1swJVKw2.js +1 -0
  27. package/dist/orders-D41GbzIa.mjs +624 -0
  28. package/dist/payment-BJHgpaeT.js +1 -0
  29. package/dist/payment-D5j-68Ig.mjs +1278 -0
  30. package/dist/refunds-D9nTeD2d.mjs +436 -0
  31. package/dist/refunds-oVB2Opib.js +1 -0
  32. package/dist/resources-8WouFvJe.js +1 -0
  33. package/dist/resources-B-D5MUhV.mjs +975 -0
  34. package/dist/server-handlers.cjs.js +1 -1
  35. package/dist/server-handlers.mjs +626 -515
  36. package/dist/src/appointments/components/edit/EventDialog/BookingsList.vue.d.ts +146 -5
  37. package/dist/src/appointments/components/edit/EventDialog/CancelRefundReservationDialog.vue.d.ts +33 -0
  38. package/dist/src/appointments/components/edit/EventDialog/OrderDetailDialog.vue.d.ts +26 -8
  39. package/dist/src/appointments/components/edit/EventDialog/RefundDetailsDisplay.vue.d.ts +8 -0
  40. package/dist/src/appointments/components/edit/EventDialog/ReservationDetailDialog.vue.d.ts +91 -5
  41. package/dist/src/appointments/components/edit/EventDialog/TransferReservationDialog.vue.d.ts +20 -3
  42. package/dist/src/appointments/components/edit/NewReservationDialog/ReservationLines.vue.d.ts +5 -0
  43. package/dist/src/appointments/components/edit/OrderMetadataDisplay.vue.d.ts +20 -2
  44. package/dist/src/appointments/components/edit/dashboard/BookingsList.vue.d.ts +240 -0
  45. package/dist/src/appointments/composables/useAgendaMetadataSchema.d.ts +37 -0
  46. package/dist/src/appointments/pages/edit/orders.vue.d.ts +225 -0
  47. package/dist/src/appointments/pages/edit/refunds.vue.d.ts +2 -0
  48. package/dist/src/appointments/server.d.ts +2 -0
  49. package/dist/src/appointments/types.d.ts +6 -0
  50. package/dist/src/appointments/utils/printReservation.d.ts +65 -3
  51. package/dist/src/contentManager/components/content-editor/tiptap-extensions/table/Table.d.ts +36 -0
  52. package/dist/src/contentManager/components/content-editor/tiptap-extensions/table/Table.vue.d.ts +95 -0
  53. package/dist/src/contentManager/components/content-editor/tiptap-extensions/table/TableCell.d.ts +13 -0
  54. package/dist/src/contentManager/components/content-editor/tiptap-extensions/table/TableHeader.d.ts +6 -0
  55. package/dist/src/contentManager/components/content-editor/tiptap-extensions/table/TableRow.d.ts +6 -0
  56. package/dist/src/contentManager/components/content-editor/tiptap-extensions/table/index.d.ts +4 -0
  57. package/dist/src/contentManager/components/content-editor/tiptap-menus/element-editor-views/TableMenu.vue.d.ts +13 -0
  58. package/dist/src/contentManager/components/content-editor/tiptap-menus/element-editor-views/index.d.ts +14 -0
  59. package/package.json +2 -2
  60. package/dist/EditLayout.vue_vue_type_script_setup_true_lang-DWMqQvHl.mjs +0 -76
  61. package/dist/EditLayout.vue_vue_type_script_setup_true_lang-kpjbVSXg.js +0 -1
  62. package/dist/NewReservationDialog.vue_vue_type_script_setup_true_lang-Baqy-rTT.js +0 -1
  63. package/dist/NewReservationDialog.vue_vue_type_script_setup_true_lang-Dx4Bpa2m.mjs +0 -1263
  64. package/dist/OrderDetailDialog.vue_vue_type_script_setup_true_lang-COrK1j0S.js +0 -1
  65. package/dist/OrderDetailDialog.vue_vue_type_script_setup_true_lang-Vb3q8EVv.mjs +0 -330
  66. package/dist/Overview-98nkJUWN.mjs +0 -481
  67. package/dist/Overview-Dl8cMlsr.js +0 -1
  68. package/dist/ReservationDetailDialog.vue_vue_type_script_setup_true_lang-CuwREvXD.js +0 -349
  69. package/dist/ReservationDetailDialog.vue_vue_type_script_setup_true_lang-GYNZ_yhD.mjs +0 -3077
  70. package/dist/TableView-CVfkyj1k.js +0 -4
  71. package/dist/TableView-zDx0IegJ.mjs +0 -6096
  72. package/dist/agenda-BaJu3-1c.js +0 -1
  73. package/dist/agenda-BwVY_8oM.mjs +0 -1165
  74. package/dist/availability-CMrRa5y2.mjs +0 -269
  75. package/dist/availability-Cf2YfMwM.js +0 -1
  76. package/dist/booking-data-DgJd0BcM.mjs +0 -889
  77. package/dist/booking-data-Di5GmH_8.js +0 -1
  78. package/dist/exceptions-B6P9UiCj.js +0 -1
  79. package/dist/exceptions-De9-FvdP.mjs +0 -646
  80. package/dist/index-DL6orwdK.js +0 -35
  81. package/dist/index-hH3e-IYz.mjs +0 -1187
  82. package/dist/orders-C65SlpJy.mjs +0 -618
  83. package/dist/orders-XVzWAgG1.js +0 -1
  84. package/dist/payment-C3ohkehF.mjs +0 -1080
  85. package/dist/payment-Dfr-Ro-a.js +0 -1
  86. package/dist/resources-CxeFd57z.js +0 -1
  87. package/dist/resources-DwYxn2Vi.mjs +0 -811
@@ -1 +1 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const n=require("h3");require("vue");function b(w){const{initFirebase:A,normalizeTimestamps:D}=w,p=n.createRouter();async function m(l){const{supabase:c,instanceId:u}=l.context.module,{data:e,error:o}=await c.from("project_modules").select("config").eq("id",u).single();if(o||!e?.config)throw n.createError({statusCode:500,statusMessage:"Failed to load module config."});const s=e.config,t=s.project,a=s.productCollection||"products";if(!t)throw n.createError({statusCode:400,statusMessage:"Products module has no Firebase integration configured."});return{firebase:await A(l,t),collection:a}}return p.get("/products",n.defineEventHandler(async l=>{try{const{firebase:c,collection:u}=await m(l);return(await c.firestore().collection(u).get()).docs.map(o=>D({id:o.id,...o.data()}))}catch(c){throw console.error("Products handler — list error:",c),n.createError({statusCode:c.statusCode||500,statusMessage:c.statusMessage||"Failed to list products"})}})),p.get("/products/:productId",n.defineEventHandler(async l=>{const c=n.getRouterParam(l,"productId");if(!c)throw n.createError({statusCode:400,statusMessage:"Product ID is required."});try{const{firebase:u,collection:e}=await m(l),o=await u.firestore().collection(e).doc(c).get();if(!o.exists)throw n.createError({statusCode:404,statusMessage:"Product not found."});return D({id:o.id,...o.data()})}catch(u){throw console.error("Products handler — get error:",u),n.createError({statusCode:u.statusCode||500,statusMessage:u.statusMessage||"Failed to get product"})}})),p.get("/empty-stock",n.defineEventHandler(async l=>{const u=n.getQuery(l).category;try{const{firebase:e,collection:o}=await m(l);let s=e.firestore().collection(o).where("stock","==",0);u&&(s=s.where("collections","array-contains",u));const a=(await s.get()).docs.map(r=>D({id:r.id,...r.data()}));return{count:a.length,products:a.map(r=>({id:r.id,title:r.title,slug:r.slug,stock:r.stock,price:r.price,currency:r.currency,status:r.status}))}}catch(e){throw console.error("Products handler — empty-stock error:",e),n.createError({statusCode:e.statusCode||500,statusMessage:e.statusMessage||"Failed to query empty stock"})}})),p.handler}function T(w){const{initFirebase:A,normalizeTimestamps:D}=w,p=n.createRouter();async function m(c){const{supabase:u,instanceId:e}=c.context.module,{data:o,error:s}=await u.from("project_modules").select("config").eq("id",e).single();if(s||!o?.config)throw n.createError({statusCode:500,statusMessage:"Failed to load module config."});const t=o.config,a=t.project,r=t.reservationsCollection||"bookings_orders",i=t.agendaCollection||"agendas";if(!a)throw n.createError({statusCode:400,statusMessage:"Appointments module has no Firebase integration configured."});return{firebase:await A(c,a),reservationsCollection:r,agendaCollection:i}}function l(c){return(c.reservations||[]).map(e=>({orderId:c.id,reservationId:e.id,customerInfo:c.customerInfo,date:e.date,startTime:e.timeslot?.startTime,endTime:e.timeslot?.endTime,spots:e.spots,status:c.status,reservationStatus:e.status,resourceId:e.resourceId,reservationPrice:e.totalPrice,reservationBasePrice:e.basePrice,reservationAddOnsPrice:e.addOnsPrice,pricingOption:e.pricingOption,amountDue:c.amountDue,amountPaid:c.amountPaid,createdAt:c.createdAt,metadata:c.metadata||{},reservationMetadata:e.metadata||{}}))}return p.get("/agendas",n.defineEventHandler(async c=>{try{const{firebase:u,agendaCollection:e}=await m(c),s=(await u.firestore().collection(e).get()).docs.map(t=>{const a=t.data();return{id:t.id,serviceName:a.serviceName||"",type:a.type||"regular"}});return{count:s.length,agendas:s}}catch(u){throw console.error("Appointments handler — list agendas error:",u),n.createError({statusCode:u.statusCode||500,statusMessage:u.statusMessage||"Failed to list agendas"})}})),p.get("/agendas/:agendaId/opening-hours",n.defineEventHandler(async c=>{const u=n.getRouterParam(c,"agendaId"),o=n.getQuery(c).date;if(!u)throw n.createError({statusCode:400,statusMessage:"Agenda ID is required."});if(!o)throw n.createError({statusCode:400,statusMessage:"Date is required (YYYY-MM-DD)."});try{const{firebase:s,agendaCollection:t}=await m(c),a=await s.firestore().collection(t).doc(u).get();if(!a.exists)throw n.createError({statusCode:404,statusMessage:"Agenda not found."});const r=a.data(),i=(r.resources||[]).filter(f=>f.isActive),d=r.exceptions||[],y=new Date(o+"T00:00:00").getDay(),h=i.map(f=>{const E=f.openingHours?.[y]||[],C=d.find(I=>{const P=o>=I.startDate&&o<=I.endDate,R=!I.resourceIds||I.resourceIds.length===0||I.resourceIds.includes(f.id);return P&&R});return C?{resourceId:f.id,resourceName:f.name,date:o,dayOfWeek:y,isClosed:C.isClosed,hours:C.isClosed?[]:(C.timeslots||[]).map(I=>({start:I.startTime,end:I.endTime})),isException:!0}:{resourceId:f.id,resourceName:f.name,date:o,dayOfWeek:y,isClosed:E.length===0,hours:E,isException:!1}});return{agendaId:u,serviceName:r.serviceName||"",date:o,resources:h}}catch(s){throw console.error("Appointments handler — opening hours error:",s),n.createError({statusCode:s.statusCode||500,statusMessage:s.statusMessage||"Failed to get opening hours"})}})),p.get("/appointments",n.defineEventHandler(async c=>{const u=n.getQuery(c),e=Math.min(Number(u.quantity)||20,100),o=u.status||"confirmed";try{const{firebase:s,reservationsCollection:t}=await m(c),r=(await s.firestore().collection(t).orderBy("createdAt","desc").where("status","==",o).limit(e).get()).docs.flatMap(i=>l(D({id:i.id,...i.data()})));return{count:r.length,appointments:r}}catch(s){throw console.error("Appointments handler — list error:",s),n.createError({statusCode:s.statusCode||500,statusMessage:s.statusMessage||"Failed to list appointments"})}})),p.get("/appointments/find",n.defineEventHandler(async c=>{const u=n.getQuery(c),e=(u.name||"").toLowerCase().trim(),o=(u.email||"").toLowerCase().trim(),s=u.date;try{const{firebase:t,reservationsCollection:a}=await m(c);let r=t.firestore().collection(a);o&&(r=r.where("customerInfo.email","==",o));let d=(await r.orderBy("createdAt","desc").limit(200).get()).docs.flatMap(g=>l(D({id:g.id,...g.data()})));return s&&(d=d.filter(g=>g.date===s)),e&&(d=d.filter(g=>{const y=(g.customerInfo?.firstName||"").toLowerCase(),h=(g.customerInfo?.lastName||"").toLowerCase();return y.includes(e)||h.includes(e)||`${y} ${h}`.includes(e)})),{count:d.length,appointments:d}}catch(t){throw console.error("Appointments handler — find error:",t),n.createError({statusCode:t.statusCode||500,statusMessage:t.statusMessage||"Failed to find appointments"})}})),p.get("/appointments/:appointmentId",n.defineEventHandler(async c=>{const u=n.getRouterParam(c,"appointmentId");if(!u)throw n.createError({statusCode:400,statusMessage:"Appointment ID is required."});try{const{firebase:e,reservationsCollection:o}=await m(c),s=await e.firestore().collection(o).doc(u).get();if(!s.exists)throw n.createError({statusCode:404,statusMessage:"Appointment not found."});const t=D({id:s.id,...s.data()});return{...t,reservations:l(t)}}catch(e){throw console.error("Appointments handler — get error:",e),n.createError({statusCode:e.statusCode||500,statusMessage:e.statusMessage||"Failed to get appointment"})}})),p.handler}const S="https://analyticsdata.googleapis.com/v1beta",N=["https://www.googleapis.com/auth/analytics.readonly","https://www.googleapis.com/auth/webmasters.readonly"],k="https://searchconsole.googleapis.com/webmasters/v3";function H(w){const{decrypt:A,getGoogleAccessToken:D}=w,p=n.createRouter();async function m(e){const{supabase:o,instanceId:s}=e.context.module,{data:t,error:a}=await o.from("project_modules").select("config").eq("id",s).single();if(a||!t?.config)throw n.createError({statusCode:500,statusMessage:"Failed to load module config."});const r=t.config,i=r.propertyId,d=r.serviceAccount,g=r.siteUrl||"";if(!d)throw n.createError({statusCode:400,statusMessage:"No Google Service Account configured for this module."});if(!i)throw n.createError({statusCode:400,statusMessage:"No GA4 Property ID configured for this module."});const{data:y,error:h}=await o.from("integrations").select("config").eq("id",d).single();let f=y?.config;if(h||!f){const{data:R,error:q}=await o.from("agency_integrations").select("config").eq("id",d).single();if(q||!R?.config)throw n.createError({statusCode:500,statusMessage:"Failed to load Google Service Account credentials."});f=R.config}const E=A(f.clientEmail),C=A(f.privateKey),I=A(f.projectId);return{accessToken:await D({clientEmail:E,privateKey:C,projectId:I},N),propertyId:i,siteUrl:g}}async function l(e,o,s){const t=await fetch(`${S}/properties/${o}:runReport`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify(s)});if(!t.ok){const a=await t.json().catch(()=>({}));throw console.error("GA4 runReport failed:",a),n.createError({statusCode:t.status,statusMessage:a?.error?.message||"GA4 API request failed"})}return t.json()}p.get("/report",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=a.match(/^(\d+)daysAgo$/),d=i?parseInt(i[1],10):30,g=`${d*2}daysAgo`,y=`${d+1}daysAgo`,h=[{name:"sessions"},{name:"totalUsers"},{name:"screenPageViews"},{name:"bounceRate"},{name:"averageSessionDuration"},{name:"newUsers"},{name:"engagementRate"},{name:"sessionsPerUser"},{name:"screenPageViewsPerSession"}];let f,E=!0;try{f=await l(o,s,{dateRanges:[{startDate:a,endDate:r,name:"current"},{startDate:g,endDate:y,name:"previous"}],dimensions:[{name:"date"}],metrics:h,metricAggregations:["TOTAL"],orderBys:[{dimension:{dimensionName:"date"}}]})}catch{E=!1,f=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"date"}],metrics:h,metricAggregations:["TOTAL"],orderBys:[{dimension:{dimensionName:"date"}}]})}return U(f,E)})),p.get("/realtime",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=await fetch(`${S}/properties/${s}:runRealtimeReport`,{method:"POST",headers:{Authorization:`Bearer ${o}`,"Content-Type":"application/json"},body:JSON.stringify({dimensions:[{name:"unifiedScreenName"}],metrics:[{name:"activeUsers"}],orderBys:[{metric:{metricName:"activeUsers"},desc:!0}],limit:5})});if(!t.ok){const d=await t.json().catch(()=>({}));throw n.createError({statusCode:t.status,statusMessage:d?.error?.message||"Realtime API failed"})}const a=await t.json(),r=(a?.rows||[]).reduce((d,g)=>d+parseInt(g.metricValues?.[0]?.value||"0",10),0),i=(a?.rows||[]).map(d=>({page:d.dimensionValues?.[0]?.value||"(not set)",activeUsers:parseInt(d.metricValues?.[0]?.value||"0",10)}));return{activeUsers:r,activePages:i}})),p.get("/top-pages",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=parseInt(t.limit||"10",10),d=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"pagePath"}],metrics:[{name:"screenPageViews"},{name:"totalUsers"},{name:"averageSessionDuration"}],orderBys:[{metric:{metricName:"screenPageViews"},desc:!0}],limit:i*2}),g=v(d,"pagePath"),y=new Map;for(const f of g.rows){const E=f.pagePath.toLowerCase().replace(/\/+$/,""),C=y.get(E);if(C){C.screenPageViews+=f.screenPageViews||0,C.totalUsers+=f.totalUsers||0;const I=C.screenPageViews;I>0&&(C.averageSessionDuration=(C.averageSessionDuration*(I-(f.screenPageViews||0))+(f.averageSessionDuration||0)*(f.screenPageViews||0))/I)}else y.set(E,{...f})}return{rows:[...y.values()].sort((f,E)=>(E.screenPageViews||0)-(f.screenPageViews||0)).slice(0,i),rowCount:g.rowCount}})),p.get("/top-sources",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"sessionSource"}],metrics:[{name:"sessions"},{name:"totalUsers"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:10});return v(i,"sessionSource")})),p.get("/devices",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"deviceCategory"}],metrics:[{name:"sessions"},{name:"totalUsers"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}]});return v(i,"deviceCategory")})),p.get("/countries",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"country"}],metrics:[{name:"sessions"},{name:"totalUsers"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:10});return v(i,"country")})),p.get("/acquisition/channels",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"sessionDefaultChannelGroup"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"newUsers"},{name:"engagementRate"},{name:"averageSessionDuration"},{name:"screenPageViewsPerSession"},{name:"conversions"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:15});return v(i,"sessionDefaultChannelGroup")})),p.get("/acquisition/source-medium",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"sessionSourceMedium"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"newUsers"},{name:"bounceRate"},{name:"averageSessionDuration"},{name:"conversions"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:20});return v(i,"sessionSourceMedium")})),p.get("/acquisition/referrals",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"sessionSource"}],dimensionFilter:{filter:{fieldName:"sessionMedium",stringFilter:{value:"referral"}}},metrics:[{name:"sessions"},{name:"totalUsers"},{name:"engagementRate"},{name:"averageSessionDuration"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:20});return v(i,"sessionSource")})),p.get("/acquisition/campaigns",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"sessionCampaignName"}],dimensionFilter:{notExpression:{filter:{fieldName:"sessionCampaignName",stringFilter:{value:"(not set)"}}}},metrics:[{name:"sessions"},{name:"totalUsers"},{name:"conversions"},{name:"engagementRate"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:20});return v(i,"sessionCampaignName")})),p.get("/content/all-pages",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=parseInt(t.limit||"50",10),d=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"pagePath"}],metrics:[{name:"screenPageViews"},{name:"totalUsers"},{name:"averageSessionDuration"},{name:"bounceRate"},{name:"engagementRate"},{name:"sessions"},{name:"userEngagementDuration"}],orderBys:[{metric:{metricName:"screenPageViews"},desc:!0}],limit:i});return v(d,"pagePath")})),p.get("/content/landing-pages",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"landingPagePlusQueryString"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"bounceRate"},{name:"averageSessionDuration"},{name:"screenPageViews"},{name:"engagementRate"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:30});return v(i,"landingPagePlusQueryString")})),p.get("/content/exit-pages",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"pagePath"}],metrics:[{name:"sessions"},{name:"screenPageViews"},{name:"totalUsers"},{name:"bounceRate"}],orderBys:[{metric:{metricName:"screenPageViews"},desc:!0}],limit:20}),d=v(i,"pagePath");return d.rows=d.rows.map(g=>({...g,exitRate:g.bounceRate||0})),d})),p.get("/content/search-terms",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s,siteUrl:t}=await m(e),a=n.getQuery(e),r=a.startDate||"30daysAgo",i=a.endDate||"today";if(t)try{const y=`${k}/sites/${encodeURIComponent(t)}/searchAnalytics/query`,h=await fetch(y,{method:"POST",headers:{Authorization:`Bearer ${o}`,"Content-Type":"application/json"},body:JSON.stringify({startDate:c(r),endDate:c(i),dimensions:["query"],rowLimit:30})});if(h.ok){const E=((await h.json()).rows||[]).map(C=>({query:C.keys[0],clicks:C.clicks,impressions:C.impressions,ctr:C.ctr,position:C.position}));if(E.length>0)return{rows:E,rowCount:E.length,source:"search_console"}}else{const f=await h.text().catch(()=>"");console.error(`[GA Module] Search Console API error (${h.status}):`,f)}}catch(y){console.error("[GA Module] Search Console request failed:",y?.message||y)}else console.log("[GA Module] No siteUrl configured, skipping Search Console");try{const y=await l(o,s,{dateRanges:[{startDate:r,endDate:i}],dimensions:[{name:"sessionGoogleAdsQuery"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"engagementRate"}],dimensionFilter:{andGroup:{expressions:[{notExpression:{filter:{fieldName:"sessionGoogleAdsQuery",stringFilter:{value:"(not set)"}}}},{notExpression:{filter:{fieldName:"sessionGoogleAdsQuery",stringFilter:{value:"(not provided)"}}}}]}},orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:30}),h=v(y,"sessionGoogleAdsQuery");if(h.rows.length>0)return{...h,source:"google_ads"}}catch{}const d=await l(o,s,{dateRanges:[{startDate:r,endDate:i}],dimensions:[{name:"landingPagePlusQueryString"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"engagementRate"}],dimensionFilter:{andGroup:{expressions:[{filter:{fieldName:"sessionMedium",stringFilter:{matchType:"EXACT",value:"organic"}}},{notExpression:{filter:{fieldName:"landingPagePlusQueryString",stringFilter:{value:"(not set)"}}}}]}},orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:30});return{...v(d,"landingPagePlusQueryString"),source:"organic_landing_pages"}})),p.get("/audience/overview",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"newVsReturning"}],metrics:[{name:"totalUsers"},{name:"sessions"},{name:"engagementRate"},{name:"averageSessionDuration"},{name:"screenPageViewsPerSession"}],metricAggregations:["TOTAL"]});return v(i,"newVsReturning")})),p.get("/audience/technology",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=t.dimension||"browser",g=["browser","operatingSystem","screenResolution"].includes(i)?i:"browser",y=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:g}],metrics:[{name:"totalUsers"},{name:"sessions"},{name:"engagementRate"}],orderBys:[{metric:{metricName:"totalUsers"},desc:!0}],limit:10});return v(y,g)})),p.get("/audience/languages",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"language"}],metrics:[{name:"totalUsers"},{name:"sessions"}],orderBys:[{metric:{metricName:"totalUsers"},desc:!0}],limit:15});return v(i,"language")})),p.get("/audience/hours",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"dayOfWeekName"},{name:"hour"}],metrics:[{name:"sessions"}],orderBys:[{dimension:{dimensionName:"dayOfWeekName"}},{dimension:{dimensionName:"hour"}}],limit:168});return M(i,["dayOfWeekName","hour"])})),p.get("/audience/cities",n.defineEventHandler(async e=>{const{accessToken:o,propertyId:s}=await m(e),t=n.getQuery(e),a=t.startDate||"30daysAgo",r=t.endDate||"today",i=await l(o,s,{dateRanges:[{startDate:a,endDate:r}],dimensions:[{name:"city"},{name:"country"}],metrics:[{name:"totalUsers"},{name:"sessions"}],dimensionFilter:{notExpression:{filter:{fieldName:"city",stringFilter:{value:"(not set)"}}}},orderBys:[{metric:{metricName:"totalUsers"},desc:!0}],limit:20});return M(i,["city","country"])}));function c(e){const o=e.match(/^(\d+)daysAgo$/);if(o){const s=new Date;return s.setDate(s.getDate()-parseInt(o[1],10)),s.toISOString().slice(0,10)}if(e==="today")return new Date().toISOString().slice(0,10);if(e==="yesterday"){const s=new Date;return s.setDate(s.getDate()-1),s.toISOString().slice(0,10)}return e}async function u(e,o,s){const t=`${k}/sites/${encodeURIComponent(o)}/searchAnalytics/query`,a=await fetch(t,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify(s)});if(!a.ok){const r=await a.json().catch(()=>({}));throw n.createError({statusCode:a.status,statusMessage:r?.error?.message||"Search Console API request failed"})}return a.json()}return p.get("/seo/keywords",n.defineEventHandler(async e=>{const{accessToken:o,siteUrl:s}=await m(e);if(!s)throw n.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured. Add it in module settings."});const t=n.getQuery(e),a=c(t.startDate||"30daysAgo"),r=c(t.endDate||"today"),i=Math.min(parseInt(t.limit)||50,100),d=await u(o,s,{startDate:a,endDate:r,dimensions:["query"],rowLimit:i});return{rows:(d.rows||[]).map(g=>({query:g.keys[0],clicks:g.clicks,impressions:g.impressions,ctr:g.ctr,position:g.position})),rowCount:d.rows?.length||0}})),p.get("/seo/pages",n.defineEventHandler(async e=>{const{accessToken:o,siteUrl:s}=await m(e);if(!s)throw n.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured."});const t=n.getQuery(e),a=c(t.startDate||"30daysAgo"),r=c(t.endDate||"today"),i=Math.min(parseInt(t.limit)||50,100),d=await u(o,s,{startDate:a,endDate:r,dimensions:["page"],rowLimit:i});return{rows:(d.rows||[]).map(g=>({page:g.keys[0],clicks:g.clicks,impressions:g.impressions,ctr:g.ctr,position:g.position})),rowCount:d.rows?.length||0}})),p.get("/seo/trends",n.defineEventHandler(async e=>{const{accessToken:o,siteUrl:s}=await m(e);if(!s)throw n.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured."});const t=n.getQuery(e),a=c(t.startDate||"30daysAgo"),r=c(t.endDate||"today"),d=((await u(o,s,{startDate:a,endDate:r,dimensions:["date"],rowLimit:500})).rows||[]).map(y=>({date:y.keys[0],clicks:y.clicks,impressions:y.impressions,ctr:y.ctr,position:y.position})).sort((y,h)=>y.date.localeCompare(h.date)),g=d.reduce((y,h)=>({clicks:y.clicks+h.clicks,impressions:y.impressions+h.impressions}),{clicks:0,impressions:0});return g.ctr=g.impressions>0?g.clicks/g.impressions:0,g.avgPosition=d.length>0?d.reduce((y,h)=>y+h.position,0)/d.length:0,{rows:d,totals:g,rowCount:d.length}})),p.get("/seo/query-pages",n.defineEventHandler(async e=>{const{accessToken:o,siteUrl:s}=await m(e);if(!s)throw n.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured."});const t=n.getQuery(e),a=c(t.startDate||"30daysAgo"),r=c(t.endDate||"today"),i=await u(o,s,{startDate:a,endDate:r,dimensions:["query","page"],rowLimit:100});return{rows:(i.rows||[]).map(d=>({query:d.keys[0],page:d.keys[1],clicks:d.clicks,impressions:d.impressions,ctr:d.ctr,position:d.position})),rowCount:i.rows?.length||0}})),p.handler}function U(w,A=!0){const D=(w.metricHeaders||[]).map(a=>a.name),m=(w.dimensionHeaders||[]).map(a=>a.name).indexOf("date"),l=[];(w.rows||[]).forEach(a=>{const r=m>=0?a.dimensionValues[m]?.value:a.dimensionValues[0]?.value;if(!r||r.length<8)return;const d={date:`${r.slice(0,4)}-${r.slice(4,6)}-${r.slice(6,8)}`};D.forEach((g,y)=>{d[g]=parseFloat(a.metricValues[y]?.value||"0")}),l.push(d)});const c={},u={};w.totals&&w.totals.length>=2?D.forEach((a,r)=>{c[a]=parseFloat(w.totals[0]?.metricValues?.[r]?.value||"0"),u[a]=parseFloat(w.totals[1]?.metricValues?.[r]?.value||"0")}):w.totals&&w.totals.length===1&&D.forEach((a,r)=>{c[a]=parseFloat(w.totals[0]?.metricValues?.[r]?.value||"0")});const e={};D.forEach(a=>{const r=c[a]||0,i=u[a];i!==void 0&&i!==0?e[a]=(r-i)/i*100:e[a]=null});const o=new Map;for(const a of l)o.has(a.date)||o.set(a.date,a);const s=Array.from(o.values()).sort((a,r)=>a.date.localeCompare(r.date)),t=A&&w.totals&&w.totals.length>=2?s.slice(-Math.ceil(s.length/2)):s;return{rows:t,totals:c,previousTotals:u,changes:e,rowCount:t.length}}function v(w,A){const D=(w.metricHeaders||[]).map(m=>m.name),p=(w.rows||[]).map(m=>{const l={[A]:m.dimensionValues[0].value};return D.forEach((c,u)=>{l[c]=parseFloat(m.metricValues[u].value)}),l});return{rows:p,rowCount:w.rowCount||p.length}}function M(w,A){const D=(w.metricHeaders||[]).map(m=>m.name),p=(w.rows||[]).map(m=>{const l={};return A.forEach((c,u)=>{l[c]=m.dimensionValues[u]?.value||""}),D.forEach((c,u)=>{l[c]=parseFloat(m.metricValues[u].value)}),l});return{rows:p,rowCount:w.rowCount||p.length}}function F(w){const{decrypt:A}=w,D=n.createRouter();async function p(m){const{supabase:l,instanceId:c}=m.context.module,{data:u,error:e}=await l.from("project_modules").select("config").eq("id",c).single();if(e||!u?.config)throw n.createError({statusCode:500,statusMessage:"Failed to load module config."});const o=u.config,s=o.githubIntegration,t=o.githubRepo,a=o.githubOwner;if(!s||!t||!a)throw n.createError({statusCode:400,statusMessage:"No GitHub integration configured for this module."});const{data:r,error:i}=await l.from("integrations").select("config").eq("id",s).single();let d=r?.config;if(i||!d){const{data:y,error:h}=await l.from("agency_integrations").select("config").eq("id",s).single();if(h||!y?.config)throw n.createError({statusCode:500,statusMessage:"Failed to load Github credentials."});d=y.config}return{token:A(d.token),repo:t,owner:a}}return D.post("/github/deploy",n.defineEventHandler(async m=>{try{const{token:l,repo:c,owner:u}=await p(m);await fetch(`https://api.github.com/repos/${u}/${c}/dispatches`,{method:"POST",headers:{Authorization:`Bearer ${l}`,Accept:"application/vnd.github+json"},body:JSON.stringify({event_type:"deploy-site"})})}catch(l){throw console.error("Error triggering GitHub deployment:",l),n.createError({statusCode:500,statusMessage:"Failed to trigger deployment."})}})),D.handler}exports.appointments=T;exports.contentManager=F;exports.googleAnalytics=H;exports.products=b;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("h3");require("vue");function O(C){const{initFirebase:R,normalizeTimestamps:A}=C,p=e.createRouter();async function l(g){const{supabase:y,instanceId:h}=g.context.module,{data:s,error:d}=await y.from("project_modules").select("config").eq("id",h).single();if(d||!s?.config)throw e.createError({statusCode:500,statusMessage:"Failed to load module config."});const c=s.config,a=c.project,n=c.productCollection||"products";if(!a)throw e.createError({statusCode:400,statusMessage:"Products module has no Firebase integration configured."});return{firebase:await R(g,a),collection:n}}return p.get("/products",e.defineEventHandler(async g=>{try{const{firebase:y,collection:h}=await l(g);return(await y.firestore().collection(h).get()).docs.map(d=>A({id:d.id,...d.data()}))}catch(y){throw console.error("Products handler — list error:",y),e.createError({statusCode:y.statusCode||500,statusMessage:y.statusMessage||"Failed to list products"})}})),p.get("/products/:productId",e.defineEventHandler(async g=>{const y=e.getRouterParam(g,"productId");if(!y)throw e.createError({statusCode:400,statusMessage:"Product ID is required."});try{const{firebase:h,collection:s}=await l(g),d=await h.firestore().collection(s).doc(y).get();if(!d.exists)throw e.createError({statusCode:404,statusMessage:"Product not found."});return A({id:d.id,...d.data()})}catch(h){throw console.error("Products handler — get error:",h),e.createError({statusCode:h.statusCode||500,statusMessage:h.statusMessage||"Failed to get product"})}})),p.get("/empty-stock",e.defineEventHandler(async g=>{const h=e.getQuery(g).category;try{const{firebase:s,collection:d}=await l(g);let c=s.firestore().collection(d).where("stock","==",0);h&&(c=c.where("collections","array-contains",h));const n=(await c.get()).docs.map(i=>A({id:i.id,...i.data()}));return{count:n.length,products:n.map(i=>({id:i.id,title:i.title,slug:i.slug,stock:i.stock,price:i.price,currency:i.currency,status:i.status}))}}catch(s){throw console.error("Products handler — empty-stock error:",s),e.createError({statusCode:s.statusCode||500,statusMessage:s.statusMessage||"Failed to query empty stock"})}})),p.handler}function V(C){const{initFirebase:R,normalizeTimestamps:A,deleteField:p}=C,l=e.createRouter(),g="privateSettings",y="refundFlow",h="metadataSchema";async function s(t){const{supabase:r,instanceId:o}=t.context.module,{data:u,error:f}=await r.from("project_modules").select("config").eq("id",o).single();if(f||!u?.config)throw e.createError({statusCode:500,statusMessage:"Failed to load module config."});const m=u.config,w=m.project,D=m.reservationsCollection||"bookings_orders",E=m.agendaCollection||"agendas";if(!w)throw e.createError({statusCode:400,statusMessage:"Appointments module has no Firebase integration configured."});return{firebase:await R(t,w),reservationsCollection:D,agendaCollection:E}}function d(t){return(t.reservations||[]).map(o=>({orderId:t.id,reservationId:o.id,customerInfo:t.customerInfo,date:o.date,startTime:o.timeslot?.startTime,endTime:o.timeslot?.endTime,spots:o.spots,status:t.status,reservationStatus:o.status,resourceId:o.resourceId,reservationPrice:o.totalPrice,reservationBasePrice:o.basePrice,reservationAddOnsPrice:o.addOnsPrice,pricingOption:o.pricingOption,amountDue:t.amountDue,amountPaid:t.amountPaid,createdAt:t.createdAt,metadata:t.metadata||{},reservationMetadata:o.metadata||{}}))}function c(t){return{showAmount:t?.showAmount!==!1,fields:Array.isArray(t?.fields)?t.fields:[]}}function a(){return p?{refundFlow:p(),refundFlowEnabled:p(),refundDialog:p()}:{refundFlow:null,refundFlowEnabled:null,refundDialog:null}}function n(t){return Array.isArray(t)?t:[]}function i(){return p?{metadataSchema:p()}:{metadataSchema:null}}return l.get("/agendas",e.defineEventHandler(async t=>{try{const{firebase:r,agendaCollection:o}=await s(t),f=(await r.firestore().collection(o).get()).docs.map(m=>{const w=m.data();return{id:m.id,serviceName:w.serviceName||"",type:w.type||"regular"}});return{count:f.length,agendas:f}}catch(r){throw console.error("Appointments handler — list agendas error:",r),e.createError({statusCode:r.statusCode||500,statusMessage:r.statusMessage||"Failed to list agendas"})}})),l.get("/agendas/:agendaId/opening-hours",e.defineEventHandler(async t=>{const r=e.getRouterParam(t,"agendaId"),u=e.getQuery(t).date;if(!r)throw e.createError({statusCode:400,statusMessage:"Agenda ID is required."});if(!u)throw e.createError({statusCode:400,statusMessage:"Date is required (YYYY-MM-DD)."});try{const{firebase:f,agendaCollection:m}=await s(t),w=await f.firestore().collection(m).doc(r).get();if(!w.exists)throw e.createError({statusCode:404,statusMessage:"Agenda not found."});const D=w.data(),E=(D.resources||[]).filter(P=>P.isActive),I=D.exceptions||[],M=new Date(u+"T00:00:00").getDay(),k=E.map(P=>{const T=P.openingHours?.[M]||[],q=I.find(b=>{const U=u>=b.startDate&&u<=b.endDate,Q=!b.resourceIds||b.resourceIds.length===0||b.resourceIds.includes(P.id);return U&&Q});return q?{resourceId:P.id,resourceName:P.name,date:u,dayOfWeek:M,isClosed:q.isClosed,hours:q.isClosed?[]:(q.timeslots||[]).map(b=>({start:b.startTime,end:b.endTime})),isException:!0}:{resourceId:P.id,resourceName:P.name,date:u,dayOfWeek:M,isClosed:T.length===0,hours:T,isException:!1}});return{agendaId:r,serviceName:D.serviceName||"",date:u,resources:k}}catch(f){throw console.error("Appointments handler — opening hours error:",f),e.createError({statusCode:f.statusCode||500,statusMessage:f.statusMessage||"Failed to get opening hours"})}})),l.get("/agendas/:agendaId/refund-flow",e.defineEventHandler(async t=>{const r=e.getRouterParam(t,"agendaId");if(!r)throw e.createError({statusCode:400,statusMessage:"Agenda ID is required."});try{const{firebase:o,agendaCollection:u}=await s(t),m=o.firestore().collection(u).doc(r),w=m.collection(g).doc(y),[D,E]=await Promise.all([m.get(),w.get()]);if(!D.exists)throw e.createError({statusCode:404,statusMessage:"Agenda not found."});const I=E.exists?E.data()||{}:{};return{refundFlow:I.flow||{},refundFlowEnabled:I.enabled!==!1,refundDialog:c(I.dialog)}}catch(o){throw console.error("Appointments handler — refund flow load error:",o),e.createError({statusCode:o.statusCode||500,statusMessage:o.statusMessage||"Failed to load refund flow"})}})),l.patch("/agendas/:agendaId/refund-flow",e.defineEventHandler(async t=>{const r=e.getRouterParam(t,"agendaId");if(!r)throw e.createError({statusCode:400,statusMessage:"Agenda ID is required."});try{const o=await e.readBody(t),{firebase:u,agendaCollection:f}=await s(t),m=u.firestore(),w=m.collection(f).doc(r),D=w.collection(g).doc(y),E=c(o?.refundDialog);if(!(await w.get()).exists)throw e.createError({statusCode:404,statusMessage:"Agenda not found."});const S=m.batch();return S.set(D,{flow:o?.refundFlow||{},enabled:o?.refundFlowEnabled!==!1,dialog:E,updatedAt:new Date().toISOString()},{merge:!0}),S.set(w,a(),{merge:!0}),await S.commit(),{success:!0}}catch(o){throw console.error("Appointments handler — refund flow save error:",o),e.createError({statusCode:o.statusCode||500,statusMessage:o.statusMessage||"Failed to save refund flow"})}})),l.get("/agendas/:agendaId/metadata-schema",e.defineEventHandler(async t=>{const r=e.getRouterParam(t,"agendaId");if(!r)throw e.createError({statusCode:400,statusMessage:"Agenda ID is required."});try{const{firebase:o,agendaCollection:u}=await s(t),m=o.firestore().collection(u).doc(r),w=m.collection(g).doc(h),[D,E]=await Promise.all([m.get(),w.get()]);if(!D.exists)throw e.createError({statusCode:404,statusMessage:"Agenda not found."});const I=E.exists?E.data()||{}:{};return{metadataSchema:n(I.schema)}}catch(o){throw console.error("Appointments handler — metadata schema load error:",o),e.createError({statusCode:o.statusCode||500,statusMessage:o.statusMessage||"Failed to load metadata schema"})}})),l.patch("/agendas/:agendaId/metadata-schema",e.defineEventHandler(async t=>{const r=e.getRouterParam(t,"agendaId");if(!r)throw e.createError({statusCode:400,statusMessage:"Agenda ID is required."});try{const o=await e.readBody(t),{firebase:u,agendaCollection:f}=await s(t),m=u.firestore(),w=m.collection(f).doc(r),D=w.collection(g).doc(h);if(!(await w.get()).exists)throw e.createError({statusCode:404,statusMessage:"Agenda not found."});const I=m.batch();return I.set(D,{schema:n(o?.metadataSchema),updatedAt:new Date().toISOString()},{merge:!0}),I.set(w,i(),{merge:!0}),await I.commit(),{success:!0}}catch(o){throw console.error("Appointments handler — metadata schema save error:",o),e.createError({statusCode:o.statusCode||500,statusMessage:o.statusMessage||"Failed to save metadata schema"})}})),l.get("/appointments",e.defineEventHandler(async t=>{const r=e.getQuery(t),o=Math.min(Number(r.quantity)||20,100),u=r.status||"confirmed";try{const{firebase:f,reservationsCollection:m}=await s(t),D=(await f.firestore().collection(m).orderBy("createdAt","desc").where("status","==",u).limit(o).get()).docs.flatMap(E=>d(A({id:E.id,...E.data()})));return{count:D.length,appointments:D}}catch(f){throw console.error("Appointments handler — list error:",f),e.createError({statusCode:f.statusCode||500,statusMessage:f.statusMessage||"Failed to list appointments"})}})),l.get("/appointments/find",e.defineEventHandler(async t=>{const r=e.getQuery(t),o=(r.name||"").toLowerCase().trim(),u=(r.email||"").toLowerCase().trim(),f=r.date;try{const{firebase:m,reservationsCollection:w}=await s(t);let D=m.firestore().collection(w);u&&(D=D.where("customerInfo.email","==",u));let I=(await D.orderBy("createdAt","desc").limit(200).get()).docs.flatMap(S=>d(A({id:S.id,...S.data()})));return f&&(I=I.filter(S=>S.date===f)),o&&(I=I.filter(S=>{const M=(S.customerInfo?.firstName||"").toLowerCase(),k=(S.customerInfo?.lastName||"").toLowerCase();return M.includes(o)||k.includes(o)||`${M} ${k}`.includes(o)})),{count:I.length,appointments:I}}catch(m){throw console.error("Appointments handler — find error:",m),e.createError({statusCode:m.statusCode||500,statusMessage:m.statusMessage||"Failed to find appointments"})}})),l.get("/appointments/:appointmentId",e.defineEventHandler(async t=>{const r=e.getRouterParam(t,"appointmentId");if(!r)throw e.createError({statusCode:400,statusMessage:"Appointment ID is required."});try{const{firebase:o,reservationsCollection:u}=await s(t),f=await o.firestore().collection(u).doc(r).get();if(!f.exists)throw e.createError({statusCode:404,statusMessage:"Appointment not found."});const m=A({id:f.id,...f.data()});return{...m,reservations:d(m)}}catch(o){throw console.error("Appointments handler — get error:",o),e.createError({statusCode:o.statusCode||500,statusMessage:o.statusMessage||"Failed to get appointment"})}})),l.handler}const F="https://analyticsdata.googleapis.com/v1beta",x=["https://www.googleapis.com/auth/analytics.readonly","https://www.googleapis.com/auth/webmasters.readonly"],H="https://searchconsole.googleapis.com/webmasters/v3";function B(C){const{decrypt:R,getGoogleAccessToken:A}=C,p=e.createRouter();async function l(s){const{supabase:d,instanceId:c}=s.context.module,{data:a,error:n}=await d.from("project_modules").select("config").eq("id",c).single();if(n||!a?.config)throw e.createError({statusCode:500,statusMessage:"Failed to load module config."});const i=a.config,t=i.propertyId,r=i.serviceAccount,o=i.siteUrl||"";if(!r)throw e.createError({statusCode:400,statusMessage:"No Google Service Account configured for this module."});if(!t)throw e.createError({statusCode:400,statusMessage:"No GA4 Property ID configured for this module."});const{data:u,error:f}=await d.from("integrations").select("config").eq("id",r).single();let m=u?.config;if(f||!m){const{data:S,error:M}=await d.from("agency_integrations").select("config").eq("id",r).single();if(M||!S?.config)throw e.createError({statusCode:500,statusMessage:"Failed to load Google Service Account credentials."});m=S.config}const w=R(m.clientEmail),D=R(m.privateKey),E=R(m.projectId);return{accessToken:await A({clientEmail:w,privateKey:D,projectId:E},x),propertyId:t,siteUrl:o}}async function g(s,d,c){const a=await fetch(`${F}/properties/${d}:runReport`,{method:"POST",headers:{Authorization:`Bearer ${s}`,"Content-Type":"application/json"},body:JSON.stringify(c)});if(!a.ok){const n=await a.json().catch(()=>({}));throw console.error("GA4 runReport failed:",n),e.createError({statusCode:a.status,statusMessage:n?.error?.message||"GA4 API request failed"})}return a.json()}p.get("/report",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=n.match(/^(\d+)daysAgo$/),r=t?parseInt(t[1],10):30,o=`${r*2}daysAgo`,u=`${r+1}daysAgo`,f=[{name:"sessions"},{name:"totalUsers"},{name:"screenPageViews"},{name:"bounceRate"},{name:"averageSessionDuration"},{name:"newUsers"},{name:"engagementRate"},{name:"sessionsPerUser"},{name:"screenPageViewsPerSession"}];let m,w=!0;try{m=await g(d,c,{dateRanges:[{startDate:n,endDate:i,name:"current"},{startDate:o,endDate:u,name:"previous"}],dimensions:[{name:"date"}],metrics:f,metricAggregations:["TOTAL"],orderBys:[{dimension:{dimensionName:"date"}}]})}catch{w=!1,m=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"date"}],metrics:f,metricAggregations:["TOTAL"],orderBys:[{dimension:{dimensionName:"date"}}]})}return $(m,w)})),p.get("/realtime",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=await fetch(`${F}/properties/${c}:runRealtimeReport`,{method:"POST",headers:{Authorization:`Bearer ${d}`,"Content-Type":"application/json"},body:JSON.stringify({dimensions:[{name:"unifiedScreenName"}],metrics:[{name:"activeUsers"}],orderBys:[{metric:{metricName:"activeUsers"},desc:!0}],limit:5})});if(!a.ok){const r=await a.json().catch(()=>({}));throw e.createError({statusCode:a.status,statusMessage:r?.error?.message||"Realtime API failed"})}const n=await a.json(),i=(n?.rows||[]).reduce((r,o)=>r+parseInt(o.metricValues?.[0]?.value||"0",10),0),t=(n?.rows||[]).map(r=>({page:r.dimensionValues?.[0]?.value||"(not set)",activeUsers:parseInt(r.metricValues?.[0]?.value||"0",10)}));return{activeUsers:i,activePages:t}})),p.get("/top-pages",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=parseInt(a.limit||"10",10),r=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"pagePath"}],metrics:[{name:"screenPageViews"},{name:"totalUsers"},{name:"averageSessionDuration"}],orderBys:[{metric:{metricName:"screenPageViews"},desc:!0}],limit:t*2}),o=v(r,"pagePath"),u=new Map;for(const m of o.rows){const w=m.pagePath.toLowerCase().replace(/\/+$/,""),D=u.get(w);if(D){D.screenPageViews+=m.screenPageViews||0,D.totalUsers+=m.totalUsers||0;const E=D.screenPageViews;E>0&&(D.averageSessionDuration=(D.averageSessionDuration*(E-(m.screenPageViews||0))+(m.averageSessionDuration||0)*(m.screenPageViews||0))/E)}else u.set(w,{...m})}return{rows:[...u.values()].sort((m,w)=>(w.screenPageViews||0)-(m.screenPageViews||0)).slice(0,t),rowCount:o.rowCount}})),p.get("/top-sources",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"sessionSource"}],metrics:[{name:"sessions"},{name:"totalUsers"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:10});return v(t,"sessionSource")})),p.get("/devices",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"deviceCategory"}],metrics:[{name:"sessions"},{name:"totalUsers"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}]});return v(t,"deviceCategory")})),p.get("/countries",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"country"}],metrics:[{name:"sessions"},{name:"totalUsers"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:10});return v(t,"country")})),p.get("/acquisition/channels",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"sessionDefaultChannelGroup"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"newUsers"},{name:"engagementRate"},{name:"averageSessionDuration"},{name:"screenPageViewsPerSession"},{name:"conversions"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:15});return v(t,"sessionDefaultChannelGroup")})),p.get("/acquisition/source-medium",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"sessionSourceMedium"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"newUsers"},{name:"bounceRate"},{name:"averageSessionDuration"},{name:"conversions"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:20});return v(t,"sessionSourceMedium")})),p.get("/acquisition/referrals",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"sessionSource"}],dimensionFilter:{filter:{fieldName:"sessionMedium",stringFilter:{value:"referral"}}},metrics:[{name:"sessions"},{name:"totalUsers"},{name:"engagementRate"},{name:"averageSessionDuration"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:20});return v(t,"sessionSource")})),p.get("/acquisition/campaigns",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"sessionCampaignName"}],dimensionFilter:{notExpression:{filter:{fieldName:"sessionCampaignName",stringFilter:{value:"(not set)"}}}},metrics:[{name:"sessions"},{name:"totalUsers"},{name:"conversions"},{name:"engagementRate"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:20});return v(t,"sessionCampaignName")})),p.get("/content/all-pages",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=parseInt(a.limit||"50",10),r=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"pagePath"}],metrics:[{name:"screenPageViews"},{name:"totalUsers"},{name:"averageSessionDuration"},{name:"bounceRate"},{name:"engagementRate"},{name:"sessions"},{name:"userEngagementDuration"}],orderBys:[{metric:{metricName:"screenPageViews"},desc:!0}],limit:t});return v(r,"pagePath")})),p.get("/content/landing-pages",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"landingPagePlusQueryString"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"bounceRate"},{name:"averageSessionDuration"},{name:"screenPageViews"},{name:"engagementRate"}],orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:30});return v(t,"landingPagePlusQueryString")})),p.get("/content/exit-pages",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"pagePath"}],metrics:[{name:"sessions"},{name:"screenPageViews"},{name:"totalUsers"},{name:"bounceRate"}],orderBys:[{metric:{metricName:"screenPageViews"},desc:!0}],limit:20}),r=v(t,"pagePath");return r.rows=r.rows.map(o=>({...o,exitRate:o.bounceRate||0})),r})),p.get("/content/search-terms",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c,siteUrl:a}=await l(s),n=e.getQuery(s),i=n.startDate||"30daysAgo",t=n.endDate||"today";if(a)try{const u=`${H}/sites/${encodeURIComponent(a)}/searchAnalytics/query`,f=await fetch(u,{method:"POST",headers:{Authorization:`Bearer ${d}`,"Content-Type":"application/json"},body:JSON.stringify({startDate:y(i),endDate:y(t),dimensions:["query"],rowLimit:30})});if(f.ok){const w=((await f.json()).rows||[]).map(D=>({query:D.keys[0],clicks:D.clicks,impressions:D.impressions,ctr:D.ctr,position:D.position}));if(w.length>0)return{rows:w,rowCount:w.length,source:"search_console"}}else{const m=await f.text().catch(()=>"");console.error(`[GA Module] Search Console API error (${f.status}):`,m)}}catch(u){console.error("[GA Module] Search Console request failed:",u?.message||u)}else console.log("[GA Module] No siteUrl configured, skipping Search Console");try{const u=await g(d,c,{dateRanges:[{startDate:i,endDate:t}],dimensions:[{name:"sessionGoogleAdsQuery"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"engagementRate"}],dimensionFilter:{andGroup:{expressions:[{notExpression:{filter:{fieldName:"sessionGoogleAdsQuery",stringFilter:{value:"(not set)"}}}},{notExpression:{filter:{fieldName:"sessionGoogleAdsQuery",stringFilter:{value:"(not provided)"}}}}]}},orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:30}),f=v(u,"sessionGoogleAdsQuery");if(f.rows.length>0)return{...f,source:"google_ads"}}catch{}const r=await g(d,c,{dateRanges:[{startDate:i,endDate:t}],dimensions:[{name:"landingPagePlusQueryString"}],metrics:[{name:"sessions"},{name:"totalUsers"},{name:"engagementRate"}],dimensionFilter:{andGroup:{expressions:[{filter:{fieldName:"sessionMedium",stringFilter:{matchType:"EXACT",value:"organic"}}},{notExpression:{filter:{fieldName:"landingPagePlusQueryString",stringFilter:{value:"(not set)"}}}}]}},orderBys:[{metric:{metricName:"sessions"},desc:!0}],limit:30});return{...v(r,"landingPagePlusQueryString"),source:"organic_landing_pages"}})),p.get("/audience/overview",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"newVsReturning"}],metrics:[{name:"totalUsers"},{name:"sessions"},{name:"engagementRate"},{name:"averageSessionDuration"},{name:"screenPageViewsPerSession"}],metricAggregations:["TOTAL"]});return v(t,"newVsReturning")})),p.get("/audience/technology",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=a.dimension||"browser",o=["browser","operatingSystem","screenResolution"].includes(t)?t:"browser",u=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:o}],metrics:[{name:"totalUsers"},{name:"sessions"},{name:"engagementRate"}],orderBys:[{metric:{metricName:"totalUsers"},desc:!0}],limit:10});return v(u,o)})),p.get("/audience/languages",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"language"}],metrics:[{name:"totalUsers"},{name:"sessions"}],orderBys:[{metric:{metricName:"totalUsers"},desc:!0}],limit:15});return v(t,"language")})),p.get("/audience/hours",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"dayOfWeekName"},{name:"hour"}],metrics:[{name:"sessions"}],orderBys:[{dimension:{dimensionName:"dayOfWeekName"}},{dimension:{dimensionName:"hour"}}],limit:168});return N(t,["dayOfWeekName","hour"])})),p.get("/audience/cities",e.defineEventHandler(async s=>{const{accessToken:d,propertyId:c}=await l(s),a=e.getQuery(s),n=a.startDate||"30daysAgo",i=a.endDate||"today",t=await g(d,c,{dateRanges:[{startDate:n,endDate:i}],dimensions:[{name:"city"},{name:"country"}],metrics:[{name:"totalUsers"},{name:"sessions"}],dimensionFilter:{notExpression:{filter:{fieldName:"city",stringFilter:{value:"(not set)"}}}},orderBys:[{metric:{metricName:"totalUsers"},desc:!0}],limit:20});return N(t,["city","country"])}));function y(s){const d=s.match(/^(\d+)daysAgo$/);if(d){const c=new Date;return c.setDate(c.getDate()-parseInt(d[1],10)),c.toISOString().slice(0,10)}if(s==="today")return new Date().toISOString().slice(0,10);if(s==="yesterday"){const c=new Date;return c.setDate(c.getDate()-1),c.toISOString().slice(0,10)}return s}async function h(s,d,c){const a=`${H}/sites/${encodeURIComponent(d)}/searchAnalytics/query`,n=await fetch(a,{method:"POST",headers:{Authorization:`Bearer ${s}`,"Content-Type":"application/json"},body:JSON.stringify(c)});if(!n.ok){const i=await n.json().catch(()=>({}));throw e.createError({statusCode:n.status,statusMessage:i?.error?.message||"Search Console API request failed"})}return n.json()}return p.get("/seo/keywords",e.defineEventHandler(async s=>{const{accessToken:d,siteUrl:c}=await l(s);if(!c)throw e.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured. Add it in module settings."});const a=e.getQuery(s),n=y(a.startDate||"30daysAgo"),i=y(a.endDate||"today"),t=Math.min(parseInt(a.limit)||50,100),r=await h(d,c,{startDate:n,endDate:i,dimensions:["query"],rowLimit:t});return{rows:(r.rows||[]).map(o=>({query:o.keys[0],clicks:o.clicks,impressions:o.impressions,ctr:o.ctr,position:o.position})),rowCount:r.rows?.length||0}})),p.get("/seo/pages",e.defineEventHandler(async s=>{const{accessToken:d,siteUrl:c}=await l(s);if(!c)throw e.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured."});const a=e.getQuery(s),n=y(a.startDate||"30daysAgo"),i=y(a.endDate||"today"),t=Math.min(parseInt(a.limit)||50,100),r=await h(d,c,{startDate:n,endDate:i,dimensions:["page"],rowLimit:t});return{rows:(r.rows||[]).map(o=>({page:o.keys[0],clicks:o.clicks,impressions:o.impressions,ctr:o.ctr,position:o.position})),rowCount:r.rows?.length||0}})),p.get("/seo/trends",e.defineEventHandler(async s=>{const{accessToken:d,siteUrl:c}=await l(s);if(!c)throw e.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured."});const a=e.getQuery(s),n=y(a.startDate||"30daysAgo"),i=y(a.endDate||"today"),r=((await h(d,c,{startDate:n,endDate:i,dimensions:["date"],rowLimit:500})).rows||[]).map(u=>({date:u.keys[0],clicks:u.clicks,impressions:u.impressions,ctr:u.ctr,position:u.position})).sort((u,f)=>u.date.localeCompare(f.date)),o=r.reduce((u,f)=>({clicks:u.clicks+f.clicks,impressions:u.impressions+f.impressions}),{clicks:0,impressions:0});return o.ctr=o.impressions>0?o.clicks/o.impressions:0,o.avgPosition=r.length>0?r.reduce((u,f)=>u+f.position,0)/r.length:0,{rows:r,totals:o,rowCount:r.length}})),p.get("/seo/query-pages",e.defineEventHandler(async s=>{const{accessToken:d,siteUrl:c}=await l(s);if(!c)throw e.createError({statusCode:400,statusMessage:"Search Console Site URL is not configured."});const a=e.getQuery(s),n=y(a.startDate||"30daysAgo"),i=y(a.endDate||"today"),t=await h(d,c,{startDate:n,endDate:i,dimensions:["query","page"],rowLimit:100});return{rows:(t.rows||[]).map(r=>({query:r.keys[0],page:r.keys[1],clicks:r.clicks,impressions:r.impressions,ctr:r.ctr,position:r.position})),rowCount:t.rows?.length||0}})),p.handler}function $(C,R=!0){const A=(C.metricHeaders||[]).map(n=>n.name),l=(C.dimensionHeaders||[]).map(n=>n.name).indexOf("date"),g=[];(C.rows||[]).forEach(n=>{const i=l>=0?n.dimensionValues[l]?.value:n.dimensionValues[0]?.value;if(!i||i.length<8)return;const r={date:`${i.slice(0,4)}-${i.slice(4,6)}-${i.slice(6,8)}`};A.forEach((o,u)=>{r[o]=parseFloat(n.metricValues[u]?.value||"0")}),g.push(r)});const y={},h={};C.totals&&C.totals.length>=2?A.forEach((n,i)=>{y[n]=parseFloat(C.totals[0]?.metricValues?.[i]?.value||"0"),h[n]=parseFloat(C.totals[1]?.metricValues?.[i]?.value||"0")}):C.totals&&C.totals.length===1&&A.forEach((n,i)=>{y[n]=parseFloat(C.totals[0]?.metricValues?.[i]?.value||"0")});const s={};A.forEach(n=>{const i=y[n]||0,t=h[n];t!==void 0&&t!==0?s[n]=(i-t)/t*100:s[n]=null});const d=new Map;for(const n of g)d.has(n.date)||d.set(n.date,n);const c=Array.from(d.values()).sort((n,i)=>n.date.localeCompare(i.date)),a=R&&C.totals&&C.totals.length>=2?c.slice(-Math.ceil(c.length/2)):c;return{rows:a,totals:y,previousTotals:h,changes:s,rowCount:a.length}}function v(C,R){const A=(C.metricHeaders||[]).map(l=>l.name),p=(C.rows||[]).map(l=>{const g={[R]:l.dimensionValues[0].value};return A.forEach((y,h)=>{g[y]=parseFloat(l.metricValues[h].value)}),g});return{rows:p,rowCount:C.rowCount||p.length}}function N(C,R){const A=(C.metricHeaders||[]).map(l=>l.name),p=(C.rows||[]).map(l=>{const g={};return R.forEach((y,h)=>{g[y]=l.dimensionValues[h]?.value||""}),A.forEach((y,h)=>{g[y]=parseFloat(l.metricValues[h].value)}),g});return{rows:p,rowCount:C.rowCount||p.length}}function G(C){const{decrypt:R}=C,A=e.createRouter();async function p(l){const{supabase:g,instanceId:y}=l.context.module,{data:h,error:s}=await g.from("project_modules").select("config").eq("id",y).single();if(s||!h?.config)throw e.createError({statusCode:500,statusMessage:"Failed to load module config."});const d=h.config,c=d.githubIntegration,a=d.githubRepo,n=d.githubOwner;if(!c||!a||!n)throw e.createError({statusCode:400,statusMessage:"No GitHub integration configured for this module."});const{data:i,error:t}=await g.from("integrations").select("config").eq("id",c).single();let r=i?.config;if(t||!r){const{data:u,error:f}=await g.from("agency_integrations").select("config").eq("id",c).single();if(f||!u?.config)throw e.createError({statusCode:500,statusMessage:"Failed to load Github credentials."});r=u.config}return{token:R(r.token),repo:a,owner:n}}return A.post("/github/deploy",e.defineEventHandler(async l=>{try{const{token:g,repo:y,owner:h}=await p(l);await fetch(`https://api.github.com/repos/${h}/${y}/dispatches`,{method:"POST",headers:{Authorization:`Bearer ${g}`,Accept:"application/vnd.github+json"},body:JSON.stringify({event_type:"deploy-site"})})}catch(g){throw console.error("Error triggering GitHub deployment:",g),e.createError({statusCode:500,statusMessage:"Failed to trigger deployment."})}})),A.handler}exports.appointments=V;exports.contentManager=G;exports.googleAnalytics=B;exports.products=O;