lula2 0.0.10-nightly.8 → 0.1.0

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.
@@ -1 +1 @@
1
- import"../chunks/DsnmJJEf.js";import{at as le,p as R,ae as y,d as k,T as $,G as m,h as u,af as F,ag as M,c as i,n as A,r as n,b as v,e as N,J as D,K as Y,ak as Q,a as U,f as P,o as xe,au as be,s as V,av as ie}from"../chunks/Cby0Z7eP.js";import{l as z,p as C,i as w,a as te,s as de}from"../chunks/DqsOU3kV.js";import{g as me}from"../chunks/BW_5Gp3K.js";import{p as _e,D as we}from"../chunks/C9-IlOVE.js";import{i as j}from"../chunks/C5zWTfmV.js";import{a as K,f as ce,w as q}from"../chunks/BtOhWAVU.js";le(["change"]);var ye=M("<title> </title>"),ke=M('<svg><!><path d="M11.41 26.59L7.83 23 28 23 28 21 7.83 21 11.41 17.41 10 16 4 22 10 28 11.41 26.59zM28 10L22 4 20.59 5.41 24.17 9 4 9 4 11 24.17 11 20.59 14.59 22 16 28 10z"></path></svg>');function $e(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=ke();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=ye(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(),n(s),v(b,s),N()}var ze=P('<a href="/setup" class="inline-flex items-center gap-2 px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2 transition-colors" title="Switch control set"><!> Switch</a>'),Ce=P('<div class="text-sm text-gray-500 dark:text-gray-400">Loading...</div>');function Me(b,t){R(t,!1);const e=()=>te(ce,"$appState",l),[l,o]=de();j();var g=Q(),f=U(g);{var r=p=>{var x=ze(),d=i(x);$e(d,{size:16}),A(),n(x),v(p,x)},s=p=>{var x=Ce();v(p,x)};w(f,p=>{e().isConnected?p(r):p(s,!1)})}v(b,g),N(),o()}var Se=M("<title> </title>"),Le=M('<svg><!><path d="M31 16L24 23 22.59 21.59 28.17 16 22.59 10.41 24 9 31 16zM1 16L8 9 9.41 10.41 3.83 16 9.41 21.59 8 23 1 16z"></path><path d="M5.91 15H26.080000000000002V17H5.91z" transform="rotate(-75 15.996 16)"></path></svg>');function Ae(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=Le();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Se(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(2),n(s),v(b,s),N()}var Ee=M("<title> </title>"),He=M('<svg><!><path d="M13 21L26.17 21 23.59 23.59 25 25 30 20 25 15 23.59 16.41 26.17 19 13 19 13 21z"></path><path d="M22,14V10a1,1,0,0,0-.29-.71l-7-7A1,1,0,0,0,14,2H4A2,2,0,0,0,2,4V28a2,2,0,0,0,2,2H20a2,2,0,0,0,2-2V26H20v2H4V4h8v6a2,2,0,0,0,2,2h6v2Zm-8-4V4.41L19.59,10Z"></path></svg>');function Ve(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=He();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Ee(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(2),n(s),v(b,s),N()}var Re=M("<title> </title>"),Ne=M('<svg><!><path d="M26 24v4H6V24H4v4H4a2 2 0 002 2H26a2 2 0 002-2h0V24zM26 14L24.59 12.59 17 20.17 17 2 15 2 15 20.17 7.41 12.59 6 14 16 24 26 14z"></path></svg>');function ne(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=Ne();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Re(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(),n(s),v(b,s),N()}var Pe=M("<title> </title>"),Te=M('<svg><!><path fill-rule="evenodd" d="M16,2a14,14,0,0,0-4.43,27.28c.7.13,1-.3,1-.67s0-1.21,0-2.38c-3.89.84-4.71-1.88-4.71-1.88A3.71,3.71,0,0,0,6.24,22.3c-1.27-.86.1-.85.1-.85A2.94,2.94,0,0,1,8.48,22.9a3,3,0,0,0,4.08,1.16,2.93,2.93,0,0,1,.88-1.87c-3.1-.36-6.37-1.56-6.37-6.92a5.4,5.4,0,0,1,1.44-3.76,5,5,0,0,1,.14-3.7s1.17-.38,3.85,1.43a13.3,13.3,0,0,1,7,0c2.67-1.81,3.84-1.43,3.84-1.43a5,5,0,0,1,.14,3.7,5.4,5.4,0,0,1,1.44,3.76c0,5.38-3.27,6.56-6.39,6.91a3.33,3.33,0,0,1,.95,2.59c0,1.87,0,3.38,0,3.84s.25.81,1,.67A14,14,0,0,0,16,2Z"></path></svg>');function Be(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=Te();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Pe(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(),n(s),v(b,s),N()}var De=(b,t)=>t("csv"),Ye=(b,t)=>t("excel"),je=(b,t)=>t("json"),Ge=P('<div class="space-y-1 p-1"><button class="w-full text-left px-3 py-2 text-sm rounded-md transition-colors duration-200 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700"><div class="flex items-center gap-2"><!> <span>Export as CSV</span></div></button> <button class="w-full text-left px-3 py-2 text-sm rounded-md transition-colors duration-200 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700"><div class="flex items-center gap-2"><!> <span>Export as Excel</span></div></button> <button class="w-full text-left px-3 py-2 text-sm rounded-md transition-colors duration-200 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700"><div class="flex items-center gap-2"><!> <span>Export as JSON</span></div></button></div>'),Ie=P('<div class="flex-1 flex justify-center items-center"><div class="text-center"><div class="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-600 mx-auto mb-4"></div> <p class="text-gray-500 dark:text-gray-400">Switching control set...</p></div></div>'),Ze=P('<div class="flex-1 flex justify-center items-center"><div class="text-center"><div class="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-600 mx-auto mb-4"></div> <p class="text-gray-500 dark:text-gray-400"> </p></div></div>'),Je=P('<header class="bg-white dark:bg-gray-900 shadow-sm border-b border-gray-200 dark:border-gray-700 flex-shrink-0"><div class="w-full px-6 lg:px-8"><div class="flex justify-between items-center h-16"><div class="flex items-center"><a href="/" class="flex items-center space-x-3 hover:opacity-80 transition-opacity"><img src="/lula.png" class="h-8 w-8" alt="Lula Logo"/> <div class="flex flex-col"><span class="text-xl font-bold text-gray-900 dark:text-white">Lula</span> <span class="text-xs text-gray-500 dark:text-gray-400 -mt-1">Gitops for Compliance</span></div></a></div> <div class="flex items-center space-x-4"><!> <!> <a class="p-2 text-gray-500 dark:text-gray-400 hover:text-gray-700 dark:hover:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-800 rounded-lg transition-colors" title="Github" href="https://github.com/defenseunicorns/lula" target="_blank"><!></a></div></div></div></header> <div class="flex-1 flex gap-6 p-6 overflow-hidden"><!></div>',1),Ue=P('<div class="h-screen flex flex-col"><!></div>');function et(b,t){R(t,!0);const e=()=>te(_e,"$page",o),l=()=>te(ce,"$appState",o),[o,g]=de();let f=!1;async function r(a){try{const c=`/api/export-controls?format=${a}`,h=document.createElement("a");h.href=c,h.download="",document.body.appendChild(h),h.click(),document.body.removeChild(h)}catch(c){console.error("Export failed:",c)}}xe(()=>{q.connect();let a=null;const c=async()=>{if(f||e().url.pathname==="/setup"||l().isSwitchingControlSet)return;const h=l();if(h.isConnected){if(f=!0,(!h.name||h.name==="Unknown Control Set"||h.id==="unknown"||h.id==="default")&&(!h.controls||h.controls.length===0)){await q.scanControlSets();const E=await new Promise(G=>{const T=O=>{window.removeEventListener("control-sets-list",T),G(O.detail)};window.addEventListener("control-sets-list",T),setTimeout(()=>{window.removeEventListener("control-sets-list",T),G(null)},2e3)});E&&Array.isArray(E)&&E.length===1?(console.log("Auto-loading single control set:",E[0].path),await q.switchControlSet(E[0].path)):me("/setup")}}else a=window.setTimeout(c,500)};return a=window.setTimeout(c,100),()=>{a&&clearTimeout(a)}}),be(()=>{q.disconnect()});var s=Ue(),p=i(s);{var x=a=>{var c=Je(),h=U(c),E=i(h),G=i(E),T=V(i(G),2),O=i(T);Me(O,{});var ae=V(O,2);{var ve=_=>{we(_,{buttonLabel:"Export",get buttonIcon(){return ne},buttonClass:"inline-flex items-center px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2 transition-colors",dropdownClass:"w-48",children:W=>{var Z=Ge(),B=i(Z);B.__click=[De,r];var S=i(B),H=i(S);Ve(H,{class:"w-4 h-4"}),A(2),n(S),n(B);var L=V(B,2);L.__click=[Ye,r];var J=i(L),X=i(J);ne(X,{class:"w-4 h-4"}),A(2),n(J),n(L);var ee=V(L,2);ee.__click=[je,r];var oe=i(ee),pe=i(oe);Ae(pe,{class:"w-4 h-4"}),A(2),n(oe),n(ee),n(Z),v(W,Z)},$$slots:{default:!0}})};w(ae,_=>{l().isConnected&&l().controls&&l().controls.length>0&&_(ve)})}var re=V(ae,2),ue=i(re);Be(ue,{class:"w-5 h-5"}),n(re),n(T),n(G),n(E),n(h);var se=V(h,2),ge=i(se);{var fe=_=>{var I=Ie();v(_,I)},he=_=>{var I=Q(),W=U(I);{var Z=S=>{var H=Ze(),L=i(H),J=V(i(L),2),X=i(J,!0);n(J),n(L),n(H),D(()=>Y(X,l().isConnected?l().fieldSchema?"Loading controls...":"Loading schema...":"Connecting...")),v(S,H)},B=S=>{var H=Q(),L=U(H);ie(L,()=>t.children),v(S,H)};w(W,S=>{!l().isConnected||!l().controls||l().controls.length===0||!l().fieldSchema?S(Z):S(B,!1)},!0)}v(_,I)};w(ge,_=>{l().isSwitchingControlSet?_(fe):_(he,!1)})}n(se),v(a,c)},d=a=>{var c=Q(),h=U(c);ie(h,()=>t.children),v(a,c)};w(p,a=>{e().url.pathname!=="/setup"?a(x):a(d,!1)})}n(s),v(b,s),N(),g()}le(["click"]);export{et as component};
1
+ import"../chunks/DsnmJJEf.js";import{at as le,p as R,ae as y,d as k,T as $,G as m,h as u,af as F,ag as M,c as i,n as A,r as n,b as v,e as N,J as D,K as Y,ak as Q,a as U,f as P,o as xe,au as be,s as V,av as ie}from"../chunks/Cby0Z7eP.js";import{l as z,p as C,i as w,a as te,s as de}from"../chunks/DqsOU3kV.js";import{g as me}from"../chunks/DqFUUtys.js";import{p as _e,D as we}from"../chunks/ByARpoaF.js";import{i as j}from"../chunks/C5zWTfmV.js";import{a as K,f as ce,w as q}from"../chunks/BtOhWAVU.js";le(["change"]);var ye=M("<title> </title>"),ke=M('<svg><!><path d="M11.41 26.59L7.83 23 28 23 28 21 7.83 21 11.41 17.41 10 16 4 22 10 28 11.41 26.59zM28 10L22 4 20.59 5.41 24.17 9 4 9 4 11 24.17 11 20.59 14.59 22 16 28 10z"></path></svg>');function $e(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=ke();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=ye(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(),n(s),v(b,s),N()}var ze=P('<a href="/setup" class="inline-flex items-center gap-2 px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2 transition-colors" title="Switch control set"><!> Switch</a>'),Ce=P('<div class="text-sm text-gray-500 dark:text-gray-400">Loading...</div>');function Me(b,t){R(t,!1);const e=()=>te(ce,"$appState",l),[l,o]=de();j();var g=Q(),f=U(g);{var r=p=>{var x=ze(),d=i(x);$e(d,{size:16}),A(),n(x),v(p,x)},s=p=>{var x=Ce();v(p,x)};w(f,p=>{e().isConnected?p(r):p(s,!1)})}v(b,g),N(),o()}var Se=M("<title> </title>"),Le=M('<svg><!><path d="M31 16L24 23 22.59 21.59 28.17 16 22.59 10.41 24 9 31 16zM1 16L8 9 9.41 10.41 3.83 16 9.41 21.59 8 23 1 16z"></path><path d="M5.91 15H26.080000000000002V17H5.91z" transform="rotate(-75 15.996 16)"></path></svg>');function Ae(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=Le();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Se(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(2),n(s),v(b,s),N()}var Ee=M("<title> </title>"),He=M('<svg><!><path d="M13 21L26.17 21 23.59 23.59 25 25 30 20 25 15 23.59 16.41 26.17 19 13 19 13 21z"></path><path d="M22,14V10a1,1,0,0,0-.29-.71l-7-7A1,1,0,0,0,14,2H4A2,2,0,0,0,2,4V28a2,2,0,0,0,2,2H20a2,2,0,0,0,2-2V26H20v2H4V4h8v6a2,2,0,0,0,2,2h6v2Zm-8-4V4.41L19.59,10Z"></path></svg>');function Ve(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=He();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Ee(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(2),n(s),v(b,s),N()}var Re=M("<title> </title>"),Ne=M('<svg><!><path d="M26 24v4H6V24H4v4H4a2 2 0 002 2H26a2 2 0 002-2h0V24zM26 14L24.59 12.59 17 20.17 17 2 15 2 15 20.17 7.41 12.59 6 14 16 24 26 14z"></path></svg>');function ne(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=Ne();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Re(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(),n(s),v(b,s),N()}var Pe=M("<title> </title>"),Te=M('<svg><!><path fill-rule="evenodd" d="M16,2a14,14,0,0,0-4.43,27.28c.7.13,1-.3,1-.67s0-1.21,0-2.38c-3.89.84-4.71-1.88-4.71-1.88A3.71,3.71,0,0,0,6.24,22.3c-1.27-.86.1-.85.1-.85A2.94,2.94,0,0,1,8.48,22.9a3,3,0,0,0,4.08,1.16,2.93,2.93,0,0,1,.88-1.87c-3.1-.36-6.37-1.56-6.37-6.92a5.4,5.4,0,0,1,1.44-3.76,5,5,0,0,1,.14-3.7s1.17-.38,3.85,1.43a13.3,13.3,0,0,1,7,0c2.67-1.81,3.84-1.43,3.84-1.43a5,5,0,0,1,.14,3.7,5.4,5.4,0,0,1,1.44,3.76c0,5.38-3.27,6.56-6.39,6.91a3.33,3.33,0,0,1,.95,2.59c0,1.87,0,3.38,0,3.84s.25.81,1,.67A14,14,0,0,0,16,2Z"></path></svg>');function Be(b,t){const e=z(t,["children","$$slots","$$events","$$legacy"]),l=z(e,["size","title"]);R(t,!1);const o=$(),g=$();let f=C(t,"size",8,16),r=C(t,"title",8,void 0);y(()=>(m(e),m(r())),()=>{k(o,e["aria-label"]||e["aria-labelledby"]||r())}),y(()=>(u(o),m(e)),()=>{k(g,{"aria-hidden":u(o)?void 0:!0,role:u(o)?"img":void 0,focusable:Number(e.tabindex)===0?!0:void 0})}),F(),j();var s=Te();K(s,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:f(),height:f(),...u(g),...l}));var p=i(s);{var x=d=>{var a=Pe(),c=i(a,!0);n(a),D(()=>Y(c,r())),v(d,a)};w(p,d=>{r()&&d(x)})}A(),n(s),v(b,s),N()}var De=(b,t)=>t("csv"),Ye=(b,t)=>t("excel"),je=(b,t)=>t("json"),Ge=P('<div class="space-y-1 p-1"><button class="w-full text-left px-3 py-2 text-sm rounded-md transition-colors duration-200 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700"><div class="flex items-center gap-2"><!> <span>Export as CSV</span></div></button> <button class="w-full text-left px-3 py-2 text-sm rounded-md transition-colors duration-200 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700"><div class="flex items-center gap-2"><!> <span>Export as Excel</span></div></button> <button class="w-full text-left px-3 py-2 text-sm rounded-md transition-colors duration-200 text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700"><div class="flex items-center gap-2"><!> <span>Export as JSON</span></div></button></div>'),Ie=P('<div class="flex-1 flex justify-center items-center"><div class="text-center"><div class="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-600 mx-auto mb-4"></div> <p class="text-gray-500 dark:text-gray-400">Switching control set...</p></div></div>'),Ze=P('<div class="flex-1 flex justify-center items-center"><div class="text-center"><div class="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-600 mx-auto mb-4"></div> <p class="text-gray-500 dark:text-gray-400"> </p></div></div>'),Je=P('<header class="bg-white dark:bg-gray-900 shadow-sm border-b border-gray-200 dark:border-gray-700 flex-shrink-0"><div class="w-full px-6 lg:px-8"><div class="flex justify-between items-center h-16"><div class="flex items-center"><a href="/" class="flex items-center space-x-3 hover:opacity-80 transition-opacity"><img src="/lula.png" class="h-8 w-8" alt="Lula Logo"/> <div class="flex flex-col"><span class="text-xl font-bold text-gray-900 dark:text-white">Lula</span> <span class="text-xs text-gray-500 dark:text-gray-400 -mt-1">Gitops for Compliance</span></div></a></div> <div class="flex items-center space-x-4"><!> <!> <a class="p-2 text-gray-500 dark:text-gray-400 hover:text-gray-700 dark:hover:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-800 rounded-lg transition-colors" title="Github" href="https://github.com/defenseunicorns/lula" target="_blank"><!></a></div></div></div></header> <div class="flex-1 flex gap-6 p-6 overflow-hidden"><!></div>',1),Ue=P('<div class="h-screen flex flex-col"><!></div>');function et(b,t){R(t,!0);const e=()=>te(_e,"$page",o),l=()=>te(ce,"$appState",o),[o,g]=de();let f=!1;async function r(a){try{const c=`/api/export-controls?format=${a}`,h=document.createElement("a");h.href=c,h.download="",document.body.appendChild(h),h.click(),document.body.removeChild(h)}catch(c){console.error("Export failed:",c)}}xe(()=>{q.connect();let a=null;const c=async()=>{if(f||e().url.pathname==="/setup"||l().isSwitchingControlSet)return;const h=l();if(h.isConnected){if(f=!0,(!h.name||h.name==="Unknown Control Set"||h.id==="unknown"||h.id==="default")&&(!h.controls||h.controls.length===0)){await q.scanControlSets();const E=await new Promise(G=>{const T=O=>{window.removeEventListener("control-sets-list",T),G(O.detail)};window.addEventListener("control-sets-list",T),setTimeout(()=>{window.removeEventListener("control-sets-list",T),G(null)},2e3)});E&&Array.isArray(E)&&E.length===1?(console.log("Auto-loading single control set:",E[0].path),await q.switchControlSet(E[0].path)):me("/setup")}}else a=window.setTimeout(c,500)};return a=window.setTimeout(c,100),()=>{a&&clearTimeout(a)}}),be(()=>{q.disconnect()});var s=Ue(),p=i(s);{var x=a=>{var c=Je(),h=U(c),E=i(h),G=i(E),T=V(i(G),2),O=i(T);Me(O,{});var ae=V(O,2);{var ve=_=>{we(_,{buttonLabel:"Export",get buttonIcon(){return ne},buttonClass:"inline-flex items-center px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2 transition-colors",dropdownClass:"w-48",children:W=>{var Z=Ge(),B=i(Z);B.__click=[De,r];var S=i(B),H=i(S);Ve(H,{class:"w-4 h-4"}),A(2),n(S),n(B);var L=V(B,2);L.__click=[Ye,r];var J=i(L),X=i(J);ne(X,{class:"w-4 h-4"}),A(2),n(J),n(L);var ee=V(L,2);ee.__click=[je,r];var oe=i(ee),pe=i(oe);Ae(pe,{class:"w-4 h-4"}),A(2),n(oe),n(ee),n(Z),v(W,Z)},$$slots:{default:!0}})};w(ae,_=>{l().isConnected&&l().controls&&l().controls.length>0&&_(ve)})}var re=V(ae,2),ue=i(re);Be(ue,{class:"w-5 h-5"}),n(re),n(T),n(G),n(E),n(h);var se=V(h,2),ge=i(se);{var fe=_=>{var I=Ie();v(_,I)},he=_=>{var I=Q(),W=U(I);{var Z=S=>{var H=Ze(),L=i(H),J=V(i(L),2),X=i(J,!0);n(J),n(L),n(H),D(()=>Y(X,l().isConnected?l().fieldSchema?"Loading controls...":"Loading schema...":"Connecting...")),v(S,H)},B=S=>{var H=Q(),L=U(H);ie(L,()=>t.children),v(S,H)};w(W,S=>{!l().isConnected||!l().controls||l().controls.length===0||!l().fieldSchema?S(Z):S(B,!1)},!0)}v(_,I)};w(ge,_=>{l().isSwitchingControlSet?_(fe):_(he,!1)})}n(se),v(a,c)},d=a=>{var c=Q(),h=U(c);ie(h,()=>t.children),v(a,c)};w(p,a=>{e().url.pathname!=="/setup"?a(x):a(d,!1)})}n(s),v(b,s),N(),g()}le(["click"]);export{et as component};
@@ -1 +1 @@
1
- import"../chunks/DsnmJJEf.js";import{i as u}from"../chunks/C5zWTfmV.js";import{p as h,f as g,a as l,J as v,b as d,e as _,c as s,r as a,s as x,K as o}from"../chunks/Cby0Z7eP.js";import{s as $,p}from"../chunks/BW_5Gp3K.js";const b={get error(){return p.error},get status(){return p.status}};$.updated.check;const i=b;var k=g("<h1> </h1> <p> </p>",1);function q(m,c){h(c,!1),u();var t=k(),r=l(t),n=s(r,!0);a(r);var e=x(r,2),f=s(e,!0);a(e),v(()=>{o(n,i.status),o(f,i.error?.message)}),d(m,t),_()}export{q as component};
1
+ import"../chunks/DsnmJJEf.js";import{i as u}from"../chunks/C5zWTfmV.js";import{p as h,f as g,a as l,J as v,b as d,e as _,c as s,r as a,s as x,K as o}from"../chunks/Cby0Z7eP.js";import{s as $,p}from"../chunks/DqFUUtys.js";const b={get error(){return p.error},get status(){return p.status}};$.updated.check;const i=b;var k=g("<h1> </h1> <p> </p>",1);function q(m,c){h(c,!1),u();var t=k(),r=l(t),n=s(r,!0);a(r);var e=x(r,2),f=s(e,!0);a(e),v(()=>{o(n,i.status),o(f,i.error?.message)}),d(m,t),_()}export{q as component};
@@ -1 +1 @@
1
- import"../chunks/DsnmJJEf.js";import"../chunks/C5zWTfmV.js";import{f,a as y,b as n,c as t,r,s as w,n as C}from"../chunks/Cby0Z7eP.js";import{i as k,s as $,a as D}from"../chunks/DqsOU3kV.js";import{C as S,a as j,D as L}from"../chunks/BR-qttkK.js";import{s as N}from"../chunks/C9-IlOVE.js";var P=f('<div class=" h-full flex flex-col"><div class="flex-1 flex items-center justify-center p-8"><div class="text-center text-gray-500 dark:text-gray-400"><!> <h3 class="text-xl font-semibold text-gray-900 dark:text-white mb-2">No Control Selected</h3> <p class="text-gray-600 dark:text-gray-400">Select a control from the list to view and edit its details</p></div></div></div>'),q=f('<div class="w-1/2 flex flex-col"><div class="bg-white dark:bg-gray-900 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm h-full flex flex-col"><!></div></div> <div class="w-1/2 flex flex-col"><!></div>',1);function H(m){const o=()=>D(N,"$selectedControl",x),[x,g]=$();var l=q(),a=y(l),i=t(a),p=t(i);S(p,{}),r(i),r(a);var d=w(a,2),_=t(d);{var h=e=>{j(e,{get control(){return o()}})},b=e=>{var s=P(),c=t(s),v=t(c),u=t(v);L(u,{class:"mx-auto h-16 w-16 mb-4"}),C(4),r(v),r(c),r(s),n(e,s)};k(_,e=>{o()?e(h):e(b,!1)})}r(d),n(m,l),g()}export{H as component};
1
+ import"../chunks/DsnmJJEf.js";import"../chunks/C5zWTfmV.js";import{f,a as y,b as n,c as t,r,s as w,n as C}from"../chunks/Cby0Z7eP.js";import{i as k,s as $,a as D}from"../chunks/DqsOU3kV.js";import{C as S,a as j,D as L}from"../chunks/DBYZiizR.js";import{s as N}from"../chunks/ByARpoaF.js";var P=f('<div class=" h-full flex flex-col"><div class="flex-1 flex items-center justify-center p-8"><div class="text-center text-gray-500 dark:text-gray-400"><!> <h3 class="text-xl font-semibold text-gray-900 dark:text-white mb-2">No Control Selected</h3> <p class="text-gray-600 dark:text-gray-400">Select a control from the list to view and edit its details</p></div></div></div>'),q=f('<div class="w-1/2 flex flex-col"><div class="bg-white dark:bg-gray-900 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm h-full flex flex-col"><!></div></div> <div class="w-1/2 flex flex-col"><!></div>',1);function H(m){const o=()=>D(N,"$selectedControl",x),[x,g]=$();var l=q(),a=y(l),i=t(a),p=t(i);S(p,{}),r(i),r(a);var d=w(a,2),_=t(d);{var h=e=>{j(e,{get control(){return o()}})},b=e=>{var s=P(),c=t(s),v=t(c),u=t(v);L(u,{class:"mx-auto h-16 w-16 mb-4"}),C(4),r(v),r(c),r(s),n(e,s)};k(_,e=>{o()?e(h):e(b,!1)})}r(d),n(m,l),g()}export{H as component};
@@ -1 +1 @@
1
- import"../chunks/DsnmJJEf.js";import{p as S,u as A,d as x,o as B,f as n,a as F,b as i,e as H,c as s,g as N,r as a,s as h,h as P,n as R}from"../chunks/Cby0Z7eP.js";import{a as C,i as b,s as U}from"../chunks/DqsOU3kV.js";import{g as V}from"../chunks/BW_5Gp3K.js";import{p as q,s as _}from"../chunks/C9-IlOVE.js";import{C as G,a as J,D as K}from"../chunks/BR-qttkK.js";import{w}from"../chunks/BtOhWAVU.js";var O=n('<div class="absolute top-4 right-4 z-10"><div class="w-8 h-8 flex items-center justify-center rounded-full bg-blue-100 dark:bg-blue-900/30" title="Loading control..."><svg class="w-5 h-5 text-blue-600 dark:text-blue-400 animate-spin" fill="none" viewBox="0 0 24 24"><circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle><path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path></svg></div></div>'),Q=n('<div class=" h-full flex flex-col"><div class="flex-1 flex items-center justify-center p-8"><div class="text-center text-gray-500 dark:text-gray-400"><!> <h3 class="text-xl font-semibold text-gray-900 dark:text-white mb-2">No Control Selected</h3> <p class="text-gray-600 dark:text-gray-400">Select a control from the list to view and edit its details</p></div></div></div>'),T=n('<div class="w-1/2 flex flex-col"><div class="bg-white dark:bg-gray-900 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm h-full flex flex-col"><!></div></div> <div class="w-1/2 flex flex-col relative"><!> <!></div>',1);function se(y,k){S(k,!0);const $=()=>C(q,"$page",c),d=()=>C(_,"$selectedControl",c),[c,D]=U();let v="",r=N(!1);A(()=>{const e=$().params.id;if(!e)return;const t=decodeURIComponent(e);t&&t!==v&&w.isConnected()&&(v=t,x(r,!0),w.getControlDetails(t))}),B(()=>{const e=t=>{const o=t.detail;x(r,!1),o?_.set(o):V("/")};return window.addEventListener("control-details",e),()=>{window.removeEventListener("control-details",e)}});var f=T(),l=F(f),p=s(l),L=s(p);G(L,{}),a(p),a(l);var g=h(l,2),m=s(g);{var I=e=>{var t=O();i(e,t)};b(m,e=>{P(r)&&e(I)})}var z=h(m,2);{var j=e=>{J(e,{get control(){return d()}})},E=e=>{var t=Q(),o=s(t),u=s(o),M=s(u);K(M,{class:"mx-auto h-16 w-16 mb-4"}),R(4),a(u),a(o),a(t),i(e,t)};b(z,e=>{d()?e(j):e(E,!1)})}a(g),i(y,f),H(),D()}export{se as component};
1
+ import"../chunks/DsnmJJEf.js";import{p as S,u as A,d as x,o as B,f as n,a as F,b as i,e as H,c as s,g as N,r as a,s as h,h as P,n as R}from"../chunks/Cby0Z7eP.js";import{a as C,i as b,s as U}from"../chunks/DqsOU3kV.js";import{g as V}from"../chunks/DqFUUtys.js";import{p as q,s as _}from"../chunks/ByARpoaF.js";import{C as G,a as J,D as K}from"../chunks/DBYZiizR.js";import{w}from"../chunks/BtOhWAVU.js";var O=n('<div class="absolute top-4 right-4 z-10"><div class="w-8 h-8 flex items-center justify-center rounded-full bg-blue-100 dark:bg-blue-900/30" title="Loading control..."><svg class="w-5 h-5 text-blue-600 dark:text-blue-400 animate-spin" fill="none" viewBox="0 0 24 24"><circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle><path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path></svg></div></div>'),Q=n('<div class=" h-full flex flex-col"><div class="flex-1 flex items-center justify-center p-8"><div class="text-center text-gray-500 dark:text-gray-400"><!> <h3 class="text-xl font-semibold text-gray-900 dark:text-white mb-2">No Control Selected</h3> <p class="text-gray-600 dark:text-gray-400">Select a control from the list to view and edit its details</p></div></div></div>'),T=n('<div class="w-1/2 flex flex-col"><div class="bg-white dark:bg-gray-900 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm h-full flex flex-col"><!></div></div> <div class="w-1/2 flex flex-col relative"><!> <!></div>',1);function se(y,k){S(k,!0);const $=()=>C(q,"$page",c),d=()=>C(_,"$selectedControl",c),[c,D]=U();let v="",r=N(!1);A(()=>{const e=$().params.id;if(!e)return;const t=decodeURIComponent(e);t&&t!==v&&w.isConnected()&&(v=t,x(r,!0),w.getControlDetails(t))}),B(()=>{const e=t=>{const o=t.detail;x(r,!1),o?_.set(o):V("/")};return window.addEventListener("control-details",e),()=>{window.removeEventListener("control-details",e)}});var f=T(),l=F(f),p=s(l),L=s(p);G(L,{}),a(p),a(l);var g=h(l,2),m=s(g);{var I=e=>{var t=O();i(e,t)};b(m,e=>{P(r)&&e(I)})}var z=h(m,2);{var j=e=>{J(e,{get control(){return d()}})},E=e=>{var t=Q(),o=s(t),u=s(o),M=s(u);K(M,{class:"mx-auto h-16 w-16 mb-4"}),R(4),a(u),a(o),a(t),i(e,t)};b(z,e=>{d()?e(j):e(E,!1)})}a(g),i(y,f),H(),D()}export{se as component};
@@ -1,4 +1,4 @@
1
- import"../chunks/DsnmJJEf.js";import{i as Ye}from"../chunks/C5zWTfmV.js";import{p as We,ae as Me,d as l,T as C,G as te,h as e,af as ar,ag as Ve,c as a,n as _e,r,b as d,e as Xe,J as H,K as P,ah as Vr,f as m,s as n,k as h,ai as b,a as we,aj as xr,ak as Ie,a0 as $e,al as hr,am as rr,an as yr,o as Gr,W as Qr}from"../chunks/Cby0Z7eP.js";import{l as tr,p as Ue,i as z,a as et,s as rt}from"../chunks/DqsOU3kV.js";import{a as Tr,s as de,r as Er,e as pe,i as De,b as qe,c as Ir,d as _r,f as Cr,w as wr}from"../chunks/BtOhWAVU.js";import{g as tt}from"../chunks/BW_5Gp3K.js";var at=Ve("<title> </title>"),ot=Ve('<svg><!><path d="M11 18L12.41 19.41 15 16.83 15 29 17 29 17 16.83 19.59 19.41 21 18 16 13 11 18z"></path><path d="M23.5,22H23V20h.5a4.5,4.5,0,0,0,.36-9L23,11l-.1-.82a7,7,0,0,0-13.88,0L9,11,8.14,11a4.5,4.5,0,0,0,.36,9H9v2H8.5A6.5,6.5,0,0,1,7.2,9.14a9,9,0,0,1,17.6,0A6.5,6.5,0,0,1,23.5,22Z"></path></svg>');function lt(fe,K){const F=tr(K,["children","$$slots","$$events","$$legacy"]),se=tr(F,["size","title"]);We(K,!1);const V=C(),T=C();let N=Ue(K,"size",8,16),x=Ue(K,"title",8,void 0);Me(()=>(te(F),te(x())),()=>{l(V,F["aria-label"]||F["aria-labelledby"]||x())}),Me(()=>(e(V),te(F)),()=>{l(T,{"aria-hidden":e(V)?void 0:!0,role:e(V)?"img":void 0,focusable:Number(F.tabindex)===0?!0:void 0})}),ar(),Ye();var f=ot();Tr(f,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:N(),height:N(),...e(T),...se}));var S=a(f);{var ie=R=>{var c=at(),q=a(c,!0);r(c),H(()=>P(q,x())),d(R,c)};z(S,R=>{x()&&R(ie)})}_e(2),r(f),d(fe,f),Xe()}var st=Ve("<title> </title>"),it=Ve('<svg><!><path d="M10 6H14V10H10zM18 6H22V10H18zM10 14H14V18H10zM18 14H22V18H18zM10 22H14V26H10zM18 22H22V26H18z"></path></svg>');function kr(fe,K){const F=tr(K,["children","$$slots","$$events","$$legacy"]),se=tr(F,["size","title"]);We(K,!1);const V=C(),T=C();let N=Ue(K,"size",8,16),x=Ue(K,"title",8,void 0);Me(()=>(te(F),te(x())),()=>{l(V,F["aria-label"]||F["aria-labelledby"]||x())}),Me(()=>(e(V),te(F)),()=>{l(T,{"aria-hidden":e(V)?void 0:!0,role:e(V)?"img":void 0,focusable:Number(F.tabindex)===0?!0:void 0})}),ar(),Ye();var f=it();Tr(f,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:N(),height:N(),...e(T),...se}));var S=a(f);{var ie=R=>{var c=st(),q=a(c,!0);r(c),H(()=>P(q,x())),d(R,c)};z(S,R=>{x()&&R(ie)})}_e(),r(f),d(fe,f),Xe()}var nt=m('<div class="p-4 text-sm text-blue-800 rounded-lg bg-blue-50 dark:bg-gray-800 dark:text-blue-400"><div class="flex items-center"><svg class="flex-shrink-0 inline w-4 h-4 mr-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20"><path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"></path></svg> <div><span class="font-medium">File loaded:</span> <div class="mt-1"><span class="font-medium">Sheets:</span> <span class="font-medium">Fields:</span> <span class="font-medium">Controls found:</span> </div></div></div></div>'),dt=m('<div class="p-4 text-sm text-red-800 rounded-lg bg-red-50 dark:bg-gray-800 dark:text-red-400"><div class="flex items-center"><svg class="flex-shrink-0 inline w-4 h-4 mr-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20"><path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"></path></svg> <span> </span></div></div>'),vt=m('<div class="p-4 text-sm text-green-800 rounded-lg bg-green-50 dark:bg-gray-800 dark:text-green-400"><div class="flex items-center"><svg class="flex-shrink-0 inline w-4 h-4 mr-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20"><path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"></path></svg> <span> </span></div></div>'),ct=m("<option> </option>"),gt=m("<option> </option>"),pt=m("<option> </option>"),ut=m('<span class="ml-auto text-xs text-blue-600 dark:text-blue-400">ID</span>'),bt=m('<div draggable="true" role="button" tabindex="0" class="flex items-center px-3 py-2 bg-gray-100 dark:bg-gray-700 text-gray-500 dark:text-gray-400 rounded text-sm cursor-move hover:bg-gray-200 dark:hover:bg-gray-600 transition-colors opacity-75"><svg class="w-3 h-3 mr-2 flex-shrink-0" fill="currentColor" viewBox="0 0 20 20"><path d="M10 6a2 2 0 110-4 2 2 0 010 4zM10 12a2 2 0 110-4 2 2 0 010 4zM10 18a2 2 0 110-4 2 2 0 010 4z"></path></svg> <span class="truncate line-through"> </span> <!></div>'),ft=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">No excluded fields</p>'),mt=m('<div draggable="true" role="button" tabindex="0"><!> <span class="truncate"> </span></div>'),xt=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">Drop fields here</p>'),ht=m('<div draggable="true" role="button" tabindex="0"><!> <span class="truncate"> </span></div>'),yt=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">Drop fields here</p>'),_t=m('<div draggable="true" role="button" tabindex="0"><!> <span class="truncate"> </span></div>'),wt=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">Drop fields here</p>'),kt=m('<th class="px-4 py-2"> </th>'),Ct=m('<td class="px-4 py-2"> </td>'),St=m('<tr class="bg-white border-b dark:bg-gray-800 dark:border-gray-700"></tr>'),$t=m('<div class="bg-gradient-to-br from-gray-50 to-gray-100 dark:from-gray-800 dark:to-gray-900 rounded-lg p-4 border border-gray-200 dark:border-gray-700"><h3 class="text-lg font-semibold text-gray-900 dark:text-white mb-4">Sample Data Preview</h3> <div class="overflow-x-auto"><table class="w-full text-sm text-left text-gray-500 dark:text-gray-400"><thead class="text-xs text-gray-700 uppercase bg-gray-100 dark:bg-gray-600 dark:text-gray-400"><tr></tr></thead><tbody></tbody></table></div></div>'),Dt=m('<span class="flex items-center"><svg class="animate-spin -ml-1 mr-3 h-5 w-5 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"><circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle><path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path></svg> Importing...</span>'),Mt=m(`<div class="bg-gradient-to-br from-gray-50 to-gray-100 dark:from-gray-800 dark:to-gray-900 rounded-lg p-4 border border-gray-200 dark:border-gray-700"><h3 class="text-lg font-semibold text-gray-900 dark:text-white mb-4">Import Options</h3> <div class="grid grid-cols-1 md:grid-cols-2 gap-4 mb-4"><div><label for="controlSetName" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Control Set Name <span class="text-red-500">*</span></label> <input type="text" id="controlSetName" placeholder="e.g., NIST 800-53 Rev 4" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white" required/> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">This will be used as the display name and folder name</p></div> <div><label for="controlSetDescription" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Description</label> <input type="text" id="controlSetDescription" placeholder="Optional description" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white"/> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Brief description of this control set</p></div></div> <div class="grid grid-cols-1 md:grid-cols-2 gap-4"><div><label for="sheet" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Sheet</label> <select id="sheet" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white"></select> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Select which worksheet contains your control data</p></div> <div><label for="headerRow" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Select Header Row</label> <select id="headerRow" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white"></select> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Select the row containing column headers</p></div> <div><label for="controlIdField" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Control ID Field <span class="text-red-500">*</span></label> <select id="controlIdField" required><option disabled>Select Control ID field</option><!></select> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Column containing unique control identifiers (e.g., AC-1, SC-7)</p></div></div></div> <div class="bg-gradient-to-br from-gray-50 to-gray-100 dark:from-gray-800 dark:to-gray-900 rounded-lg p-4 border border-gray-200 dark:border-gray-700"><h3 class="text-lg font-semibold text-gray-900 dark:text-white mb-2">Organize Fields</h3> <p class="text-sm text-gray-600 dark:text-gray-400 mb-4">Drag fields to organize them. <strong>Overview fields</strong> will appear as table columns in
1
+ import"../chunks/DsnmJJEf.js";import{i as Ye}from"../chunks/C5zWTfmV.js";import{p as We,ae as Me,d as l,T as C,G as te,h as e,af as ar,ag as Ve,c as a,n as _e,r,b as d,e as Xe,J as H,K as P,ah as Vr,f as m,s as n,k as h,ai as b,a as we,aj as xr,ak as Ie,a0 as $e,al as hr,am as rr,an as yr,o as Gr,W as Qr}from"../chunks/Cby0Z7eP.js";import{l as tr,p as Ue,i as z,a as et,s as rt}from"../chunks/DqsOU3kV.js";import{a as Tr,s as de,r as Er,e as pe,i as De,b as qe,c as Ir,d as _r,f as Cr,w as wr}from"../chunks/BtOhWAVU.js";import{g as tt}from"../chunks/DqFUUtys.js";var at=Ve("<title> </title>"),ot=Ve('<svg><!><path d="M11 18L12.41 19.41 15 16.83 15 29 17 29 17 16.83 19.59 19.41 21 18 16 13 11 18z"></path><path d="M23.5,22H23V20h.5a4.5,4.5,0,0,0,.36-9L23,11l-.1-.82a7,7,0,0,0-13.88,0L9,11,8.14,11a4.5,4.5,0,0,0,.36,9H9v2H8.5A6.5,6.5,0,0,1,7.2,9.14a9,9,0,0,1,17.6,0A6.5,6.5,0,0,1,23.5,22Z"></path></svg>');function lt(fe,K){const F=tr(K,["children","$$slots","$$events","$$legacy"]),se=tr(F,["size","title"]);We(K,!1);const V=C(),T=C();let N=Ue(K,"size",8,16),x=Ue(K,"title",8,void 0);Me(()=>(te(F),te(x())),()=>{l(V,F["aria-label"]||F["aria-labelledby"]||x())}),Me(()=>(e(V),te(F)),()=>{l(T,{"aria-hidden":e(V)?void 0:!0,role:e(V)?"img":void 0,focusable:Number(F.tabindex)===0?!0:void 0})}),ar(),Ye();var f=ot();Tr(f,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:N(),height:N(),...e(T),...se}));var S=a(f);{var ie=R=>{var c=at(),q=a(c,!0);r(c),H(()=>P(q,x())),d(R,c)};z(S,R=>{x()&&R(ie)})}_e(2),r(f),d(fe,f),Xe()}var st=Ve("<title> </title>"),it=Ve('<svg><!><path d="M10 6H14V10H10zM18 6H22V10H18zM10 14H14V18H10zM18 14H22V18H18zM10 22H14V26H10zM18 22H22V26H18z"></path></svg>');function kr(fe,K){const F=tr(K,["children","$$slots","$$events","$$legacy"]),se=tr(F,["size","title"]);We(K,!1);const V=C(),T=C();let N=Ue(K,"size",8,16),x=Ue(K,"title",8,void 0);Me(()=>(te(F),te(x())),()=>{l(V,F["aria-label"]||F["aria-labelledby"]||x())}),Me(()=>(e(V),te(F)),()=>{l(T,{"aria-hidden":e(V)?void 0:!0,role:e(V)?"img":void 0,focusable:Number(F.tabindex)===0?!0:void 0})}),ar(),Ye();var f=it();Tr(f,()=>({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 32 32",fill:"currentColor",preserveAspectRatio:"xMidYMid meet",width:N(),height:N(),...e(T),...se}));var S=a(f);{var ie=R=>{var c=st(),q=a(c,!0);r(c),H(()=>P(q,x())),d(R,c)};z(S,R=>{x()&&R(ie)})}_e(),r(f),d(fe,f),Xe()}var nt=m('<div class="p-4 text-sm text-blue-800 rounded-lg bg-blue-50 dark:bg-gray-800 dark:text-blue-400"><div class="flex items-center"><svg class="flex-shrink-0 inline w-4 h-4 mr-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20"><path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"></path></svg> <div><span class="font-medium">File loaded:</span> <div class="mt-1"><span class="font-medium">Sheets:</span> <span class="font-medium">Fields:</span> <span class="font-medium">Controls found:</span> </div></div></div></div>'),dt=m('<div class="p-4 text-sm text-red-800 rounded-lg bg-red-50 dark:bg-gray-800 dark:text-red-400"><div class="flex items-center"><svg class="flex-shrink-0 inline w-4 h-4 mr-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20"><path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"></path></svg> <span> </span></div></div>'),vt=m('<div class="p-4 text-sm text-green-800 rounded-lg bg-green-50 dark:bg-gray-800 dark:text-green-400"><div class="flex items-center"><svg class="flex-shrink-0 inline w-4 h-4 mr-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20"><path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"></path></svg> <span> </span></div></div>'),ct=m("<option> </option>"),gt=m("<option> </option>"),pt=m("<option> </option>"),ut=m('<span class="ml-auto text-xs text-blue-600 dark:text-blue-400">ID</span>'),bt=m('<div draggable="true" role="button" tabindex="0" class="flex items-center px-3 py-2 bg-gray-100 dark:bg-gray-700 text-gray-500 dark:text-gray-400 rounded text-sm cursor-move hover:bg-gray-200 dark:hover:bg-gray-600 transition-colors opacity-75"><svg class="w-3 h-3 mr-2 flex-shrink-0" fill="currentColor" viewBox="0 0 20 20"><path d="M10 6a2 2 0 110-4 2 2 0 010 4zM10 12a2 2 0 110-4 2 2 0 010 4zM10 18a2 2 0 110-4 2 2 0 010 4z"></path></svg> <span class="truncate line-through"> </span> <!></div>'),ft=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">No excluded fields</p>'),mt=m('<div draggable="true" role="button" tabindex="0"><!> <span class="truncate"> </span></div>'),xt=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">Drop fields here</p>'),ht=m('<div draggable="true" role="button" tabindex="0"><!> <span class="truncate"> </span></div>'),yt=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">Drop fields here</p>'),_t=m('<div draggable="true" role="button" tabindex="0"><!> <span class="truncate"> </span></div>'),wt=m('<p class="text-xs text-gray-400 dark:text-gray-500 text-center py-4">Drop fields here</p>'),kt=m('<th class="px-4 py-2"> </th>'),Ct=m('<td class="px-4 py-2"> </td>'),St=m('<tr class="bg-white border-b dark:bg-gray-800 dark:border-gray-700"></tr>'),$t=m('<div class="bg-gradient-to-br from-gray-50 to-gray-100 dark:from-gray-800 dark:to-gray-900 rounded-lg p-4 border border-gray-200 dark:border-gray-700"><h3 class="text-lg font-semibold text-gray-900 dark:text-white mb-4">Sample Data Preview</h3> <div class="overflow-x-auto"><table class="w-full text-sm text-left text-gray-500 dark:text-gray-400"><thead class="text-xs text-gray-700 uppercase bg-gray-100 dark:bg-gray-600 dark:text-gray-400"><tr></tr></thead><tbody></tbody></table></div></div>'),Dt=m('<span class="flex items-center"><svg class="animate-spin -ml-1 mr-3 h-5 w-5 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"><circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle><path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path></svg> Importing...</span>'),Mt=m(`<div class="bg-gradient-to-br from-gray-50 to-gray-100 dark:from-gray-800 dark:to-gray-900 rounded-lg p-4 border border-gray-200 dark:border-gray-700"><h3 class="text-lg font-semibold text-gray-900 dark:text-white mb-4">Import Options</h3> <div class="grid grid-cols-1 md:grid-cols-2 gap-4 mb-4"><div><label for="controlSetName" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Control Set Name <span class="text-red-500">*</span></label> <input type="text" id="controlSetName" placeholder="e.g., NIST 800-53 Rev 4" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white" required/> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">This will be used as the display name and folder name</p></div> <div><label for="controlSetDescription" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Description</label> <input type="text" id="controlSetDescription" placeholder="Optional description" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white"/> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Brief description of this control set</p></div></div> <div class="grid grid-cols-1 md:grid-cols-2 gap-4"><div><label for="sheet" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Sheet</label> <select id="sheet" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white"></select> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Select which worksheet contains your control data</p></div> <div><label for="headerRow" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Select Header Row</label> <select id="headerRow" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-600 dark:border-gray-500 dark:text-white"></select> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Select the row containing column headers</p></div> <div><label for="controlIdField" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">Control ID Field <span class="text-red-500">*</span></label> <select id="controlIdField" required><option disabled>Select Control ID field</option><!></select> <p class="mt-1 text-xs text-gray-500 dark:text-gray-400">Column containing unique control identifiers (e.g., AC-1, SC-7)</p></div></div></div> <div class="bg-gradient-to-br from-gray-50 to-gray-100 dark:from-gray-800 dark:to-gray-900 rounded-lg p-4 border border-gray-200 dark:border-gray-700"><h3 class="text-lg font-semibold text-gray-900 dark:text-white mb-2">Organize Fields</h3> <p class="text-sm text-gray-600 dark:text-gray-400 mb-4">Drag fields to organize them. <strong>Overview fields</strong> will appear as table columns in
2
2
  the controls list.</p> <div class="grid grid-cols-1 lg:grid-cols-4 gap-4"><div class="border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800"><div class="p-3 border-b border-gray-200 dark:border-gray-700 bg-gray-50 dark:bg-gray-700 rounded-t-lg"><h4 class="text-sm font-semibold text-gray-700 dark:text-gray-300">Excluded Fields</h4> <p class="text-xs text-gray-500 dark:text-gray-400 mt-1">Not imported</p></div> <div role="region" aria-label="Excluded fields drop zone"><!> <!></div></div> <div class="border border-blue-300 dark:border-blue-700 rounded-lg bg-white dark:bg-gray-800"><div class="p-3 border-b border-blue-200 dark:border-blue-800 bg-blue-50 dark:bg-blue-900/20 rounded-t-lg"><h4 class="text-sm font-semibold text-blue-700 dark:text-blue-300">Overview Tab</h4> <p class="text-xs text-blue-600 dark:text-blue-400 mt-1">Shows in details & table columns</p></div> <div role="region" aria-label="Overview tab drop zone"><!> <!></div></div> <div class="border border-green-300 dark:border-green-700 rounded-lg bg-white dark:bg-gray-800"><div class="p-3 border-b border-green-200 dark:border-green-800 bg-green-50 dark:bg-green-900/20 rounded-t-lg"><h4 class="text-sm font-semibold text-green-700 dark:text-green-300">Implementation Tab</h4> <p class="text-xs text-green-600 dark:text-green-400 mt-1">Status & compliance</p></div> <div role="region" aria-label="Implementation tab drop zone"><!> <!></div></div> <div class="border border-purple-300 dark:border-purple-700 rounded-lg bg-white dark:bg-gray-800"><div class="p-3 border-b border-purple-200 dark:border-purple-800 bg-purple-50 dark:bg-purple-900/20 rounded-t-lg"><h4 class="text-sm font-semibold text-purple-700 dark:text-purple-300">Custom Tab</h4> <p class="text-xs text-purple-600 dark:text-purple-400 mt-1">Additional fields</p></div> <div role="region" aria-label="Custom fields drop zone"><!> <!></div></div></div></div> <!> <div class="flex justify-center"><button class="px-5 py-2.5 text-white bg-blue-600 hover:bg-blue-700 focus:ring-4 focus:ring-blue-300 font-medium rounded-lg text-sm disabled:opacity-50 disabled:cursor-not-allowed"><!></button></div>`,1),zt=m('<div class="space-y-6"><div role="button" tabindex="0" class="relative"><label><div class="flex flex-col items-center justify-center pt-5 pb-6"><!> <p class="mb-2 text-sm text-gray-500 dark:text-gray-400"><span class="font-semibold">Click to upload</span> or drag and drop</p> <p class="text-xs text-gray-500 dark:text-gray-400">XLSX, XLS or CSV files</p></div> <input type="file" class="hidden" accept=".xlsx,.xls,.csv"/></label></div> <!> <!> <!> <!></div>');function At(fe,K){We(K,!1);const F=C(),se=Vr();let V=C(null),T=C(""),N=C(""),x=C([]),f=C([]),S=C([]),ie=C(0),R=C([]),c=C(new Map),q=C(1),j=C(""),ue=C(""),U=C(""),Y=C(!1),W=C(""),ve=C(""),ye=C(!1),ae=C(!1),E=C(null),X=C(null),oe=C(null);function ke(){l(f,[]),l(S,[]),l(ie,0),e(c).clear(),l(c,new Map),l(j,""),l(W,""),l(ve,""),l(E,null),l(X,null),l(oe,null)}function p(t){t.preventDefault(),l(ae,!0)}function u(){l(ae,!1)}function $(t){t.preventDefault(),l(ae,!1);const i=t.dataTransfer?.files;i&&i.length>0&&I(i[0])}function L(t){const i=t.target;i.files&&i.files.length>0&&I(i.files[0])}async function I(t){ke(),l(T,t.name),l(V,t),l(W,""),l(Y,!0),l(ue,e(T).replace(/\.[^.]+$/,"").replace(/[-_]/g," ")),l(U,`Imported from ${e(T)}`);try{const i=new FormData;i.append("file",t);const y=await fetch("/api/parse-excel",{method:"POST",body:i});if(!y.ok){const D=await y.json();throw new Error(D.error||"Failed to parse file")}const _=await y.json();l(x,_.sheets||[]),l(N,_.selectedSheet||e(x)[0]),l(R,_.rowPreviews||[]),e(R).length>0&&e(q)===1&&l(q,e(R)[0].row),await B(),l(ye,!0)}catch(i){l(W,"Error reading file: "+i.message)}finally{l(Y,!1)}}async function B(){if(!(!e(V)||!e(N))){l(Y,!0),e(c).clear(),l(c,new Map),l(j,"");try{const t=new FormData;t.append("file",e(V)),t.append("sheetName",e(N)),t.append("headerRow",e(q).toString());const i=await fetch("/api/parse-excel-sheet",{method:"POST",body:t});if(!i.ok){const _=await i.json();throw new Error(_.error||"Failed to parse sheet")}const y=await i.json();if(l(f,y.fields||[]),l(S,y.sampleData||[]),l(ie,y.controlCount||0),e(f).forEach((_,D)=>{const v=_.toLowerCase();let O="custom",Z="text";v.includes("implementation")||v.includes("status")||v.includes("narrative")||v.includes("guidance")?O="implementation":(v.includes("id")||v.includes("title")||v.includes("family")||v.includes("cci")||v.includes("control")||v.includes("acronym"))&&(O="overview"),v.includes("description")||v.includes("narrative")||v.includes("guidance")||v.includes("statement")?Z="textarea":v.includes("status")||v.includes("type")||v.includes("designation")?Z="select":v.includes("date")?Z="date":(v.includes("count")||v.includes("number"))&&(Z="number"),e(c).set(_,{originalName:_,tab:O,displayOrder:D,fieldType:Z,required:v.includes("id")||v.includes("title")})}),l(c,e(c)),e(j)&&!e(f).includes(e(j))&&l(j,""),!e(j)&&e(f).includes("AP Acronym")){const _=!e(S).length||e(S).every(Q=>!Q["AP Acronym"]||String(Q["AP Acronym"]).length<25),D=e(S).map(Q=>Q["AP Acronym"]).filter(Q=>Q!=null&&Q!==""&&String(Q).trim()!==""),v=new Set(D),O=!D.length||v.size===D.length,Z=D.length>0;_&&O&&Z&&l(j,"AP Acronym")}}catch(t){l(W,"Error loading sheet data: "+t.message)}finally{l(Y,!1)}}}function ce(t){if(!t)return t;let i=t.trim().replace(/\r?\n/g," ").replace(/\s+/g," ").trim();return Te(i)}function Te(t){return t.replace(/\W+/g," ").split(/ |\s/).map(i=>i.toLowerCase()).join("-")}function be(t,i){l(E,i),t.dataTransfer&&(t.dataTransfer.effectAllowed="move",t.dataTransfer.setData("text/plain",i))}function le(){l(E,null),l(X,null),l(oe,null)}function M(t,i){t.preventDefault(),l(X,i),t.dataTransfer&&(t.dataTransfer.dropEffect="move")}function ge(){l(X,null)}function ne(t,i,y){if(t.preventDefault(),e(E)&&e(c).has(e(E))){const _=e(c).get(e(E));if(_.tab=i,y!==void 0&&i!==null){const v=Array.from(e(c).entries()).filter(([O,Z])=>Z.tab===i).sort((O,Z)=>O[1].displayOrder-Z[1].displayOrder).filter(([O])=>O!==e(E));v.splice(y,0,[e(E),_]),v.forEach(([O,Z],Q)=>{Z.displayOrder=Q,e(c).set(O,Z)})}else if(i!==null){const D=Math.max(0,...Array.from(e(c).values()).filter(v=>v.tab===i).map(v=>v.displayOrder));_.displayOrder=D+1}e(c).set(e(E),_),l(c,e(c))}l(E,null),l(X,null)}function ze(t,i){t.preventDefault(),t.stopPropagation(),l(oe,i)}function Ce(){l(oe,null)}function Se(t,i,y){if(t.preventDefault(),t.stopPropagation(),e(E)&&e(E)!==i){const D=Array.from(e(c).entries()).filter(([v,O])=>O.tab===y).sort((v,O)=>v[1].displayOrder-O[1].displayOrder).findIndex(([v])=>v===i);D!==-1&&ne(t,y,D)}l(oe,null)}async function Le(){if(!(!e(V)||!e(T))){if(!e(j)){l(W,"Please select a Control ID field before importing"),l(ve,"");return}if(!e(ue)||e(ue).trim()===""){l(W,"Please enter a Control Set Name before importing"),l(ve,"");return}l(Y,!0),l(W,""),l(ve,"");try{const t=new FormData;t.append("file",e(V),e(T)),t.append("controlIdField",e(j)),t.append("startRow",e(q).toString()),t.append("namingConvention","kebab-case"),t.append("skipEmpty","true"),t.append("skipEmptyRows","true"),t.append("controlSetName",e(ue)||e(T).replace(/\.[^.]+$/,"")),t.append("controlSetDescription",e(U)||`Imported from ${e(T)}`);const i=Array.from(e(c).entries()).filter(([D,v])=>v.tab!==null).map(([D,v])=>({fieldName:ce(D),...v}));t.append("fieldSchema",JSON.stringify(i));const y=await fetch("/api/import-spreadsheet",{method:"POST",body:t});if(!y.ok){const D=await y.json();throw new Error(D.error||"Import failed")}const _=await y.json();l(ve,`Successfully imported ${_.controlCount} controls into ${_.families.length} families`),se("created",{path:_.outputDir})}catch(t){l(W,"Error importing spreadsheet: "+t.message)}finally{l(Y,!1)}}}Me(()=>e(f),()=>{l(F,e(f).filter(t=>e(f).includes(t)))}),ar(),Ye();var Ne=zt(),me=a(Ne),je=a(me),Ae=a(je),or=a(Ae);lt(or,{class:"w-8 h-8 mb-4 text-gray-500 dark:text-gray-400"}),_e(4),r(Ae);var Je=n(Ae,2);r(je),r(me);var Fe=n(me,2);{var lr=t=>{var i=nt(),y=a(i),_=n(a(y),2),D=n(a(_)),v=n(D),O=n(a(v)),Z=n(O,2),Q=n(Z,2);r(v),r(_),r(y),r(i),H(()=>{P(D,` ${e(T)??""} `),P(O,` ${e(x),h(()=>e(x).length)??""} | `),P(Z,` ${e(f),h(()=>e(f).length)??""} | `),P(Q,` ${e(ie)??""}`)}),d(t,i)};z(Fe,t=>{e(T)&&t(lr)})}var Ke=n(Fe,2);{var sr=t=>{var i=dt(),y=a(i),_=n(a(y),2),D=a(_,!0);r(_),r(y),r(i),H(()=>P(D,e(W))),d(t,i)};z(Ke,t=>{e(W)&&t(sr)})}var Be=n(Ke,2);{var ir=t=>{var i=vt(),y=a(i),_=n(a(y),2),D=a(_,!0);r(_),r(y),r(i),H(()=>P(D,e(ve))),d(t,i)};z(Be,t=>{e(ve)&&t(ir)})}var nr=n(Be,2);{var dr=t=>{var i=Mt(),y=we(i),_=n(a(y),2),D=a(_),v=n(a(D),2);Er(v),_e(2),r(D);var O=n(D,2),Z=n(a(O),2);Er(Z),_e(2),r(O),r(_);var Q=n(_,2),vr=a(Q),Ge=n(a(vr),2);H(()=>{e(N),xr(()=>{e(x)})}),pe(Ge,5,()=>e(x),De,(o,s)=>{var w=ct(),k=a(w,!0);r(w);var g={};H(()=>{P(k,e(s)),g!==(g=e(s))&&(w.value=(w.__value=e(s))??"")}),d(o,w)}),r(Ge),_e(2),r(vr);var cr=n(vr,2),Qe=n(a(cr),2);H(()=>{e(q),xr(()=>{e(R)})}),pe(Qe,5,()=>e(R),De,(o,s)=>{var w=gt(),k=a(w);r(w);var g={};H(()=>{P(k,`Row ${e(s),h(()=>e(s).row)??""}: ${e(s),h(()=>e(s).preview)??""}`),g!==(g=(e(s),h(()=>e(s).row)))&&(w.value=(w.__value=(e(s),h(()=>e(s).row)))??"")}),d(o,w)}),r(Qe),_e(2),r(cr);var Sr=n(cr,2),er=n(a(Sr),2);H(()=>{e(j),xr(()=>{e(f),e(S)})});var gr=a(er);gr.value=gr.__value="";var Lr=n(gr);pe(Lr,1,()=>e(f),De,(o,s)=>{const w=$e(()=>(e(S),e(s),h(()=>e(S).length>0&&e(S)[0][e(s)]?String(e(S)[0][e(s)]).slice(0,30):""))),k=$e(()=>(e(S),e(s),h(()=>!e(S).length||e(S).every(re=>!re[e(s)]||String(re[e(s)]).length<25)))),g=$e(()=>(e(S),e(s),h(()=>e(S).map(re=>re[e(s)]).filter(re=>re!=null&&re!==""&&String(re).trim()!=="")))),G=$e(()=>new Set(e(g))),ee=$e(()=>(te(e(g)),te(e(G)),h(()=>!e(g).length||e(G).size===e(g).length))),J=$e(()=>(te(e(g)),h(()=>e(g).length>0)));var A=Ie(),xe=we(A);{var Re=re=>{var he=pt(),mr=a(he);r(he);var Pr={};H(()=>{P(mr,`${e(s)??""}${e(w)?` (e.g., ${e(w)})`:""}`),Pr!==(Pr=e(s))&&(he.value=(he.__value=e(s))??"")}),d(re,he)};z(xe,re=>{e(k)&&e(ee)&&e(J)&&re(Re)})}d(o,A)}),r(er),_e(2),r(Sr),r(Q),r(y);var pr=n(y,2),$r=n(a(pr),4),ur=a($r),Oe=n(a(ur),2),Dr=a(Oe);pe(Dr,1,()=>(e(f),e(c),h(()=>e(f).filter(o=>!e(c).get(o)||e(c).get(o)?.tab===null))),o=>o,(o,s)=>{var w=bt(),k=n(a(w),2),g=a(k,!0);r(k);var G=n(k,2);{var ee=J=>{var A=ut();d(J,A)};z(G,J=>{e(s)===e(j)&&J(ee)})}r(w),H(()=>{qe(w,"aria-label",`Drag ${e(s)??""} field`),P(g,e(s))}),b("dragstart",w,J=>be(J,e(s))),b("dragend",w,le),d(o,w)});var Nr=n(Dr,2);{var jr=o=>{var s=ft();d(o,s)};z(Nr,o=>{e(f),e(c),h(()=>e(f).filter(s=>!e(c).get(s)||e(c).get(s)?.tab===null).length===0)&&o(jr)})}r(Oe),r(ur);var br=n(ur,2),He=n(a(br),2),Mr=a(He);pe(Mr,3,()=>(e(c),h(()=>Array.from(e(c).entries()).filter(([o,s])=>s.tab==="overview").sort((o,s)=>o[1].displayOrder-s[1].displayOrder))),([o,s])=>o,(o,s)=>{var w=hr(()=>yr(e(s),2));let k=()=>e(w)[0];var g=mt(),G=a(g);kr(G,{class:"w-3 h-3 mr-2 flex-shrink-0"});var ee=n(G,2),J=a(ee,!0);r(ee),r(g),H(()=>{qe(g,"aria-label",`${k()??""} field in Overview tab`),de(g,1,`flex items-center px-3 py-2 bg-blue-100 dark:bg-blue-900/30 text-blue-800 dark:text-blue-300 rounded text-sm cursor-move hover:bg-blue-200 dark:hover:bg-blue-800/30 transition-colors
3
3
  ${e(oe)===k()&&e(E)!==k()?"border-t-2 border-blue-500":""}`),P(J,k())}),b("dragstart",g,A=>be(A,k())),b("dragend",g,le),b("dragover",g,A=>ze(A,k())),b("dragleave",g,Ce),b("drop",g,A=>Se(A,k(),"overview")),d(o,g)});var Br=n(Mr,2);{var Zr=o=>{var s=xt();d(o,s)};z(Br,o=>{e(c),h(()=>Array.from(e(c).entries()).filter(([s,w])=>w.tab==="overview").length===0)&&o(Zr)})}r(He),r(br);var fr=n(br,2),Pe=n(a(fr),2),zr=a(Pe);pe(zr,3,()=>(e(c),h(()=>Array.from(e(c).entries()).filter(([o,s])=>s.tab==="implementation").sort((o,s)=>o[1].displayOrder-s[1].displayOrder))),([o,s])=>o,(o,s)=>{var w=hr(()=>yr(e(s),2));let k=()=>e(w)[0];var g=ht(),G=a(g);kr(G,{class:"w-3 h-3 mr-2 flex-shrink-0"});var ee=n(G,2),J=a(ee,!0);r(ee),r(g),H(()=>{qe(g,"aria-label",`${k()??""} field in Implementation tab`),de(g,1,`flex items-center px-3 py-2 bg-green-100 dark:bg-green-900/30 text-green-800 dark:text-green-300 rounded text-sm cursor-move hover:bg-green-200 dark:hover:bg-green-800/30 transition-colors
4
4
  ${e(oe)===k()&&e(E)!==k()?"border-t-2 border-green-500":""}`),P(J,k())}),b("dragstart",g,A=>be(A,k())),b("dragend",g,le),b("dragover",g,A=>ze(A,k())),b("dragleave",g,Ce),b("drop",g,A=>Se(A,k(),"implementation")),d(o,g)});var Rr=n(zr,2);{var qr=o=>{var s=yt();d(o,s)};z(Rr,o=>{e(c),h(()=>Array.from(e(c).entries()).filter(([s,w])=>w.tab==="implementation").length===0)&&o(qr)})}r(Pe),r(fr);var Ar=n(fr,2),Ee=n(a(Ar),2),Fr=a(Ee);pe(Fr,3,()=>(e(c),h(()=>Array.from(e(c).entries()).filter(([o,s])=>s.tab==="custom").sort((o,s)=>o[1].displayOrder-s[1].displayOrder))),([o,s])=>o,(o,s)=>{var w=hr(()=>yr(e(s),2));let k=()=>e(w)[0];var g=_t(),G=a(g);kr(G,{class:"w-3 h-3 mr-2 flex-shrink-0"});var ee=n(G,2),J=a(ee,!0);r(ee),r(g),H(()=>{qe(g,"aria-label",`${k()??""} field in Custom tab`),de(g,1,`flex items-center px-3 py-2 bg-purple-100 dark:bg-purple-900/30 text-purple-800 dark:text-purple-300 rounded text-sm cursor-move hover:bg-purple-200 dark:hover:bg-purple-800/30 transition-colors
@@ -1 +1 @@
1
- {"version":"1757982650827"}
1
+ {"version":"1758050641523"}
@@ -1,8 +1,32 @@
1
1
  // cli/commands/crawl.ts
2
2
  import fs from "fs";
3
3
  import { Octokit } from "@octokit/rest";
4
- import { Command } from "commander";
4
+ import { Command, Option } from "commander";
5
5
  import { createHash } from "crypto";
6
+ var closingBody = `
7
+
8
+ ---
9
+
10
+ <sub>**Tip:** Customize your compliance reviews with <a href="https://github.com/defenseunicorns/lula.git" class="Link--inTextBlock" target="_blank" rel="noopener noreferrer">Lula</a>.</sub>`;
11
+ async function postFinding(params) {
12
+ const { octokit, postMode, owner, repo, pull_number, body } = params;
13
+ if (postMode === "comment") {
14
+ await octokit.issues.createComment({
15
+ owner,
16
+ repo,
17
+ issue_number: pull_number,
18
+ body
19
+ });
20
+ return;
21
+ }
22
+ await octokit.pulls.createReview({
23
+ owner,
24
+ repo,
25
+ pull_number,
26
+ body,
27
+ event: "REQUEST_CHANGES"
28
+ });
29
+ }
6
30
  function getPRContext() {
7
31
  const fallbackOwner = process.env.OWNER;
8
32
  const fallbackRepo = process.env.REPO;
@@ -83,16 +107,27 @@ function getChangedBlocks(oldText, newText) {
83
107
  return changed;
84
108
  }
85
109
  function crawlCommand() {
86
- return new Command().command("crawl").description("Detect compliance-related changes between @lulaStart and @lulaEnd in PR files").action(async () => {
110
+ return new Command().command("crawl").description("Detect compliance-related changes between @lulaStart and @lulaEnd in PR files").addOption(
111
+ new Option("--post-mode <mode>", "How to post findings").choices(["review", "comment"]).default("review")
112
+ ).action(async (opts) => {
87
113
  const { owner, repo, pull_number } = getPRContext();
88
- console.log(`Analyzing PR #${pull_number} in ${owner}/${repo} for compliance changes...
89
- `);
114
+ console.log(`Analyzing PR #${pull_number} in ${owner}/${repo} for compliance changes...`);
90
115
  const octokit = new Octokit({ auth: process.env.GITHUB_TOKEN });
91
116
  const pr = await octokit.pulls.get({ owner, repo, pull_number });
92
117
  const prBranch = pr.data.head.ref;
93
118
  const { data: files } = await octokit.pulls.listFiles({ owner, repo, pull_number });
119
+ let commentBody = `## Lula Compliance Overview
120
+
121
+ Please review the changes to ensure they meet compliance standards.
122
+
123
+ ### Reviewed Changes
124
+
125
+ Lula reviewed ${files.length} files changed that affect compliance.
126
+
127
+ `;
94
128
  for (const file of files) {
95
129
  if (file.status === "added") continue;
130
+ console.log(`Commenting regarding \`${file.filename}\`.`);
96
131
  try {
97
132
  const [oldText, newText] = await Promise.all([
98
133
  fetchRawFileViaAPI({ octokit, owner, repo, path: file.filename, ref: "main" }),
@@ -100,28 +135,43 @@ function crawlCommand() {
100
135
  ]);
101
136
  const changedBlocks = getChangedBlocks(oldText, newText);
102
137
  for (const block of changedBlocks) {
138
+ commentBody += `
139
+
140
+ ---
141
+ | File | Lines Changed |
142
+ | ---- | ------------- |
143
+ `;
103
144
  const newBlockText = newText.split("\n").slice(block.startLine, block.endLine).join("\n");
104
145
  const blockSha256 = createHash("sha256").update(newBlockText).digest("hex");
105
- const commentBody = `**Compliance Alert**:\`${file.filename}\` changed between lines ${block.startLine + 1}\u2013${block.endLine}.
106
- UUID \`${block.uuid}\` may be out of compliance.
107
- SHA-256 of block contents: \`${blockSha256}\`.
108
-
109
- Please review the changes to ensure they meet compliance standards.
146
+ commentBody += `| \`${file.filename}\` | \`${block.startLine + 1}\u2013${block.endLine}\` |
147
+ > **uuid**-\`${block.uuid}\`
148
+ **sha256** \`${blockSha256}\`
110
149
 
111
150
  `;
112
- console.log(`Commenting on ${file.filename}: ${commentBody}`);
113
- await octokit.pulls.createReview({
114
- owner,
115
- repo,
116
- pull_number,
117
- body: commentBody,
118
- event: "REQUEST_CHANGES"
119
- });
120
151
  }
121
152
  } catch (err) {
122
153
  console.error(`Error processing ${file.filename}: ${err}`);
123
154
  }
124
155
  }
156
+ if (files.length > 0) {
157
+ await postFinding({
158
+ octokit,
159
+ postMode: opts.postMode,
160
+ owner,
161
+ repo,
162
+ pull_number,
163
+ body: commentBody + closingBody
164
+ });
165
+ const header = `Posted (${opts.postMode})`;
166
+ const underline = "-".repeat(header.length);
167
+ console.log(`
168
+ ${header}
169
+ ${underline}
170
+
171
+ ${commentBody + closingBody}
172
+
173
+ `);
174
+ }
125
175
  });
126
176
  }
127
177
  export {
@@ -129,5 +179,6 @@ export {
129
179
  extractMapBlocks,
130
180
  fetchRawFileViaAPI,
131
181
  getChangedBlocks,
132
- getPRContext
182
+ getPRContext,
183
+ postFinding
133
184
  };
package/dist/index.html CHANGED
@@ -6,10 +6,10 @@
6
6
  <link rel="icon" href="/lula.png" />
7
7
  <meta name="viewport" content="width=device-width, initial-scale=1" />
8
8
 
9
- <link rel="modulepreload" href="/_app/immutable/entry/start.DTGsPX6N.js">
10
- <link rel="modulepreload" href="/_app/immutable/chunks/BW_5Gp3K.js">
9
+ <link rel="modulepreload" href="/_app/immutable/entry/start.CBELVcNd.js">
10
+ <link rel="modulepreload" href="/_app/immutable/chunks/DqFUUtys.js">
11
11
  <link rel="modulepreload" href="/_app/immutable/chunks/Cby0Z7eP.js">
12
- <link rel="modulepreload" href="/_app/immutable/entry/app.B1mUL4qo.js">
12
+ <link rel="modulepreload" href="/_app/immutable/entry/app.D7eWHMdQ.js">
13
13
  <link rel="modulepreload" href="/_app/immutable/chunks/DsnmJJEf.js">
14
14
  <link rel="modulepreload" href="/_app/immutable/chunks/DqsOU3kV.js">
15
15
  <link rel="modulepreload" href="/_app/immutable/chunks/CoF2vljD.js">
@@ -19,15 +19,15 @@
19
19
  <div style="display: contents">
20
20
  <script>
21
21
  {
22
- __sveltekit_ft8ml8 = {
22
+ __sveltekit_nre5h8 = {
23
23
  base: ""
24
24
  };
25
25
 
26
26
  const element = document.currentScript.parentElement;
27
27
 
28
28
  Promise.all([
29
- import("/_app/immutable/entry/start.DTGsPX6N.js"),
30
- import("/_app/immutable/entry/app.B1mUL4qo.js")
29
+ import("/_app/immutable/entry/start.CBELVcNd.js"),
30
+ import("/_app/immutable/entry/app.D7eWHMdQ.js")
31
31
  ]).then(([kit, app]) => {
32
32
  kit.start(app, element);
33
33
  });
package/dist/index.js CHANGED
@@ -5204,8 +5204,32 @@ function getVersion() {
5204
5204
  // cli/commands/crawl.ts
5205
5205
  import fs4 from "fs";
5206
5206
  import { Octokit } from "@octokit/rest";
5207
- import { Command } from "commander";
5207
+ import { Command, Option } from "commander";
5208
5208
  import { createHash as createHash2 } from "crypto";
5209
+ var closingBody = `
5210
+
5211
+ ---
5212
+
5213
+ <sub>**Tip:** Customize your compliance reviews with <a href="https://github.com/defenseunicorns/lula.git" class="Link--inTextBlock" target="_blank" rel="noopener noreferrer">Lula</a>.</sub>`;
5214
+ async function postFinding(params) {
5215
+ const { octokit, postMode, owner, repo, pull_number, body } = params;
5216
+ if (postMode === "comment") {
5217
+ await octokit.issues.createComment({
5218
+ owner,
5219
+ repo,
5220
+ issue_number: pull_number,
5221
+ body
5222
+ });
5223
+ return;
5224
+ }
5225
+ await octokit.pulls.createReview({
5226
+ owner,
5227
+ repo,
5228
+ pull_number,
5229
+ body,
5230
+ event: "REQUEST_CHANGES"
5231
+ });
5232
+ }
5209
5233
  function getPRContext() {
5210
5234
  const fallbackOwner = process.env.OWNER;
5211
5235
  const fallbackRepo = process.env.REPO;
@@ -5286,16 +5310,27 @@ function getChangedBlocks(oldText, newText) {
5286
5310
  return changed;
5287
5311
  }
5288
5312
  function crawlCommand() {
5289
- return new Command().command("crawl").description("Detect compliance-related changes between @lulaStart and @lulaEnd in PR files").action(async () => {
5313
+ return new Command().command("crawl").description("Detect compliance-related changes between @lulaStart and @lulaEnd in PR files").addOption(
5314
+ new Option("--post-mode <mode>", "How to post findings").choices(["review", "comment"]).default("review")
5315
+ ).action(async (opts) => {
5290
5316
  const { owner, repo, pull_number } = getPRContext();
5291
- console.log(`Analyzing PR #${pull_number} in ${owner}/${repo} for compliance changes...
5292
- `);
5317
+ console.log(`Analyzing PR #${pull_number} in ${owner}/${repo} for compliance changes...`);
5293
5318
  const octokit = new Octokit({ auth: process.env.GITHUB_TOKEN });
5294
5319
  const pr = await octokit.pulls.get({ owner, repo, pull_number });
5295
5320
  const prBranch = pr.data.head.ref;
5296
5321
  const { data: files } = await octokit.pulls.listFiles({ owner, repo, pull_number });
5322
+ let commentBody = `## Lula Compliance Overview
5323
+
5324
+ Please review the changes to ensure they meet compliance standards.
5325
+
5326
+ ### Reviewed Changes
5327
+
5328
+ Lula reviewed ${files.length} files changed that affect compliance.
5329
+
5330
+ `;
5297
5331
  for (const file of files) {
5298
5332
  if (file.status === "added") continue;
5333
+ console.log(`Commenting regarding \`${file.filename}\`.`);
5299
5334
  try {
5300
5335
  const [oldText, newText] = await Promise.all([
5301
5336
  fetchRawFileViaAPI({ octokit, owner, repo, path: file.filename, ref: "main" }),
@@ -5303,28 +5338,43 @@ function crawlCommand() {
5303
5338
  ]);
5304
5339
  const changedBlocks = getChangedBlocks(oldText, newText);
5305
5340
  for (const block of changedBlocks) {
5341
+ commentBody += `
5342
+
5343
+ ---
5344
+ | File | Lines Changed |
5345
+ | ---- | ------------- |
5346
+ `;
5306
5347
  const newBlockText = newText.split("\n").slice(block.startLine, block.endLine).join("\n");
5307
5348
  const blockSha256 = createHash2("sha256").update(newBlockText).digest("hex");
5308
- const commentBody = `**Compliance Alert**:\`${file.filename}\` changed between lines ${block.startLine + 1}\u2013${block.endLine}.
5309
- UUID \`${block.uuid}\` may be out of compliance.
5310
- SHA-256 of block contents: \`${blockSha256}\`.
5311
-
5312
- Please review the changes to ensure they meet compliance standards.
5349
+ commentBody += `| \`${file.filename}\` | \`${block.startLine + 1}\u2013${block.endLine}\` |
5350
+ > **uuid**-\`${block.uuid}\`
5351
+ **sha256** \`${blockSha256}\`
5313
5352
 
5314
5353
  `;
5315
- console.log(`Commenting on ${file.filename}: ${commentBody}`);
5316
- await octokit.pulls.createReview({
5317
- owner,
5318
- repo,
5319
- pull_number,
5320
- body: commentBody,
5321
- event: "REQUEST_CHANGES"
5322
- });
5323
5354
  }
5324
5355
  } catch (err) {
5325
5356
  console.error(`Error processing ${file.filename}: ${err}`);
5326
5357
  }
5327
5358
  }
5359
+ if (files.length > 0) {
5360
+ await postFinding({
5361
+ octokit,
5362
+ postMode: opts.postMode,
5363
+ owner,
5364
+ repo,
5365
+ pull_number,
5366
+ body: commentBody + closingBody
5367
+ });
5368
+ const header = `Posted (${opts.postMode})`;
5369
+ const underline = "-".repeat(header.length);
5370
+ console.log(`
5371
+ ${header}
5372
+ ${underline}
5373
+
5374
+ ${commentBody + closingBody}
5375
+
5376
+ `);
5377
+ }
5328
5378
  });
5329
5379
  }
5330
5380
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "lula2",
3
- "version": "0.0.10-nightly.8",
3
+ "version": "0.1.0",
4
4
  "description": "A tool for managing compliance as code in your GitHub repositories.",
5
5
  "bin": {
6
6
  "lula2": "./dist/lula2"
@@ -33,6 +33,25 @@
33
33
  "!dist/**/*.test.js*",
34
34
  "!dist/**/*.test.d.ts*"
35
35
  ],
36
+ "scripts": {
37
+ "dev": "vite dev --port 5173",
38
+ "dev:api": "tsx --watch index.ts --debug ui --port 3000 --no-open-browser",
39
+ "dev:full": "concurrently \"npm run dev:api\" \"npm run dev\"",
40
+ "build": "npm run build:svelte && npm run build:cli && npm run postbuild:cli",
41
+ "build:svelte": "vite build",
42
+ "build:cli": "esbuild index.ts cli/**/*.ts --bundle --platform=node --target=node22 --format=esm --outdir=dist --external:express --external:commander --external:js-yaml --external:yaml --external:isomorphic-git --external:glob --external:open --external:ws --external:cors --external:multer --external:@octokit/rest --external:undici --external:exceljs --external:csv-parse",
43
+ "postbuild:cli": "cp cli-wrapper.mjs dist/lula2 && chmod +x dist/lula2",
44
+ "preview": "vite preview",
45
+ "prepare": "svelte-kit sync || echo ''",
46
+ "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json && tsc --noEmit",
47
+ "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch",
48
+ "format": "prettier --write 'src/**/*.{ts,js,svelte}' 'cli/**/*.ts' 'index.ts' 'tests/**/*.ts'",
49
+ "format:check": "prettier --check 'src/**/*.{ts,js,svelte}' 'cli/**/*.ts' 'index.ts' 'tests/**/*.ts'",
50
+ "lint": "prettier --check 'src/**/*.{ts,js,svelte}' 'cli/**/*.ts' 'index.ts' 'tests/**/*.ts' && eslint src cli",
51
+ "test": "npm run test:unit -- --run --coverage",
52
+ "test:integration": "vitest --config integration/vitest.config.integration.ts",
53
+ "test:unit": "vitest"
54
+ },
36
55
  "dependencies": {
37
56
  "@octokit/rest": "^22.0.0",
38
57
  "@types/ws": "^8.18.1",
@@ -104,23 +123,5 @@
104
123
  "main",
105
124
  "next"
106
125
  ]
107
- },
108
- "scripts": {
109
- "dev": "vite dev --port 5173",
110
- "dev:api": "tsx --watch index.ts --debug ui --port 3000 --no-open-browser",
111
- "dev:full": "concurrently \"npm run dev:api\" \"npm run dev\"",
112
- "build": "npm run build:svelte && npm run build:cli && npm run postbuild:cli",
113
- "build:svelte": "vite build",
114
- "build:cli": "esbuild index.ts cli/**/*.ts --bundle --platform=node --target=node22 --format=esm --outdir=dist --external:express --external:commander --external:js-yaml --external:yaml --external:isomorphic-git --external:glob --external:open --external:ws --external:cors --external:multer --external:@octokit/rest --external:undici --external:exceljs --external:csv-parse",
115
- "postbuild:cli": "cp cli-wrapper.mjs dist/lula2 && chmod +x dist/lula2",
116
- "preview": "vite preview",
117
- "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json && tsc --noEmit",
118
- "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch",
119
- "format": "prettier --write 'src/**/*.{ts,js,svelte}' 'cli/**/*.ts' 'index.ts' 'tests/**/*.ts'",
120
- "format:check": "prettier --check 'src/**/*.{ts,js,svelte}' 'cli/**/*.ts' 'index.ts' 'tests/**/*.ts'",
121
- "lint": "prettier --check 'src/**/*.{ts,js,svelte}' 'cli/**/*.ts' 'index.ts' 'tests/**/*.ts' && eslint src cli",
122
- "test": "npm run test:unit -- --run --coverage",
123
- "test:integration": "vitest --config integration/vitest.config.integration.ts",
124
- "test:unit": "vitest"
125
126
  }
126
- }
127
+ }
@@ -1 +0,0 @@
1
- import{l as o,a as r}from"../chunks/BW_5Gp3K.js";export{o as load_css,r as start};