From dcf9131707400a6ce4aebd85800631769c2d1afa Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 07:02:19 +0100 Subject: [PATCH 01/19] unused --- user_model.ml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/user_model.ml b/user_model.ml index dafdccc0..e737a795 100644 --- a/user_model.ml +++ b/user_model.ml @@ -547,7 +547,7 @@ let is_valid_cookie (cookie : cookie) now = let is_email_verified user = Option.is_some user.email_verified let password_validation password = String.length password >= 8 -let verify_email_token u uuid timestamp = +let verify_email_token u _uuid timestamp = match u with | None -> Logs.err (fun m -> m "email verification: Token couldn't be found."); From a375425aa6755f31cf611beaa75da75c3ed2fb02 Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 07:02:31 +0100 Subject: [PATCH 02/19] add chart.js to header --- header_layout.ml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/header_layout.ml b/header_layout.ml index 602f2ff5..8517a0a8 100644 --- a/header_layout.ml +++ b/header_layout.ml @@ -32,4 +32,6 @@ let header ?(page_title = "Mollymawk") ~icon () = "https://cdn.jsdelivr.net/npm/alpinejs@3.x.x/dist/cdn.min.js"; ] (txt ""); + (* chart.js is used for chart elements like pie charts. We use this to visualize usage data of policies*) + script ~a:[ a_src "https://cdn.jsdelivr.net/npm/chart.js" ] (txt ""); ]) From b8596c0bb41fafce5def32245e06a6cd4b0227fa Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 07:02:39 +0100 Subject: [PATCH 03/19] update styles --- assets/style.css | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/assets/style.css b/assets/style.css index 3f1c6963..56497af1 100755 --- a/assets/style.css +++ b/assets/style.css @@ -1 +1 @@ -/*! tailwindcss v3.4.4 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb}:after,:before{--tw-content:""}:host,html{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:initial}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:initial;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:initial}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dd,dl,figure,h1,h2,h3,h4,h5,h6,hr,p,pre{margin:0}fieldset{margin:0}fieldset,legend{padding:0}menu,ol,ul{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}[role=button],button{cursor:pointer}:disabled{cursor:default}audio,canvas,embed,iframe,img,object,svg,video{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]{display:none}*,::backdrop,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:#3b82f680;--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }.container{width:100%}@media (min-width:640px){.container{max-width:640px}}@media (min-width:768px){.container{max-width:768px}}@media (min-width:1024px){.container{max-width:1024px}}@media (min-width:1280px){.container{max-width:1280px}}@media (min-width:1536px){.container{max-width:1536px}}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border-width:0}.pointer-events-none{pointer-events:none}.pointer-events-auto{pointer-events:auto}.static{position:static}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.inset-0{inset:0}.inset-1{inset:.25rem}.-inset-x-12{left:-3rem;right:-3rem}.inset-x-1{left:.25rem;right:.25rem}.inset-x-4{left:1rem;right:1rem}.-bottom-0{bottom:0}.-bottom-0\.5{bottom:-.125rem}.-bottom-2{bottom:-.5rem}.-bottom-6{bottom:-1.5rem}.-left-1\/4{left:-25%}.-left-2{left:-.5rem}.-top-1\/4{top:-25%}.-top-4{top:-1rem}.-top-6{top:-1.5rem}.bottom-0{bottom:0}.bottom-10{bottom:2.5rem}.bottom-6{bottom:1.5rem}.left-0{left:0}.left-1\/2{left:50%}.left-8{left:2rem}.right-4{right:1rem}.top-0{top:0}.top-1\/4{top:25%}.z-50{z-index:50}.z-\[-1\]{z-index:-1}.z-\[500\]{z-index:500}.col-span-1{grid-column:span 1/span 1}.col-span-10{grid-column:span 10/span 10}.col-span-2{grid-column:span 2/span 2}.col-span-3{grid-column:span 3/span 3}.col-span-4{grid-column:span 4/span 4}.col-span-7{grid-column:span 7/span 7}.-m-1{margin:-.25rem}.-m-1\.5{margin:-.375rem}.m-auto{margin:auto}.mx-auto{margin-left:auto;margin-right:auto}.my-1{margin-top:.25rem;margin-bottom:.25rem}.my-10{margin-top:2.5rem;margin-bottom:2.5rem}.my-2{margin-top:.5rem;margin-bottom:.5rem}.my-3{margin-top:.75rem;margin-bottom:.75rem}.my-4{margin-top:1rem;margin-bottom:1rem}.my-5{margin-top:1.25rem;margin-bottom:1.25rem}.my-6{margin-top:1.5rem;margin-bottom:1.5rem}.-mb-12{margin-bottom:-3rem}.-mb-6{margin-bottom:-1.5rem}.-ml-2{margin-left:-.5rem}.-mr-2{margin-right:-.5rem}.-mr-2\.5{margin-right:-.625rem}.-mt-1{margin-top:-.25rem}.-mt-24{margin-top:-6rem}.-mt-3{margin-top:-.75rem}.mb-0{margin-bottom:0}.mb-14{margin-bottom:3.5rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-5{margin-bottom:1.25rem}.mb-7{margin-bottom:1.75rem}.mb-8{margin-bottom:2rem}.mb-9{margin-bottom:2.25rem}.mb-auto{margin-bottom:auto}.mb-px{margin-bottom:1px}.ml-1{margin-left:.25rem}.ml-5{margin-left:1.25rem}.mr-2{margin-right:.5rem}.mr-3{margin-right:.75rem}.mt-1{margin-top:.25rem}.mt-1\.5{margin-top:.375rem}.mt-10{margin-top:2.5rem}.mt-12{margin-top:3rem}.mt-16{margin-top:4rem}.mt-2{margin-top:.5rem}.mt-4{margin-top:1rem}.mt-8{margin-top:2rem}.mt-auto{margin-top:auto}.ml-3{margin-left:.75rem}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.table{display:table}.grid{display:grid}.hidden{display:none}.size-2{width:.5rem;height:.5rem}.size-2\.5{width:.625rem;height:.625rem}.h-12{height:3rem}.h-2{height:.5rem}.h-2\.5{height:.625rem}.h-24{height:6rem}.h-32{height:8rem}.h-4{height:1rem}.h-6{height:1.5rem}.h-8{height:2rem}.h-96{height:24rem}.h-\[150\%\]{height:150%}.h-full{height:100%}.h-px{height:1px}.max-h-screen{max-height:100vh}.min-h-screen{min-height:100vh}.w-0{width:0}.w-14{width:3.5rem}.w-16{width:4rem}.w-32{width:8rem}.w-4{width:1rem}.w-6{width:1.5rem}.w-60{width:15rem}.w-\[150\%\]{width:150%}.w-fit{width:-moz-fit-content;width:fit-content}.w-full{width:100%}.w-max{width:-moz-max-content;width:max-content}.w-px{width:1px}.w-8{width:2rem}.w-1\/2{width:50%}.min-w-full{min-width:100%}.max-w-3xl{max-width:48rem}.max-w-5xl{max-width:64rem}.max-w-6xl{max-width:72rem}.max-w-7xl{max-width:80rem}.max-w-\[94\%\]{max-width:94%}.max-w-full{max-width:100%}.max-w-lg{max-width:32rem}.max-w-md{max-width:28rem}.max-w-none{max-width:none}.max-w-sm{max-width:24rem}.max-w-xl{max-width:36rem}.flex-none{flex:none}.shrink-0{flex-shrink:0}.table-auto{table-layout:auto}.border-collapse{border-collapse:collapse}.origin-bottom{transform-origin:bottom}.-translate-x-1\/2{--tw-translate-x:-50%}.-rotate-3,.-translate-x-1\/2{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-rotate-3{--tw-rotate:-3deg}.rotate-3{--tw-rotate:3deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-0{--tw-scale-x:0;--tw-scale-y:0}.scale-0,.scale-x-0{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-x-0{--tw-scale-x:0}.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform-gpu{transform:translate3d(var(--tw-translate-x),var(--tw-translate-y),0) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes spin{to{transform:rotate(1turn)}}.animate-spin{animation:spin 1s linear infinite}.cursor-pointer{cursor:pointer}.list-none{list-style-type:none}.appearance-none{-webkit-appearance:none;-moz-appearance:none;appearance:none}.grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.flex-row{flex-direction:row}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-start{align-items:flex-start}.items-end{align-items:flex-end}.items-center{align-items:center}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-items-center{justify-items:center}.gap-0{gap:0}.gap-0\.5{gap:.125rem}.gap-10{gap:2.5rem}.gap-12{gap:3rem}.gap-20{gap:5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-8{gap:2rem}.gap-x-2{-moz-column-gap:.5rem;column-gap:.5rem}.gap-x-8{-moz-column-gap:2rem;column-gap:2rem}.gap-y-16{row-gap:4rem}.gap-y-4{row-gap:1rem}.space-x-1>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.25rem*var(--tw-space-x-reverse));margin-left:calc(.25rem*(1 - var(--tw-space-x-reverse)))}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.5rem*var(--tw-space-x-reverse));margin-left:calc(.5rem*(1 - var(--tw-space-x-reverse)))}.space-x-20>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(5rem*var(--tw-space-x-reverse));margin-left:calc(5rem*(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1rem*var(--tw-space-x-reverse));margin-left:calc(1rem*(1 - var(--tw-space-x-reverse)))}.space-x-5>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1.25rem*var(--tw-space-x-reverse));margin-left:calc(1.25rem*(1 - var(--tw-space-x-reverse)))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem*var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.75rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem*var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem*var(--tw-space-y-reverse))}.space-y-5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.25rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.25rem*var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem*var(--tw-space-y-reverse))}.divide-y>:not([hidden])~:not([hidden]){--tw-divide-y-reverse:0;border-top-width:calc(1px*(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px*var(--tw-divide-y-reverse))}.divide-gray-200>:not([hidden])~:not([hidden]){--tw-divide-opacity:1;border-color:rgb(229 231 235/var(--tw-divide-opacity))}.justify-self-start{justify-self:start}.justify-self-end{justify-self:end}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-x-auto{overflow-x:auto}.whitespace-normal{white-space:normal}.whitespace-nowrap{white-space:nowrap}.text-wrap{text-wrap:wrap}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:1rem}.rounded-3xl{border-radius:1.5rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-md{border-radius:.375rem}.rounded-xl{border-radius:.75rem}.rounded-l-\[20px\]{border-top-left-radius:20px;border-bottom-left-radius:20px}.rounded-r-\[20px\]{border-top-right-radius:20px;border-bottom-right-radius:20px}.rounded-r-\[8px\]{border-top-right-radius:8px;border-bottom-right-radius:8px}.rounded-t-full{border-top-left-radius:9999px;border-top-right-radius:9999px}.border{border-width:1px}.border-x-0{border-left-width:0;border-right-width:0}.border-b{border-bottom-width:1px}.border-b-0{border-bottom-width:0}.border-b-2{border-bottom-width:2px}.border-l-0{border-left-width:0}.border-r-0{border-right-width:0}.border-t-0{border-top-width:0}.border-none{border-style:none}.border-\[\#D7DFE9\]{--tw-border-opacity:1;border-color:rgb(215 223 233/var(--tw-border-opacity))}.border-gray-400{--tw-border-opacity:1;border-color:rgb(156 163 175/var(--tw-border-opacity))}.border-primary-200{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.border-primary-400{--tw-border-opacity:1;border-color:rgb(78 179 161/var(--tw-border-opacity))}.border-primary-500{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.border-primary-600{--tw-border-opacity:1;border-color:rgb(40 121 110/var(--tw-border-opacity))}.border-primary-700{--tw-border-opacity:1;border-color:rgb(35 98 90/var(--tw-border-opacity))}.border-secondary-500{--tw-border-opacity:1;border-color:rgb(255 78 51/var(--tw-border-opacity))}.border-transparent{border-color:#0000}.border-y-secondary-300{--tw-border-opacity:1;border-top-color:rgb(255 170 157/var(--tw-border-opacity));border-bottom-color:rgb(255 170 157/var(--tw-border-opacity))}.bg-gray-100{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}.bg-gray-50{--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity))}.bg-gray-500\/25{background-color:#6b728040}.bg-primary-100{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.bg-primary-200{--tw-bg-opacity:1;background-color:rgb(171 228 214/var(--tw-bg-opacity))}.bg-primary-300{--tw-bg-opacity:1;background-color:rgb(122 206 189/var(--tw-bg-opacity))}.bg-primary-50{--tw-bg-opacity:1;background-color:rgb(243 250 249/var(--tw-bg-opacity))}.bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.bg-primary-800{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.bg-primary-950{--tw-bg-opacity:1;background-color:rgb(13 38 37/var(--tw-bg-opacity))}.bg-secondary-300{--tw-bg-opacity:1;background-color:rgb(255 170 157/var(--tw-bg-opacity))}.bg-secondary-500{--tw-bg-opacity:1;background-color:rgb(255 78 51/var(--tw-bg-opacity))}.bg-secondary-700{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.bg-transparent{background-color:initial}.bg-gray-200{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.bg-opacity-0{--tw-bg-opacity:0}.bg-gradient-to-b{background-image:linear-gradient(to bottom,var(--tw-gradient-stops))}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.bg-gradient-to-t{background-image:linear-gradient(to top,var(--tw-gradient-stops))}.from-gray-100{--tw-gradient-from:#f3f4f6 var(--tw-gradient-from-position);--tw-gradient-to:#f3f4f600 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-900\/50{--tw-gradient-from:#1f423e80 var(--tw-gradient-from-position);--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-950{--tw-gradient-from:#0d2625 var(--tw-gradient-from-position);--tw-gradient-to:#0d262500 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-transparent{--tw-gradient-from:#0000 var(--tw-gradient-from-position);--tw-gradient-to:#0000 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-10\%{--tw-gradient-from-position:10%}.via-primary-900{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-primary-900\/50{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e80 var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-10\%{--tw-gradient-via-position:10%}.to-primary-950{--tw-gradient-to:#0d2625 var(--tw-gradient-to-position)}.to-primary-950\/50{--tw-gradient-to:#0d262580 var(--tw-gradient-to-position)}.to-transparent{--tw-gradient-to:#0000 var(--tw-gradient-to-position)}.to-90\%{--tw-gradient-to-position:90%}.bg-\[length\:100px_auto\]{background-size:100px auto}.bg-cover{background-size:cover}.bg-clip-padding{background-clip:padding-box}.bg-center{background-position:50%}.fill-primary-400{fill:#4eb3a1}.object-contain{-o-object-fit:contain;object-fit:contain}.object-cover{-o-object-fit:cover;object-fit:cover}.object-bottom{-o-object-position:bottom;object-position:bottom}.p-0{padding:0}.p-1{padding:.25rem}.p-1\.5{padding:.375rem}.p-10{padding:2.5rem}.p-16{padding:4rem}.p-2{padding:.5rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.p-6{padding:1.5rem}.px-10{padding-left:2.5rem;padding-right:2.5rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-40{padding-left:10rem;padding-right:10rem}.px-5{padding-left:1.25rem;padding-right:1.25rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.px-7{padding-left:1.75rem;padding-right:1.75rem}.py-0{padding-top:0;padding-bottom:0}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-10{padding-top:2.5rem;padding-bottom:2.5rem}.py-16{padding-top:4rem;padding-bottom:4rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-2\.5{padding-top:.625rem;padding-bottom:.625rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-5{padding-top:1.25rem;padding-bottom:1.25rem}.py-6{padding-top:1.5rem;padding-bottom:1.5rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pb-12{padding-bottom:3rem}.pb-16{padding-bottom:4rem}.pb-3{padding-bottom:.75rem}.pb-3\.5{padding-bottom:.875rem}.pb-32{padding-bottom:8rem}.pb-8{padding-bottom:2rem}.pl-3{padding-left:.75rem}.pl-4{padding-left:1rem}.pr-3{padding-right:.75rem}.pr-7{padding-right:1.75rem}.pr-\[10px\]{padding-right:10px}.pt-12{padding-top:3rem}.pt-24{padding-top:6rem}.pt-4{padding-top:1rem}.text-left{text-align:left}.text-center{text-align:center}.text-right{text-align:right}.text-start{text-align:start}.align-middle{vertical-align:middle}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-5xl{font-size:3rem;line-height:1}.text-7xl{font-size:4.5rem;line-height:1}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-\[450\]{font-weight:450}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.lowercase{text-transform:lowercase}.leading-6{line-height:1.5rem}.leading-none{line-height:1}.leading-tight{line-height:1.25}.tracking-tight{letter-spacing:-.025em}.tracking-wider{letter-spacing:.05em}.text-gray-100{--tw-text-opacity:1;color:rgb(243 244 246/var(--tw-text-opacity))}.text-gray-200{--tw-text-opacity:1;color:rgb(229 231 235/var(--tw-text-opacity))}.text-gray-50{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.text-gray-500{--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity))}.text-gray-600{--tw-text-opacity:1;color:rgb(75 85 99/var(--tw-text-opacity))}.text-gray-700{--tw-text-opacity:1;color:rgb(55 65 81/var(--tw-text-opacity))}.text-gray-800{--tw-text-opacity:1;color:rgb(31 41 55/var(--tw-text-opacity))}.text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.text-primary-300{--tw-text-opacity:1;color:rgb(122 206 189/var(--tw-text-opacity))}.text-primary-400{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.text-primary-50{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.text-primary-500{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.text-primary-600{--tw-text-opacity:1;color:rgb(40 121 110/var(--tw-text-opacity))}.text-primary-800{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.text-primary-900{--tw-text-opacity:1;color:rgb(31 66 62/var(--tw-text-opacity))}.text-primary-950{--tw-text-opacity:1;color:rgb(13 38 37/var(--tw-text-opacity))}.text-secondary-300{--tw-text-opacity:1;color:rgb(255 170 157/var(--tw-text-opacity))}.text-secondary-50{--tw-text-opacity:1;color:rgb(255 243 241/var(--tw-text-opacity))}.text-secondary-500{--tw-text-opacity:1;color:rgb(255 78 51/var(--tw-text-opacity))}.text-secondary-700{--tw-text-opacity:1;color:rgb(200 38 13/var(--tw-text-opacity))}.accent-primary-500{accent-color:#369c8c}.opacity-0{opacity:0}.opacity-50{opacity:.5}.shadow{--tw-shadow:0 1px 3px 0 #0000001a,0 1px 2px -1px #0000001a;--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color)}.shadow,.shadow-lg{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px #0000001a,0 4px 6px -4px #0000001a;--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.shadow-md{--tw-shadow:0 4px 6px -1px #0000001a,0 2px 4px -2px #0000001a;--tw-shadow-colored:0 4px 6px -1px var(--tw-shadow-color),0 2px 4px -2px var(--tw-shadow-color)}.shadow-md,.shadow-sm{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 2px 0 #0000000d;--tw-shadow-colored:0 1px 2px 0 var(--tw-shadow-color)}.shadow-gray-800\/5{--tw-shadow-color:#1f29370d;--tw-shadow:var(--tw-shadow-colored)}.outline-0{outline-width:0}.ring-1{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.ring-inset{--tw-ring-inset:inset}.ring-gray-800\/\[\.075\]{--tw-ring-color:rgba(31,41,55,.075)}.ring-primary-100{--tw-ring-opacity:1;--tw-ring-color:rgb(213 242 235/var(--tw-ring-opacity))}.ring-primary-200{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.blur{--tw-blur:blur(8px)}.blur,.blur-3xl{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.blur-3xl{--tw-blur:blur(64px)}.brightness-0{--tw-brightness:brightness(0)}.brightness-0,.drop-shadow{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.drop-shadow{--tw-drop-shadow:drop-shadow(0 1px 2px #0000001a) drop-shadow(0 1px 1px #0000000f)}.drop-shadow-sm{--tw-drop-shadow:drop-shadow(0 1px 1px #0000000d)}.drop-shadow-sm,.invert{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.invert{--tw-invert:invert(100%)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur-sm{--tw-backdrop-blur:blur(4px)}.backdrop-blur-sm,.backdrop-blur-xl{-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-xl{--tw-backdrop-blur:blur(24px)}.transition{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.duration-500{transition-duration:.5s}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}.\[-webkit-mask-image\:linear-gradient\(to_bottom\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(180deg,#fff 75%,#fff0)}.\[-webkit-mask-image\:linear-gradient\(to_top\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(0deg,#fff 75%,#fff0)}.hover\:isolate:hover{isolation:isolate}.hover\:border-primary-200:hover{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.hover\:border-transparent:hover{border-color:#0000}.hover\:bg-gray-200:hover{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.hover\:bg-primary-100:hover{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.hover\:bg-primary-700:hover{--tw-bg-opacity:1;background-color:rgb(35 98 90/var(--tw-bg-opacity))}.hover\:bg-primary-800:hover{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.hover\:bg-secondary-100:hover{--tw-bg-opacity:1;background-color:rgb(255 227 223/var(--tw-bg-opacity))}.hover\:bg-secondary-700:hover{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.hover\:bg-opacity-50:hover{--tw-bg-opacity:0.5}.hover\:font-bold:hover{font-weight:700}.hover\:text-gray-50:hover{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.hover\:text-primary-400:hover{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.hover\:text-primary-50:hover{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.hover\:text-primary-500:hover{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.hover\:text-primary-700:hover{--tw-text-opacity:1;color:rgb(35 98 90/var(--tw-text-opacity))}.hover\:text-primary-800:hover{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.focus\:isolate:focus{isolation:isolate}.focus\:border-primary-300:focus{--tw-border-opacity:1;border-color:rgb(122 206 189/var(--tw-border-opacity))}.focus\:border-primary-500:focus{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.focus\:border-transparent:focus{border-color:#0000}.focus\:bg-opacity-50:focus{--tw-bg-opacity:0.5}.focus\:text-primary-800:focus{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.focus\:outline-none:focus{outline:2px solid #0000;outline-offset:2px}.focus\:ring-\[1px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-\[1px\]:focus,.focus\:ring-\[3px\]:focus{box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.focus\:ring-\[3px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-primary-200:focus{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:opacity-50:disabled{opacity:.5}.group\/btn:hover .group-hover\/btn\:w-2{width:.5rem}.group\/btn:hover .group-hover\/btn\:w-2\.5{width:.625rem}.group:hover .group-hover\:scale-100{--tw-scale-x:1;--tw-scale-y:1}.group:hover .group-hover\:scale-100,.group:hover .group-hover\:scale-x-100{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.group:hover .group-hover\:scale-x-100{--tw-scale-x:1}.group:hover .group-hover\:opacity-100,.group\/btn:hover .group-hover\/btn\:opacity-100{opacity:1}@media (min-width:640px){.sm\:left-14{left:3.5rem}.sm\:w-auto{width:auto}.sm\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.sm\:px-6{padding-left:1.5rem;padding-right:1.5rem}.sm\:pb-40{padding-bottom:10rem}.sm\:pl-20{padding-left:5rem}.sm\:text-3xl{font-size:1.875rem;line-height:2.25rem}.sm\:text-5xl{font-size:3rem;line-height:1}.sm\:text-sm{font-size:.875rem;line-height:1.25rem}}@media (min-width:768px){.md\:block{display:block}.md\:h-screen{height:100vh}.md\:w-16{width:4rem}.md\:w-20{width:5rem}.md\:w-24{width:6rem}.md\:max-w-2xl{max-width:42rem}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.md\:gap-16{gap:4rem}.md\:px-14{padding-left:3.5rem;padding-right:3.5rem}.md\:py-10{padding-top:2.5rem;padding-bottom:2.5rem}.md\:pb-52{padding-bottom:13rem}.md\:text-3xl{font-size:1.875rem;line-height:2.25rem}.md\:text-4xl{font-size:2.25rem;line-height:2.5rem}.md\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width:1024px){.lg\:left-20{left:5rem}.lg\:order-last{order:9999}.lg\:col-span-3{grid-column:span 3/span 3}.lg\:col-span-4{grid-column:span 4/span 4}.lg\:-my-11{margin-top:-2.75rem;margin-bottom:-2.75rem}.lg\:-mr-9{margin-right:-2.25rem}.lg\:mb-0{margin-bottom:0}.lg\:mb-16{margin-bottom:4rem}.lg\:mt-0{margin-top:0}.lg\:block{display:block}.lg\:flex{display:flex}.lg\:grid{display:grid}.lg\:max-w-4xl{max-width:56rem}.lg\:max-w-7xl{max-width:80rem}.lg\:max-w-max{max-width:-moz-max-content;max-width:max-content}.lg\:max-w-xl{max-width:36rem}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-7{grid-template-columns:repeat(7,minmax(0,1fr))}.lg\:gap-5{gap:1.25rem}.lg\:p-10{padding:2.5rem}.lg\:px-8{padding-left:2rem;padding-right:2rem}.lg\:py-12{padding-top:3rem;padding-bottom:3rem}.lg\:py-24{padding-top:6rem;padding-bottom:6rem}.lg\:pb-20{padding-bottom:5rem}.lg\:pb-60{padding-bottom:15rem}.lg\:pl-28{padding-left:7rem}.lg\:pt-20{padding-top:5rem}.lg\:pt-40{padding-top:10rem}.lg\:text-4xl{font-size:2.25rem;line-height:2.5rem}.lg\:text-5xl{font-size:3rem;line-height:1}.lg\:text-6xl{font-size:3.75rem;line-height:1}.lg\:text-lg{font-size:1.125rem;line-height:1.75rem}}@media (min-width:1280px){.xl\:col-span-3{grid-column:span 3/span 3}.xl\:mx-auto{margin-left:auto;margin-right:auto}.xl\:flex{display:flex}.xl\:hidden{display:none}.xl\:w-auto{width:auto}.xl\:w-full{width:100%}.xl\:max-w-lg{max-width:32rem}.xl\:max-w-xl{max-width:36rem}.xl\:gap-16{gap:4rem}.xl\:gap-x-16{-moz-column-gap:4rem;column-gap:4rem}.xl\:px-20{padding-left:5rem;padding-right:5rem}.xl\:py-20{padding-top:5rem;padding-bottom:5rem}.xl\:py-32{padding-top:8rem}.xl\:pb-32,.xl\:py-32{padding-bottom:8rem}.xl\:pb-\[16\.5rem\]{padding-bottom:16.5rem}.xl\:text-xl{font-size:1.25rem;line-height:1.75rem}} +/*! tailwindcss v3.4.4 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb}:after,:before{--tw-content:""}:host,html{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:initial}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:initial;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:initial}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dd,dl,figure,h1,h2,h3,h4,h5,h6,hr,p,pre{margin:0}fieldset{margin:0}fieldset,legend{padding:0}menu,ol,ul{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}[role=button],button{cursor:pointer}:disabled{cursor:default}audio,canvas,embed,iframe,img,object,svg,video{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]{display:none}*,::backdrop,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:#3b82f680;--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }.container{width:100%}@media (min-width:640px){.container{max-width:640px}}@media (min-width:768px){.container{max-width:768px}}@media (min-width:1024px){.container{max-width:1024px}}@media (min-width:1280px){.container{max-width:1280px}}@media (min-width:1536px){.container{max-width:1536px}}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border-width:0}.pointer-events-none{pointer-events:none}.pointer-events-auto{pointer-events:auto}.visible{visibility:visible}.static{position:static}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.inset-0{inset:0}.inset-1{inset:.25rem}.-inset-x-12{left:-3rem;right:-3rem}.inset-x-1{left:.25rem;right:.25rem}.inset-x-4{left:1rem;right:1rem}.-bottom-0{bottom:0}.-bottom-0\.5{bottom:-.125rem}.-bottom-2{bottom:-.5rem}.-bottom-6{bottom:-1.5rem}.-left-1\/4{left:-25%}.-left-2{left:-.5rem}.-top-1\/4{top:-25%}.-top-4{top:-1rem}.-top-6{top:-1.5rem}.bottom-0{bottom:0}.bottom-10{bottom:2.5rem}.bottom-6{bottom:1.5rem}.left-0{left:0}.left-1\/2{left:50%}.left-8{left:2rem}.right-4{right:1rem}.top-0{top:0}.top-1\/4{top:25%}.z-30{z-index:30}.z-50{z-index:50}.z-\[-1\]{z-index:-1}.z-\[500\]{z-index:500}.col-span-1{grid-column:span 1/span 1}.col-span-10{grid-column:span 10/span 10}.col-span-2{grid-column:span 2/span 2}.col-span-3{grid-column:span 3/span 3}.col-span-4{grid-column:span 4/span 4}.col-span-7{grid-column:span 7/span 7}.-m-1{margin:-.25rem}.-m-1\.5{margin:-.375rem}.m-auto{margin:auto}.mx-auto{margin-left:auto;margin-right:auto}.my-1{margin-top:.25rem;margin-bottom:.25rem}.my-10{margin-top:2.5rem;margin-bottom:2.5rem}.my-2{margin-top:.5rem;margin-bottom:.5rem}.my-3{margin-top:.75rem;margin-bottom:.75rem}.my-4{margin-top:1rem;margin-bottom:1rem}.my-5{margin-top:1.25rem;margin-bottom:1.25rem}.my-6{margin-top:1.5rem;margin-bottom:1.5rem}.-mb-12{margin-bottom:-3rem}.-mb-6{margin-bottom:-1.5rem}.-ml-2{margin-left:-.5rem}.-mr-2{margin-right:-.5rem}.-mr-2\.5{margin-right:-.625rem}.-mt-1{margin-top:-.25rem}.-mt-24{margin-top:-6rem}.-mt-3{margin-top:-.75rem}.mb-0{margin-bottom:0}.mb-14{margin-bottom:3.5rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-5{margin-bottom:1.25rem}.mb-7{margin-bottom:1.75rem}.mb-8{margin-bottom:2rem}.mb-9{margin-bottom:2.25rem}.mb-auto{margin-bottom:auto}.mb-px{margin-bottom:1px}.ml-1{margin-left:.25rem}.ml-3{margin-left:.75rem}.ml-5{margin-left:1.25rem}.mr-2{margin-right:.5rem}.mr-3{margin-right:.75rem}.mt-1{margin-top:.25rem}.mt-1\.5{margin-top:.375rem}.mt-10{margin-top:2.5rem}.mt-12{margin-top:3rem}.mt-16{margin-top:4rem}.mt-2{margin-top:.5rem}.mt-4{margin-top:1rem}.mt-8{margin-top:2rem}.mt-auto{margin-top:auto}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.table{display:table}.grid{display:grid}.hidden{display:none}.size-2{width:.5rem;height:.5rem}.size-2\.5{width:.625rem;height:.625rem}.h-12{height:3rem}.h-2{height:.5rem}.h-2\.5{height:.625rem}.h-24{height:6rem}.h-32{height:8rem}.h-4{height:1rem}.h-5{height:1.25rem}.h-6{height:1.5rem}.h-8{height:2rem}.h-96{height:24rem}.h-\[150\%\]{height:150%}.h-full{height:100%}.h-px{height:1px}.max-h-screen{max-height:100vh}.min-h-screen{min-height:100vh}.w-0{width:0}.w-11{width:2.75rem}.w-14{width:3.5rem}.w-16{width:4rem}.w-32{width:8rem}.w-4{width:1rem}.w-5{width:1.25rem}.w-6{width:1.5rem}.w-60{width:15rem}.w-\[150\%\]{width:150%}.w-fit{width:-moz-fit-content;width:fit-content}.w-full{width:100%}.w-max{width:-moz-max-content;width:max-content}.w-px{width:1px}.min-w-full{min-width:100%}.max-w-3xl{max-width:48rem}.max-w-5xl{max-width:64rem}.max-w-6xl{max-width:72rem}.max-w-7xl{max-width:80rem}.max-w-\[94\%\]{max-width:94%}.max-w-full{max-width:100%}.max-w-lg{max-width:32rem}.max-w-md{max-width:28rem}.max-w-none{max-width:none}.max-w-sm{max-width:24rem}.max-w-xl{max-width:36rem}.max-w-4xl{max-width:56rem}.max-w-2xl{max-width:42rem}.flex-none{flex:none}.shrink-0{flex-shrink:0}.table-auto{table-layout:auto}.border-collapse{border-collapse:collapse}.origin-bottom{transform-origin:bottom}.-translate-x-1\/2{--tw-translate-x:-50%}.-rotate-3,.-translate-x-1\/2{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-rotate-3{--tw-rotate:-3deg}.rotate-3{--tw-rotate:3deg}.rotate-3,.scale-0{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-0{--tw-scale-x:0;--tw-scale-y:0}.scale-100{--tw-scale-x:1;--tw-scale-y:1}.scale-100,.scale-50{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-50{--tw-scale-x:.5;--tw-scale-y:.5}.scale-x-0{--tw-scale-x:0}.scale-x-0,.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform-gpu{transform:translate3d(var(--tw-translate-x),var(--tw-translate-y),0) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes spin{to{transform:rotate(1turn)}}.animate-spin{animation:spin 1s linear infinite}.cursor-pointer{cursor:pointer}.list-none{list-style-type:none}.appearance-none{-webkit-appearance:none;-moz-appearance:none;appearance:none}.grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.flex-row{flex-direction:row}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-start{align-items:flex-start}.items-end{align-items:flex-end}.items-center{align-items:center}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-items-center{justify-items:center}.gap-0{gap:0}.gap-0\.5{gap:.125rem}.gap-10{gap:2.5rem}.gap-12{gap:3rem}.gap-2{gap:.5rem}.gap-20{gap:5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-8{gap:2rem}.gap-x-2{-moz-column-gap:.5rem;column-gap:.5rem}.gap-x-8{-moz-column-gap:2rem;column-gap:2rem}.gap-y-16{row-gap:4rem}.gap-y-4{row-gap:1rem}.space-x-1>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.25rem*var(--tw-space-x-reverse));margin-left:calc(.25rem*(1 - var(--tw-space-x-reverse)))}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.5rem*var(--tw-space-x-reverse));margin-left:calc(.5rem*(1 - var(--tw-space-x-reverse)))}.space-x-20>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(5rem*var(--tw-space-x-reverse));margin-left:calc(5rem*(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1rem*var(--tw-space-x-reverse));margin-left:calc(1rem*(1 - var(--tw-space-x-reverse)))}.space-x-5>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1.25rem*var(--tw-space-x-reverse));margin-left:calc(1.25rem*(1 - var(--tw-space-x-reverse)))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem*var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.75rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem*var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem*var(--tw-space-y-reverse))}.space-y-5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.25rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.25rem*var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem*var(--tw-space-y-reverse))}.divide-y>:not([hidden])~:not([hidden]){--tw-divide-y-reverse:0;border-top-width:calc(1px*(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px*var(--tw-divide-y-reverse))}.divide-gray-200>:not([hidden])~:not([hidden]){--tw-divide-opacity:1;border-color:rgb(229 231 235/var(--tw-divide-opacity))}.justify-self-start{justify-self:start}.justify-self-end{justify-self:end}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-x-auto{overflow-x:auto}.whitespace-normal{white-space:normal}.whitespace-nowrap{white-space:nowrap}.text-wrap{text-wrap:wrap}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:1rem}.rounded-3xl{border-radius:1.5rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-md{border-radius:.375rem}.rounded-xl{border-radius:.75rem}.rounded-l-\[20px\]{border-top-left-radius:20px;border-bottom-left-radius:20px}.rounded-r-\[20px\]{border-top-right-radius:20px;border-bottom-right-radius:20px}.rounded-r-\[8px\]{border-top-right-radius:8px;border-bottom-right-radius:8px}.rounded-t-full{border-top-left-radius:9999px;border-top-right-radius:9999px}.border{border-width:1px}.border-x-0{border-left-width:0;border-right-width:0}.border-b{border-bottom-width:1px}.border-b-0{border-bottom-width:0}.border-b-2{border-bottom-width:2px}.border-l-0{border-left-width:0}.border-r-0{border-right-width:0}.border-t{border-top-width:1px}.border-t-0{border-top-width:0}.border-none{border-style:none}.border-\[\#D7DFE9\]{--tw-border-opacity:1;border-color:rgb(215 223 233/var(--tw-border-opacity))}.border-gray-300{--tw-border-opacity:1;border-color:rgb(209 213 219/var(--tw-border-opacity))}.border-gray-400{--tw-border-opacity:1;border-color:rgb(156 163 175/var(--tw-border-opacity))}.border-primary-200{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.border-primary-400{--tw-border-opacity:1;border-color:rgb(78 179 161/var(--tw-border-opacity))}.border-primary-500{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.border-primary-600{--tw-border-opacity:1;border-color:rgb(40 121 110/var(--tw-border-opacity))}.border-primary-700{--tw-border-opacity:1;border-color:rgb(35 98 90/var(--tw-border-opacity))}.border-secondary-500{--tw-border-opacity:1;border-color:rgb(255 78 51/var(--tw-border-opacity))}.border-transparent{border-color:#0000}.border-y-secondary-300{--tw-border-opacity:1;border-top-color:rgb(255 170 157/var(--tw-border-opacity));border-bottom-color:rgb(255 170 157/var(--tw-border-opacity))}.bg-black{--tw-bg-opacity:1;background-color:rgb(0 0 0/var(--tw-bg-opacity))}.bg-black\/20{background-color:#0003}.bg-gray-100{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}.bg-gray-50{--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity))}.bg-gray-500\/25{background-color:#6b728040}.bg-primary-100{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.bg-primary-200{--tw-bg-opacity:1;background-color:rgb(171 228 214/var(--tw-bg-opacity))}.bg-primary-300{--tw-bg-opacity:1;background-color:rgb(122 206 189/var(--tw-bg-opacity))}.bg-primary-50{--tw-bg-opacity:1;background-color:rgb(243 250 249/var(--tw-bg-opacity))}.bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.bg-primary-800{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.bg-primary-950{--tw-bg-opacity:1;background-color:rgb(13 38 37/var(--tw-bg-opacity))}.bg-secondary-300{--tw-bg-opacity:1;background-color:rgb(255 170 157/var(--tw-bg-opacity))}.bg-secondary-500{--tw-bg-opacity:1;background-color:rgb(255 78 51/var(--tw-bg-opacity))}.bg-secondary-700{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.bg-transparent{background-color:initial}.bg-opacity-0{--tw-bg-opacity:0}.bg-gradient-to-b{background-image:linear-gradient(to bottom,var(--tw-gradient-stops))}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.bg-gradient-to-t{background-image:linear-gradient(to top,var(--tw-gradient-stops))}.from-gray-100{--tw-gradient-from:#f3f4f6 var(--tw-gradient-from-position);--tw-gradient-to:#f3f4f600 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-900\/50{--tw-gradient-from:#1f423e80 var(--tw-gradient-from-position);--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-950{--tw-gradient-from:#0d2625 var(--tw-gradient-from-position);--tw-gradient-to:#0d262500 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-transparent{--tw-gradient-from:#0000 var(--tw-gradient-from-position);--tw-gradient-to:#0000 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-10\%{--tw-gradient-from-position:10%}.via-primary-900{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-primary-900\/50{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e80 var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-10\%{--tw-gradient-via-position:10%}.to-primary-950{--tw-gradient-to:#0d2625 var(--tw-gradient-to-position)}.to-primary-950\/50{--tw-gradient-to:#0d262580 var(--tw-gradient-to-position)}.to-transparent{--tw-gradient-to:#0000 var(--tw-gradient-to-position)}.to-90\%{--tw-gradient-to-position:90%}.bg-\[length\:100px_auto\]{background-size:100px auto}.bg-cover{background-size:cover}.bg-clip-padding{background-clip:padding-box}.bg-center{background-position:50%}.fill-primary-400{fill:#4eb3a1}.object-contain{-o-object-fit:contain;object-fit:contain}.object-cover{-o-object-fit:cover;object-fit:cover}.object-bottom{-o-object-position:bottom;object-position:bottom}.p-0{padding:0}.p-1{padding:.25rem}.p-1\.5{padding:.375rem}.p-10{padding:2.5rem}.p-16{padding:4rem}.p-2{padding:.5rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.p-6{padding:1.5rem}.px-10{padding-left:2.5rem;padding-right:2.5rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-40{padding-left:10rem;padding-right:10rem}.px-5{padding-left:1.25rem;padding-right:1.25rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.px-7{padding-left:1.75rem;padding-right:1.75rem}.py-0{padding-top:0;padding-bottom:0}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-10{padding-top:2.5rem;padding-bottom:2.5rem}.py-16{padding-top:4rem;padding-bottom:4rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-2\.5{padding-top:.625rem;padding-bottom:.625rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-5{padding-top:1.25rem;padding-bottom:1.25rem}.py-6{padding-top:1.5rem;padding-bottom:1.5rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pb-12{padding-bottom:3rem}.pb-16{padding-bottom:4rem}.pb-3{padding-bottom:.75rem}.pb-3\.5{padding-bottom:.875rem}.pb-32{padding-bottom:8rem}.pb-8{padding-bottom:2rem}.pl-3{padding-left:.75rem}.pl-4{padding-left:1rem}.pr-3{padding-right:.75rem}.pr-7{padding-right:1.75rem}.pr-\[10px\]{padding-right:10px}.pt-12{padding-top:3rem}.pt-24{padding-top:6rem}.pt-4{padding-top:1rem}.text-left{text-align:left}.text-center{text-align:center}.text-right{text-align:right}.text-start{text-align:start}.align-middle{vertical-align:middle}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-5xl{font-size:3rem;line-height:1}.text-7xl{font-size:4.5rem;line-height:1}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-\[450\]{font-weight:450}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.lowercase{text-transform:lowercase}.leading-6{line-height:1.5rem}.leading-none{line-height:1}.leading-tight{line-height:1.25}.tracking-tight{letter-spacing:-.025em}.tracking-wide{letter-spacing:.025em}.tracking-wider{letter-spacing:.05em}.text-gray-100{--tw-text-opacity:1;color:rgb(243 244 246/var(--tw-text-opacity))}.text-gray-200{--tw-text-opacity:1;color:rgb(229 231 235/var(--tw-text-opacity))}.text-gray-50{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.text-gray-500{--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity))}.text-gray-600{--tw-text-opacity:1;color:rgb(75 85 99/var(--tw-text-opacity))}.text-gray-700{--tw-text-opacity:1;color:rgb(55 65 81/var(--tw-text-opacity))}.text-gray-800{--tw-text-opacity:1;color:rgb(31 41 55/var(--tw-text-opacity))}.text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.text-primary-300{--tw-text-opacity:1;color:rgb(122 206 189/var(--tw-text-opacity))}.text-primary-400{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.text-primary-50{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.text-primary-500{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.text-primary-600{--tw-text-opacity:1;color:rgb(40 121 110/var(--tw-text-opacity))}.text-primary-800{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.text-primary-900{--tw-text-opacity:1;color:rgb(31 66 62/var(--tw-text-opacity))}.text-primary-950{--tw-text-opacity:1;color:rgb(13 38 37/var(--tw-text-opacity))}.text-secondary-300{--tw-text-opacity:1;color:rgb(255 170 157/var(--tw-text-opacity))}.text-secondary-50{--tw-text-opacity:1;color:rgb(255 243 241/var(--tw-text-opacity))}.text-secondary-500{--tw-text-opacity:1;color:rgb(255 78 51/var(--tw-text-opacity))}.text-secondary-700{--tw-text-opacity:1;color:rgb(200 38 13/var(--tw-text-opacity))}.accent-primary-500{accent-color:#369c8c}.opacity-0{opacity:0}.opacity-100{opacity:1}.opacity-50{opacity:.5}.shadow{--tw-shadow:0 1px 3px 0 #0000001a,0 1px 2px -1px #0000001a;--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color)}.shadow,.shadow-lg{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px #0000001a,0 4px 6px -4px #0000001a;--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.shadow-md{--tw-shadow:0 4px 6px -1px #0000001a,0 2px 4px -2px #0000001a;--tw-shadow-colored:0 4px 6px -1px var(--tw-shadow-color),0 2px 4px -2px var(--tw-shadow-color)}.shadow-md,.shadow-sm{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 2px 0 #0000000d;--tw-shadow-colored:0 1px 2px 0 var(--tw-shadow-color)}.shadow-gray-800\/5{--tw-shadow-color:#1f29370d;--tw-shadow:var(--tw-shadow-colored)}.outline-0{outline-width:0}.ring-1{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.ring-inset{--tw-ring-inset:inset}.ring-gray-800\/\[\.075\]{--tw-ring-color:rgba(31,41,55,.075)}.ring-primary-100{--tw-ring-opacity:1;--tw-ring-color:rgb(213 242 235/var(--tw-ring-opacity))}.ring-primary-200{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.blur{--tw-blur:blur(8px)}.blur,.blur-3xl{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.blur-3xl{--tw-blur:blur(64px)}.brightness-0{--tw-brightness:brightness(0)}.brightness-0,.drop-shadow{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.drop-shadow{--tw-drop-shadow:drop-shadow(0 1px 2px #0000001a) drop-shadow(0 1px 1px #0000000f)}.drop-shadow-sm{--tw-drop-shadow:drop-shadow(0 1px 1px #0000000d)}.drop-shadow-sm,.invert{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.invert{--tw-invert:invert(100%)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur-md{--tw-backdrop-blur:blur(12px)}.backdrop-blur-md,.backdrop-blur-sm{-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-sm{--tw-backdrop-blur:blur(4px)}.backdrop-blur-xl{--tw-backdrop-blur:blur(24px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.delay-100{transition-delay:.1s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.duration-500{transition-duration:.5s}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}.\[-webkit-mask-image\:linear-gradient\(to_bottom\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(180deg,#fff 75%,#fff0)}.\[-webkit-mask-image\:linear-gradient\(to_top\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(0deg,#fff 75%,#fff0)}.after\:absolute:after{content:var(--tw-content);position:absolute}.after\:bottom-0:after{content:var(--tw-content);bottom:0}.after\:left-\[0\.0625rem\]:after{content:var(--tw-content);left:.0625rem}.after\:top-0:after{content:var(--tw-content);top:0}.after\:my-auto:after{content:var(--tw-content);margin-top:auto;margin-bottom:auto}.after\:h-5:after{content:var(--tw-content);height:1.25rem}.after\:w-5:after{content:var(--tw-content);width:1.25rem}.after\:rounded-full:after{content:var(--tw-content);border-radius:9999px}.after\:bg-gray-600:after{content:var(--tw-content);--tw-bg-opacity:1;background-color:rgb(75 85 99/var(--tw-bg-opacity))}.after\:transition-all:after{content:var(--tw-content);transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.after\:content-\[\'\'\]:after{--tw-content:"";content:var(--tw-content)}.hover\:isolate:hover{isolation:isolate}.hover\:border-primary-200:hover{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.hover\:border-transparent:hover{border-color:#0000}.hover\:bg-gray-200:hover{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.hover\:bg-primary-100:hover{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.hover\:bg-primary-700:hover{--tw-bg-opacity:1;background-color:rgb(35 98 90/var(--tw-bg-opacity))}.hover\:bg-primary-800:hover{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.hover\:bg-secondary-100:hover{--tw-bg-opacity:1;background-color:rgb(255 227 223/var(--tw-bg-opacity))}.hover\:bg-secondary-700:hover{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.hover\:bg-opacity-50:hover{--tw-bg-opacity:0.5}.hover\:font-bold:hover{font-weight:700}.hover\:text-gray-50:hover{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.hover\:text-primary-400:hover{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.hover\:text-primary-50:hover{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.hover\:text-primary-500:hover{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.hover\:text-primary-700:hover{--tw-text-opacity:1;color:rgb(35 98 90/var(--tw-text-opacity))}.hover\:text-primary-800:hover{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.hover\:opacity-75:hover{opacity:.75}.focus\:isolate:focus{isolation:isolate}.focus\:border-primary-300:focus{--tw-border-opacity:1;border-color:rgb(122 206 189/var(--tw-border-opacity))}.focus\:border-primary-500:focus{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.focus\:border-transparent:focus{border-color:#0000}.focus\:bg-opacity-50:focus{--tw-bg-opacity:0.5}.focus\:text-primary-800:focus{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.focus\:outline-none:focus{outline:2px solid #0000;outline-offset:2px}.focus\:ring-\[1px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-\[1px\]:focus,.focus\:ring-\[3px\]:focus{box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.focus\:ring-\[3px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-primary-200:focus{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.focus-visible\:outline:focus-visible{outline-style:solid}.focus-visible\:outline-2:focus-visible{outline-width:2px}.focus-visible\:outline-offset-2:focus-visible{outline-offset:2px}.focus-visible\:outline-black:focus-visible{outline-color:#000}.active\:opacity-100:active{opacity:1}.active\:outline-offset-0:active{outline-offset:0}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:opacity-50:disabled{opacity:.5}.group\/btn:hover .group-hover\/btn\:w-2{width:.5rem}.group\/btn:hover .group-hover\/btn\:w-2\.5{width:.625rem}.group:hover .group-hover\:scale-100{--tw-scale-x:1;--tw-scale-y:1}.group:hover .group-hover\:scale-100,.group:hover .group-hover\:scale-x-100{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.group:hover .group-hover\:scale-x-100{--tw-scale-x:1}.group:hover .group-hover\:opacity-100,.group\/btn:hover .group-hover\/btn\:opacity-100{opacity:1}.peer:checked~.peer-checked\:bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.peer:checked~.peer-checked\:text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.peer:checked~.peer-checked\:after\:translate-x-5:after{content:var(--tw-content);--tw-translate-x:1.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.peer:focus~.peer-focus\:outline{outline-style:solid}.peer:focus~.peer-focus\:outline-2{outline-width:2px}.peer:focus~.peer-focus\:outline-offset-2{outline-offset:2px}.peer:focus~.peer-focus\:outline-gray-800{outline-color:#1f2937}.peer:focus:checked~.peer-focus\:peer-checked\:outline-primary-500{outline-color:#369c8c}.peer:active~.peer-active\:outline-offset-0{outline-offset:0}.peer:disabled~.peer-disabled\:cursor-not-allowed{cursor:not-allowed}.peer:disabled~.peer-disabled\:opacity-70{opacity:.7}@media (prefers-reduced-motion:reduce){.motion-reduce\:transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}}@media (min-width:640px){.sm\:left-14{left:3.5rem}.sm\:w-auto{width:auto}.sm\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.sm\:flex-row{flex-direction:row}.sm\:items-center{align-items:center}.sm\:px-6{padding-left:1.5rem;padding-right:1.5rem}.sm\:pb-40{padding-bottom:10rem}.sm\:pl-20{padding-left:5rem}.sm\:text-3xl{font-size:1.875rem;line-height:2.25rem}.sm\:text-5xl{font-size:3rem;line-height:1}.sm\:text-sm{font-size:.875rem;line-height:1.25rem}}@media (min-width:768px){.md\:block{display:block}.md\:h-screen{height:100vh}.md\:w-16{width:4rem}.md\:w-20{width:5rem}.md\:w-24{width:6rem}.md\:max-w-2xl{max-width:42rem}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.md\:justify-end{justify-content:flex-end}.md\:gap-16{gap:4rem}.md\:px-14{padding-left:3.5rem;padding-right:3.5rem}.md\:py-10{padding-top:2.5rem;padding-bottom:2.5rem}.md\:pb-52{padding-bottom:13rem}.md\:text-3xl{font-size:1.875rem;line-height:2.25rem}.md\:text-4xl{font-size:2.25rem;line-height:2.5rem}.md\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width:1024px){.lg\:left-20{left:5rem}.lg\:order-last{order:9999}.lg\:col-span-3{grid-column:span 3/span 3}.lg\:col-span-4{grid-column:span 4/span 4}.lg\:-my-11{margin-top:-2.75rem;margin-bottom:-2.75rem}.lg\:-mr-9{margin-right:-2.25rem}.lg\:mb-0{margin-bottom:0}.lg\:mb-16{margin-bottom:4rem}.lg\:mt-0{margin-top:0}.lg\:block{display:block}.lg\:flex{display:flex}.lg\:grid{display:grid}.lg\:max-w-4xl{max-width:56rem}.lg\:max-w-7xl{max-width:80rem}.lg\:max-w-max{max-width:-moz-max-content;max-width:max-content}.lg\:max-w-xl{max-width:36rem}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-7{grid-template-columns:repeat(7,minmax(0,1fr))}.lg\:gap-5{gap:1.25rem}.lg\:p-10{padding:2.5rem}.lg\:p-8{padding:2rem}.lg\:px-8{padding-left:2rem;padding-right:2rem}.lg\:py-12{padding-top:3rem;padding-bottom:3rem}.lg\:py-24{padding-top:6rem;padding-bottom:6rem}.lg\:pb-20{padding-bottom:5rem}.lg\:pb-60{padding-bottom:15rem}.lg\:pl-28{padding-left:7rem}.lg\:pt-20{padding-top:5rem}.lg\:pt-40{padding-top:10rem}.lg\:text-4xl{font-size:2.25rem;line-height:2.5rem}.lg\:text-5xl{font-size:3rem;line-height:1}.lg\:text-6xl{font-size:3.75rem;line-height:1}.lg\:text-lg{font-size:1.125rem;line-height:1.75rem}}@media (min-width:1280px){.xl\:col-span-3{grid-column:span 3/span 3}.xl\:mx-auto{margin-left:auto;margin-right:auto}.xl\:flex{display:flex}.xl\:hidden{display:none}.xl\:w-auto{width:auto}.xl\:w-full{width:100%}.xl\:max-w-lg{max-width:32rem}.xl\:max-w-xl{max-width:36rem}.xl\:gap-16{gap:4rem}.xl\:gap-x-16{-moz-column-gap:4rem;column-gap:4rem}.xl\:px-20{padding-left:5rem;padding-right:5rem}.xl\:py-20{padding-top:5rem;padding-bottom:5rem}.xl\:py-32{padding-top:8rem}.xl\:pb-32,.xl\:py-32{padding-bottom:8rem}.xl\:pb-\[16\.5rem\]{padding-bottom:16.5rem}.xl\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (prefers-color-scheme:dark){.dark\:border-gray-700{--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}.dark\:bg-gray-900{--tw-bg-opacity:1;background-color:rgb(17 24 39/var(--tw-bg-opacity))}.dark\:text-black{--tw-text-opacity:1;color:rgb(0 0 0/var(--tw-text-opacity))}.dark\:text-gray-300{--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity))}.dark\:after\:bg-gray-300:after{content:var(--tw-content);--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity))}.peer:checked~.dark\:peer-checked\:bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.peer:focus~.dark\:peer-focus\:outline-gray-300{outline-color:#d1d5db}.peer:focus:checked~.dark\:peer-focus\:peer-checked\:outline-primary-500{outline-color:#369c8c}} From 0ef00f06b8cc86299143dfb555e5b596442a10c4 Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 07:04:53 +0100 Subject: [PATCH 04/19] display volumes and delete volumes --- assets/main.js | 39 +++++ unikernel.ml | 75 +++++++++ volume_index.ml | 424 ++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 538 insertions(+) create mode 100644 volume_index.ml diff --git a/assets/main.js b/assets/main.js index 17c1696f..138ce827 100644 --- a/assets/main.js +++ b/assets/main.js @@ -529,3 +529,42 @@ async function logout() { buttonLoading(logoutButton, false, "Logout") } } +async function deleteVolume(block_name) { + const deleteButton = document.getElementById(`delete-block-button-${block_name}`); + const molly_csrf = document.getElementById("molly-csrf").value; + const formAlert = document.getElementById("form-alert"); + try { + buttonLoading(deleteButton, true, "Deleting...") + const response = await fetch("/api/volume/delete", { + method: 'POST', + headers: { + "Content-Type": "application/json", + }, + body: JSON.stringify( + { + "block_name": block_name, + "molly_csrf": molly_csrf + }) + }) + const data = await response.json(); + if (data.status === 200) { + formAlert.classList.remove("hidden", "text-secondary-500"); + formAlert.classList.add("text-primary-500"); + formAlert.textContent = "Succesfully deleted"; + postAlert("bg-primary-300", "Volume deleted succesfully"); + setTimeout(() => window.location.reload(), 1000); + buttonLoading(deleteButton, false, "Delete") + } else { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = data.data + buttonLoading(deleteButton, false, "Delete") + } + } catch (error) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = error + buttonLoading(deleteButton, false, "Delete") + } +} + diff --git a/unikernel.ml b/unikernel.ml index 00bf3ee1..75de8c4e 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1201,6 +1201,69 @@ struct (Yojson.Basic.to_string (`Assoc json_dict))) `Bad_request + let volumes store albatross reqd (user : User_model.user) = + (Albatross.query albatross ~domain:user.name (`Block_cmd `Block_info) + >|= function + | Error msg -> + Logs.err (fun m -> + m "error while communicating with albatross: %s" msg); + [] + | Ok (_hdr, `Success (`Block_devices blocks)) -> blocks + | Ok reply -> + Logs.err (fun m -> + m "expected a block info reply, received %a" + (Vmm_commands.pp_wire ~verbose:false) + reply); + []) + >>= fun blocks -> + let policy = + match Albatross.policy ~domain:user.name albatross with + | Ok p -> p + | Error _ -> None + in + let now = Ptime.v (P.now_d_ps ()) in + generate_csrf_token store user now reqd >>= function + | Ok csrf -> + Lwt.return + (reply reqd ~content_type:"text/html" + (Dashboard.dashboard_layout ~csrf user + ~page_title:(String.capitalize_ascii user.name ^ " | Mollymawk") + ~content:(Volume_index.volume_index_layout blocks policy) + ~icon:"/images/robur.png" ()) + ~header_list:[ ("X-MOLLY-CSRF", csrf) ] + `OK) + | Error err -> + Lwt.return + (reply reqd ~content_type:"text/html" + (Guest_layout.guest_layout ~page_title:"500 | Mollymawk" + ~content:(Error_page.error_layout err) + ~icon:"/images/robur.png" ()) + `Internal_server_error) + + let delete_volume json_dict albatross reqd (user : User_model.user) = + match Utils.Json.get "block_name" json_dict with + | Some (`String block_name) -> ( + Albatross.query albatross ~domain:user.name ~name:block_name + (`Block_cmd `Block_remove) + >>= function + | Error msg -> + Logs.err (fun m -> m "Error querying albatross: %s" msg); + Middleware.http_response reqd ~title:"Error" + ~data:("Error querying albatross: " ^ msg) + `Internal_server_error + | Ok (_hdr, res) -> ( + match Albatross_json.res res with + | Ok res -> + Middleware.http_response reqd ~title:"Success" + ~data:(Yojson.Basic.to_string res) + `OK + | Error (`String res) -> + Middleware.http_response reqd ~title:"Error" + ~data:(Yojson.Basic.to_string (`String res)) + `Internal_server_error)) + | _ -> + Middleware.http_response reqd ~title:"Error" + ~data:"Couldn't find block name in json" `Bad_request let request_handler stack albatross js_file css_file imgs store (_ipaddr, _port) reqd = Lwt.async (fun () -> @@ -1309,6 +1372,18 @@ struct Middleware.http_response reqd ~title:"Error" ~data:"An error occured. Please refresh and try again" `Bad_request) + | "/volumes" -> + check_meth `GET (fun () -> + authenticate store reqd (volumes store !albatross reqd)) + | "/api/volume/delete" -> + check_meth `POST (fun () -> + extract_csrf_token reqd >>= function + | Ok (form_csrf, json_dict) -> + authenticate ~form_csrf ~api_meth:true store reqd + (delete_volume json_dict !albatross reqd) + | Error (`Msg msg) -> + Middleware.http_response reqd ~title:"Error" + ~data:(String.escaped msg) `Bad_request) | "/admin/users" -> check_meth `GET (fun () -> authenticate ~check_admin:true store reqd (users store reqd)) diff --git a/volume_index.ml b/volume_index.ml new file mode 100644 index 00000000..051d8d04 --- /dev/null +++ b/volume_index.ml @@ -0,0 +1,424 @@ +let volume_index_layout volumes policy = + let total_volume_used = + List.fold_left (fun total_size (_, size, _) -> total_size + size) 0 volumes + in + let total_free_space = + match policy with + | Some policy -> + Option.value ~default:0 policy.Vmm_core.Policy.block - total_volume_used + | None -> 0 + in + Tyxml_html.( + section + ~a:[ a_class [ "col-span-7 p-4 bg-gray-50 my-1" ] ] + [ + section + ~a:[ a_id "block-display"; a_class [ "block" ] ] + [ + div + ~a:[ a_class [ "px-3 flex justify-between items-center" ] ] + [ + div + [ + p + ~a:[ a_class [ "font-bold text-gray-700" ] ] + [ + txt + ("Volumes (" + ^ string_of_int (List.length volumes) + ^ ")"); + ]; + ]; + div + ~a:[ Unsafe.string_attrib "x-data" "{modalIsOpen: false}" ] + [ + button + ~a: + [ + Unsafe.string_attrib "x-on:click" "modalIsOpen = true"; + a_class + [ + "py-3 px-3 rounded bg-primary-500 \ + hover:bg-primary-800 w-full text-gray-50 \ + font-semibold"; + ]; + ] + [ txt "Create block device" ]; + div + ~a: + [ + Unsafe.string_attrib "x-cloak" ""; + Unsafe.string_attrib "x-show" "modalIsOpen"; + Unsafe.string_attrib + "x-transition.opacity.duration.200ms" ""; + Unsafe.string_attrib "x-trap.inert.noscroll" + "modalIsOpen"; + Unsafe.string_attrib "x-on:keydown.esc.window" + "modalIsOpen = false"; + Unsafe.string_attrib "x-on:click.self" + "modalIsOpen = false"; + a_class + [ + "fixed inset-0 z-30 flex items-end \ + justify-center bg-black/20 p-4 backdrop-blur-md \ + sm:items-center"; + ]; + a_role [ "dialog" ]; + a_aria "modal" [ "true" ]; + ] + [ + div + ~a: + [ + Unsafe.string_attrib "x-show" "modalIsOpen"; + Unsafe.string_attrib "x-transition:enter" + "transition ease-out duration-200 delay-100 \ + motion-reduce:transition-opacity"; + Unsafe.string_attrib "x-transition:enter-start" + "opacity-0 scale-50"; + Unsafe.string_attrib "x-transition:enter-end" + "opacity-100 scale-100"; + a_class + [ + "flex max-w-xl flex-col gap-4 \ + overflow-hidden rounded-md border \ + border-neutral-300 bg-gray-50"; + ]; + ] + [ + div + ~a: + [ + a_class + [ + "flex items-center justify-between \ + border-b p-4"; + ]; + ] + [ + h3 + ~a:[ a_class [ "font-bold text-gray-700" ] ] + [ txt "Create a volume" ]; + i + ~a: + [ + a_class + [ + "fa-solid fa-x text-sm cursor-pointer"; + ]; + Unsafe.string_attrib "x-on:click" + "modalIsOpen = false"; + ] + []; + ]; + div + ~a:[ a_class [ "px-4" ] ] + [ Volume_create.create_volume total_free_space ]; + ]; + ]; + ]; + ]; + div + ~a:[ a_class [ "mx-auto text-center" ] ] + [ + div + ~a: + [ + Unsafe.string_attrib "x-data" "chart: null"; + Unsafe.string_attrib "x-init" + ("\n\ + \ chart = new \ + Chart(document.getElementById('usageChart').getContext('2d'), \ + {\n\ + \ type: 'pie',\n\ + \ data: {\n\ + \ labels: ['Free storage (" + ^ string_of_int total_free_space + ^ "MB)','Used storage (" + ^ string_of_int total_volume_used + ^ "MB)'],\n\ + \ datasets: [{\n\ + \ label: 'Size',\n\ + \ data: [" + ^ string_of_int total_free_space + ^ ", " + ^ string_of_int total_volume_used + ^ "],\n\ + \ backgroundColor: ['rgb(156, \ + 156, 156)','rgb(54, 156, 140)'],\n\ + \ hoverOffset: 4,\n\ + \ }]\n\ + \ },\n\ + \ options: {}\n\ + \ });\n\ + \ "); + a_class [ "flex justify-center items-center" ]; + a_style "position: relative; height:30vh; width:70vw;"; + ] + [ canvas ~a:[ a_id "usageChart" ] [] ]; + ]; + div + [ + input + ~a: + [ + a_onkeyup "filterData()"; + a_placeholder "search"; + a_id "searchQuery"; + a_name "searchQuery"; + a_input_type `Text; + a_class + [ + "rounded py-2 px-3 border border-primary-200 \ + focus:border-primary-500 outline-0"; + ]; + ] + (); + ]; + hr ~a:[ a_class [ "border border-primary-500 my-5" ] ] (); + div + ~a:[ a_class [ "flex flex-col" ] ] + [ + div + ~a:[ a_class [ "-m-1.5 overflow-x-auto" ] ] + [ + div + ~a: + [ + a_class + [ "p-1.5 min-w-full inline-block align-middle" ]; + ] + [ + div + ~a: + [ + Unsafe.string_attrib "x-data" "sort_data()"; + a_class [ "overflow-hidden" ]; + ] + [ + table + ~a: + [ + a_id "data-table"; + a_class + [ + "table-auto min-w-full divide-y \ + divide-gray-200"; + ]; + ] + ~thead: + (thead + [ + tr + [ + th + ~a: + [ + Unsafe.string_attrib "x-on:click" + "sortByColumn"; + a_class + [ + "px-6 py-3 text-start \ + text-xs font-bold \ + text-primary-600 uppercase \ + cursor-pointer select-none"; + ]; + ] + [ + txt "Host Device"; + span + ~a:[ a_class [ "px-2" ] ] + [ + i + ~a: + [ + a_class + [ "fa-solid fa-sort" ]; + ] + []; + ]; + ]; + th + ~a: + [ + Unsafe.string_attrib "x-on:click" + "sortByColumn"; + a_class + [ + "px-6 py-3 text-start \ + text-xs font-bold \ + text-primary-600 uppercase \ + cursor-pointer select-none"; + ]; + ] + [ + txt "Size (MB)"; + span + ~a:[ a_class [ "px-2" ] ] + [ + i + ~a: + [ + a_class + [ "fa-solid fa-sort" ]; + ] + []; + ]; + ]; + th + ~a: + [ + Unsafe.string_attrib "x-on:click" + "sortByColumn"; + a_class + [ + "px-6 py-3 text-start \ + text-xs font-bold \ + text-primary-600 uppercase \ + cursor-pointer select-none"; + ]; + ] + [ + txt "Used"; + span + ~a:[ a_class [ "px-2" ] ] + [ + i + ~a: + [ + a_class + [ "fa-solid fa-sort" ]; + ] + []; + ]; + ]; + th + ~a: + [ + a_class + [ + "px-6 py-3 text-start \ + text-xs font-bold \ + text-primary-600 uppercase \ + cursor-pointer select-none"; + ]; + ] + [ txt "Action" ]; + ]; + ]) + (List.map + (fun ((name, size, used) : + Vmm_core.Name.t * int * bool) -> + let name = + Option.value ~default:"no name" + (Vmm_core.Name.name name) + in + tr + ~a: + [ + a_class [ "border-b border-gray-200" ]; + ] + [ + td + ~a: + [ + a_class + [ + "px-6 py-4 whitespace-nowrap \ + text-sm font-medium \ + text-gray-800"; + ]; + ] + [ txt name ]; + td + ~a: + [ + a_class + [ + "px-6 py-4 whitespace-nowrap \ + text-sm font-medium \ + text-gray-800"; + ]; + ] + [ txt (string_of_int size ^ "MB") ]; + td + ~a: + [ + a_class + [ + "px-6 py-4 whitespace-nowrap \ + text-sm font-medium \ + text-gray-800"; + ]; + ] + [ txt (string_of_bool used) ]; + td + ~a: + [ + a_class + [ + "px-6 py-4 whitespace-nowrap \ + text-sm font-medium \ + text-gray-800 space-x-5"; + ]; + ] + [ + (if used then + a + ~a: + [ + a_href + ("/unikernel/info/" ^ name); + a_class + [ + "inline-flex \ + items-center gap-x-2 \ + text-sm font-semibold \ + rounded border \ + border-1 py-1 px-2 \ + border-primary-400 \ + text-primary-600 \ + hover:text-primary-500 \ + focus:outline-none \ + focus:text-primary-800 \ + disabled:opacity-50 \ + disabled:pointer-events-none"; + ]; + ] + [ txt "View" ] + else + span + ~a: + [ + a_class + [ "gap-x-2 py-1 px-2" ]; + ] + []); + button + ~a: + [ + a_id + ("delete-block-button-" + ^ name); + a_onclick + ("deleteVolume('" ^ name + ^ "')"); + a_class + [ + "py-1 px-2 mx-2 rounded \ + bg-secondary-500 \ + hover:bg-secondary-800 \ + text-gray-50 \ + font-semibold"; + ]; + ] + [ txt "Delete" ]; + ]; + ]) + volumes); + ]; + ]; + ]; + ]; + ]; + ]) From 255cf2107402bfcf8cda25719cf7ba4f30c53673 Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 07:05:50 +0100 Subject: [PATCH 05/19] refactor multipart reading to separate function --- unikernel.ml | 59 ++++++++++++++++++++++++++++++++-------------------- 1 file changed, 36 insertions(+), 23 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index 75de8c4e..51cc488a 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -202,6 +202,39 @@ struct let resp = Httpaf.Response.create ~headers status in Httpaf.Reqd.respond_with_string reqd resp data + let read_multipart_data reqd = + let response_body = Httpaf.Reqd.request_body reqd in + let finished, notify_finished = Lwt.wait () in + let wakeup v = Lwt.wakeup_later notify_finished v in + let on_eof data () = wakeup data in + let f acc s = acc ^ s in + let rec on_read on_eof acc bs ~off ~len = + let str = Bigstringaf.substring ~off ~len bs in + let acc = acc >>= fun acc -> Lwt.return (f acc str) in + Httpaf.Body.schedule_read response_body ~on_read:(on_read on_eof acc) + ~on_eof:(on_eof acc) + in + let f_init = Lwt.return "" in + Httpaf.Body.schedule_read response_body ~on_read:(on_read on_eof f_init) + ~on_eof:(on_eof f_init); + finished >>= fun data -> + data >>= fun data -> + let content_type = + Httpaf.( + Headers.get_exn (Reqd.request reqd).Request.headers "content-type") + in + let ct = Multipart_form.Content_type.of_string (content_type ^ "\r\n") in + match ct with + | Error (`Msg msg) -> + Logs.warn (fun m -> m "couldn't content-type: %S" msg); + Error (`Msg ("couldn't content-type:" ^ msg)) |> Lwt.return + | Ok ct -> ( + match Multipart_form.of_string_to_list data ct with + | Error (`Msg msg) -> + Logs.warn (fun m -> m "couldn't multipart: %s" msg); + Error (`Msg ("Couldn't multipart: " ^ msg)) |> Lwt.return + | Ok (m, assoc) -> Ok (m, assoc) |> Lwt.return) + let sign_up reqd = let now = Ptime.v (P.now_d_ps ()) in let csrf = Middleware.generate_csrf_cookie now reqd in @@ -917,28 +950,8 @@ struct ~data:"Couldn't find unikernel name in json" `Bad_request let unikernel_create albatross reqd (user : User_model.user) = - let response_body = Httpaf.Reqd.request_body reqd in - let finished, notify_finished = Lwt.wait () in - let wakeup v = Lwt.wakeup_later notify_finished v in - let on_eof data () = wakeup data in - let f acc s = acc ^ s in - let rec on_read on_eof acc bs ~off ~len = - let str = Bigstringaf.substring ~off ~len bs in - let acc = acc >>= fun acc -> Lwt.return (f acc str) in - Httpaf.Body.schedule_read response_body ~on_read:(on_read on_eof acc) - ~on_eof:(on_eof acc) - in - let f_init = Lwt.return "" in - Httpaf.Body.schedule_read response_body ~on_read:(on_read on_eof f_init) - ~on_eof:(on_eof f_init); - finished >>= fun data -> - data >>= fun data -> - let content_type = - Httpaf.( - Headers.get_exn (Reqd.request reqd).Request.headers "content-type") - in - let ct = Multipart_form.Content_type.of_string (content_type ^ "\r\n") in - match ct with + read_multipart_data reqd >>= fun result -> + match result with | Error (`Msg msg) -> Logs.warn (fun m -> m "couldn't content-type: %S" msg); Middleware.http_response reqd ~title:"Error" @@ -997,7 +1010,7 @@ struct | _ -> Logs.warn (fun m -> m "couldn't find fields"); Middleware.http_response reqd ~title:"Error" - ~data:"Couldn't find fields" `Bad_request)) + ~data:"Couldn't find fields" `Bad_request) let unikernel_console albatross name reqd (user : User_model.user) = (* TODO use uuid in the future *) From 4945bc3e7efdc268a6e51e3b0d4619eccbeaa891 Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 07:06:25 +0100 Subject: [PATCH 06/19] create volumes --- assets/main.js | 64 ++++++++++++ unikernel.ml | 155 +++++++++++++++++++---------- volume_create.ml | 250 +++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 415 insertions(+), 54 deletions(-) create mode 100644 volume_create.ml diff --git a/assets/main.js b/assets/main.js index 138ce827..7385b11e 100644 --- a/assets/main.js +++ b/assets/main.js @@ -529,6 +529,7 @@ async function logout() { buttonLoading(logoutButton, false, "Logout") } } + async function deleteVolume(block_name) { const deleteButton = document.getElementById(`delete-block-button-${block_name}`); const molly_csrf = document.getElementById("molly-csrf").value; @@ -568,3 +569,66 @@ async function deleteVolume(block_name) { } } +async function createVolume() { + const createButton = document.getElementById("create-block-button"); + const block_name = document.getElementById("block_name").value; + const block_size = document.getElementById("block_size").innerText; + const data_toggle = document.getElementById("dataToggle").checked; + const molly_csrf = document.getElementById("molly-csrf").value; + const formAlert = document.getElementById("form-alert"); + const block_compressed = document.getElementById("block_compressed").checked; + const block_data = document.getElementById("block_data").files[0]; + try { + if (block_name === "") { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = "Please enter a name for this volume" + buttonLoading(createButton, false, "Create volume") + } + else if (Number(block_size) < 1) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = "Volume size must be 1MB or greater." + buttonLoading(createButton, false, "Create volume") + } + else { + buttonLoading(createButton, true, "Creating...") + let formData = new FormData(); + formData.append("block_name", block_name); + formData.append("block_size", Number(block_size)) + if (data_toggle && !block_data) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = "You must upload a file else switch 'Dumb data to this volume' off" + buttonLoading(createButton, false, "Create volume") + return; + } + formData.append("block_compressed", block_compressed) + formData.append("block_data", block_data) + formData.append("molly_csrf", molly_csrf) + const response = await fetch("/api/volume/create", { + method: 'POST', + body: formData + }) + const data = await response.json(); + if (data.status === 200) { + formAlert.classList.remove("hidden", "text-secondary-500"); + formAlert.classList.add("text-primary-500"); + formAlert.textContent = "Succesfully deleted"; + postAlert("bg-primary-300", "Volume created succesfully"); + setTimeout(() => window.location.reload(), 000); + buttonLoading(createButton, false, "Create volume") + } else { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = data.data + buttonLoading(createButton, false, "Create volume") + } + } + } catch (error) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = error + buttonLoading(createButton, false, "Create volume") + } +} diff --git a/unikernel.ml b/unikernel.ml index 51cc488a..3a37255f 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -953,63 +953,55 @@ struct read_multipart_data reqd >>= fun result -> match result with | Error (`Msg msg) -> - Logs.warn (fun m -> m "couldn't content-type: %S" msg); Middleware.http_response reqd ~title:"Error" - ~data:("Couldn't content-type: " ^ msg) + ~data:("Couldn't multipart: " ^ msg) `Bad_request - | Ok ct -> ( - match Multipart_form.of_string_to_list data ct with - | Error (`Msg msg) -> - Logs.warn (fun m -> m "couldn't multipart: %s" msg); - Middleware.http_response reqd ~title:"Error" - ~data:("Couldn't multipart: " ^ msg) - `Bad_request - | Ok (m, assoc) -> ( - let m, _r = to_map ~assoc m in - match - ( Map.find_opt "arguments" m, - Map.find_opt "name" m, - Map.find_opt "binary" m, - Map.find_opt "molly_csrf" m ) - with - | ( Some (_, args), - Some (_, name), - Some (_, binary), - Some (_, form_csrf_token) ) -> - let now = Ptime.v (P.now_d_ps ()) in - Middleware.csrf_verification user now form_csrf_token - (fun reqd -> - match Albatross_json.config_of_json args with - | Ok cfg -> ( - let config = { cfg with image = binary } in - (* TODO use uuid in the future *) - Albatross.query albatross ~domain:user.name ~name - (`Unikernel_cmd (`Unikernel_create config)) - >>= function - | Error err -> - Logs.warn (fun m -> - m "Error querying albatross: %s" err); + | Ok (m, assoc) -> ( + let m, _r = to_map ~assoc m in + match + ( Map.find_opt "arguments" m, + Map.find_opt "name" m, + Map.find_opt "binary" m, + Map.find_opt "molly_csrf" m ) + with + | ( Some (_, args), + Some (_, name), + Some (_, binary), + Some (_, form_csrf_token) ) -> + let now = Ptime.v (P.now_d_ps ()) in + Middleware.csrf_verification user now form_csrf_token + (fun reqd -> + match Albatross_json.config_of_json args with + | Ok cfg -> ( + let config = { cfg with image = binary } in + (* TODO use uuid in the future *) + Albatross.query albatross ~domain:user.name ~name + (`Unikernel_cmd (`Unikernel_create config)) + >>= function + | Error err -> + Logs.warn (fun m -> + m "Error querying albatross: %s" err); + Middleware.http_response reqd ~title:"Error" + ~data:("Error while querying Albatross: " ^ err) + `Internal_server_error + | Ok (_hdr, res) -> ( + match Albatross_json.res res with + | Ok res -> + Middleware.http_response reqd ~title:"Success" + ~data:(Yojson.Basic.to_string res) + `OK + | Error (`String res) -> Middleware.http_response reqd ~title:"Error" - ~data:("Error while querying Albatross: " ^ err) - `Internal_server_error - | Ok (_hdr, res) -> ( - match Albatross_json.res res with - | Ok res -> - Middleware.http_response reqd ~title:"Success" - ~data:(Yojson.Basic.to_string res) - `OK - | Error (`String res) -> - Middleware.http_response reqd ~title:"Error" - ~data:(Yojson.Basic.to_string (`String res)) - `Internal_server_error)) - | Error (`Msg err) -> - Logs.warn (fun m -> m "couldn't decode data %s" err); - Middleware.http_response reqd ~title:"Error" ~data:err - `Internal_server_error) - reqd - | _ -> - Logs.warn (fun m -> m "couldn't find fields"); - Middleware.http_response reqd ~title:"Error" + ~data:(Yojson.Basic.to_string (`String res)) + `Internal_server_error)) + | Error (`Msg err) -> + Logs.warn (fun m -> m "couldn't decode data %s" err); + Middleware.http_response reqd ~title:"Error" ~data:err + `Internal_server_error) + reqd + | _ -> + Logs.warn (fun m -> m "couldn't find fields"); + Middleware.http_response reqd ~title:"Error" ~data:"Couldn't find fields" `Bad_request) let unikernel_console albatross name reqd (user : User_model.user) = @@ -1277,6 +1269,58 @@ struct | _ -> Middleware.http_response reqd ~title:"Error" ~data:"Couldn't find block name in json" `Bad_request + + let create_volume albatross reqd (user : User_model.user) = + read_multipart_data reqd >>= fun result -> + match result with + | Error (`Msg msg) -> + Middleware.http_response reqd ~title:"Error" + ~data:("Couldn't multipart: " ^ msg) + `Bad_request + | Ok (m, assoc) -> ( + let m, _r = to_map ~assoc m in + match + ( Map.find_opt "block_name" m, + Map.find_opt "block_size" m, + Map.find_opt "block_data" m, + Map.find_opt "block_compressed" m, + Map.find_opt "molly_csrf" m ) + with + | ( Some (_, block_name), + Some (_, block_size), + Some (_, block_data), + Some (_, block_compressed), + Some (_, form_csrf_token) ) -> + (let now = Ptime.v (P.now_d_ps ()) in + Middleware.csrf_verification user now form_csrf_token (fun reqd -> + Albatross.query albatross ~domain:user.name ~name:block_name + (`Block_cmd + (`Block_add + ( int_of_string block_size, + bool_of_string block_compressed, + Some block_data ))) + >>= function + | Error msg -> + Logs.err (fun m -> m "Error querying albatross: %s" msg); + Middleware.http_response reqd ~title:"Error" + ~data:("Error querying albatross: " ^ msg) + `Internal_server_error + | Ok (_hdr, res) -> ( + match Albatross_json.res res with + | Ok res -> + Middleware.http_response reqd ~title:"Success" + ~data:(Yojson.Basic.to_string res) + `OK + | Error (`String res) -> + Middleware.http_response reqd ~title:"Error" + ~data:(Yojson.Basic.to_string (`String res)) + `Internal_server_error))) + reqd + | _ -> + Logs.warn (fun m -> m "couldn't find fields"); + Middleware.http_response reqd ~title:"Error" + ~data:"Couldn't find fields" `Bad_request) + let request_handler stack albatross js_file css_file imgs store (_ipaddr, _port) reqd = Lwt.async (fun () -> @@ -1397,6 +1441,9 @@ struct | Error (`Msg msg) -> Middleware.http_response reqd ~title:"Error" ~data:(String.escaped msg) `Bad_request) + | "/api/volume/create" -> + check_meth `POST (fun () -> + authenticate store reqd (create_volume !albatross reqd)) | "/admin/users" -> check_meth `GET (fun () -> authenticate ~check_admin:true store reqd (users store reqd)) diff --git a/volume_create.ml b/volume_create.ml new file mode 100644 index 00000000..32fbcd0f --- /dev/null +++ b/volume_create.ml @@ -0,0 +1,250 @@ +let create_volume total_free_space = + Tyxml_html.( + section + ~a:[ a_id "block-create"; a_class [ "w-full mx-auto" ] ] + [ + p ~a:[ a_id "form-alert" ] []; + div + ~a:[ a_class [ "my-6" ] ] + [ + label + ~a: + [ + a_class [ "block text-sm font-medium" ]; + a_label_for "block name"; + ] + [ txt "Volume name" ]; + input + ~a: + [ + a_autocomplete `On; + a_input_type `Text; + a_name "block_name"; + a_id "block_name"; + a_class + [ + "ring-primary-100 mt-1.5 transition appearance-none \ + block w-full px-3 py-3 rounded-xl shadow-sm border \ + hover:border-primary-200\n\ + \ \ + focus:border-primary-300 bg-primary-50 bg-opacity-0 \ + hover:bg-opacity-50 focus:bg-opacity-50 \ + ring-primary-200 focus:ring-primary-200\n\ + \ \ + focus:ring-[1px] focus:outline-none"; + ]; + ] + (); + ]; + div + ~a:[ a_class [ "my-6" ] ] + [ + label + ~a: + [ + a_class [ "block text-sm font-medium" ]; + a_label_for "block size"; + ] + [ txt "Volume size" ]; + small + ~a:[ a_class [ "text-sm" ] ] + [ + txt + ("can assign up to: " ^ string_of_int total_free_space ^ " MB"); + ]; + div + ~a: + [ + a_class [ "space-x-5 my-4" ]; + Unsafe.string_attrib "x-data" "{count : 50}"; + ] + [ + button + ~a: + [ + a_class + [ + "border py-2 px-3 border-primary-500 \ + hover:bg-primary-100 rounded-md"; + ]; + Unsafe.string_attrib "x-on:click" "count = count + 50"; + ] + [ i ~a:[ a_class [ "fa-solid fa-plus" ] ] [] ]; + span + ~a: + [ + a_id "block_size"; + a_contenteditable true; + a_class [ "text-4xl border px-4" ]; + a_user_data "x-on:keydown.enter.prevent" ""; + Unsafe.string_attrib "x-on:input" + "let value = \ + $event.target.innerText.replace(/[^0-9]/g,'');\n\ + \ \ + $event.target.innerText = value;\n\ + \ count = \ + parseInt(value) || 1;"; + Unsafe.string_attrib "x-text" "count"; + Unsafe.string_attrib "x-on:blur" + "$event.target.innerText = count;"; + ] + []; + span ~a:[ a_class [ "text-4xl" ] ] [ txt "MB" ]; + button + ~a: + [ + a_class + [ + "border py-2 px-3 border-secondary-500 \ + hover:bg-secondary-100 rounded-md"; + ]; + Unsafe.string_attrib "x-on:click" + "if (count > 1) count = count - 50"; + ] + [ i ~a:[ a_class [ "fa-solid fa-minus" ] ] [] ]; + ]; + ]; + div + ~a: + [ + a_class [ "my-6" ]; + Unsafe.string_attrib "x-data" "{ showUpload : false}"; + ] + [ + label + ~a: + [ + a_label_for "dataToggle"; + a_class [ "inline-flex cursor-pointer items-center gap-3" ]; + ] + [ + input + ~a: + [ + a_id "dataToggle"; + a_input_type `Checkbox; + a_class [ "peer sr-only" ]; + a_role [ "switch" ]; + Unsafe.string_attrib "x-on:click" + "showUpload = !showUpload"; + ] + (); + span + ~a: + [ + a_aria "hidden" [ "true" ]; + a_class + [ + "relative h-6 w-11 after:h-5 after:w-5 \ + peer-checked:after:translate-x-5 rounded-full \ + border border-gray-300 bg-gray-50 after:absolute \ + after:bottom-0 after:left-[0.0625rem] after:top-0 \ + after:my-auto after:rounded-full after:bg-gray-600 \ + after:transition-all after:content-[''] \ + peer-checked:bg-primary-500 \ + peer-checked:after:bg-white peer-focus:outline \ + peer-focus:outline-2 peer-focus:outline-offset-2 \ + peer-focus:outline-gray-800 \ + peer-focus:peer-checked:outline-primary-500 \ + peer-active:outline-offset-0 \ + peer-disabled:cursor-not-allowed \ + peer-disabled:opacity-70 dark:border-gray-700 \ + dark:bg-gray-900 dark:after:bg-gray-300 \ + dark:peer-checked:bg-primary-500 \ + dark:peer-checked:after:bg-white \ + dark:peer-focus:outline-gray-300 \ + dark:peer-focus:peer-checked:outline-primary-500"; + ]; + ] + []; + span + ~a: + [ + a_class + [ + "trancking-wide text-sm font-medium text-gray-600 \ + peer-checked:text-gray-900 \ + peer-disabled:cursor-not-allowed \n\ + \ \ + dark:peer-checked:text-white"; + ]; + ] + [ txt "Dumb data to this volume" ]; + ]; + div + ~a: + [ + Unsafe.string_attrib "x-show" "showUpload"; a_class [ "my-4" ]; + ] + [ + div + ~a:[ a_class [ "my-4" ] ] + [ + label + ~a: + [ + a_class [ "block text-sm font-medium" ]; + a_label_for "data"; + ] + [ txt "Select a file to dump to the volume" ]; + input + ~a: + [ + a_input_type `File; + a_name "block_data"; + a_id "block_data"; + a_class + [ + "ring-primary-100 mt-1.5 transition \ + appearance-none block w-full px-3 py-3 \ + rounded-xl shadow-sm border \ + hover:border-primary-200\n\ + \ \ + focus:border-primary-300 bg-primary-50 \ + bg-opacity-0 hover:bg-opacity-50 \ + focus:bg-opacity-50 ring-primary-200 \ + focus:ring-primary-200\n\ + \ \ + focus:ring-[1px] focus:outline-none"; + ]; + ] + (); + ]; + div + ~a:[ a_class [ "my-4 flex items-center" ] ] + [ + input + ~a: + [ + a_input_type `Checkbox; + a_name "block_compressed"; + a_id "block_compressed"; + a_class [ "accent-primary-500 mr-2 w-6 h-6" ]; + ] + (); + label + ~a: + [ + a_class [ "text-sm font-medium" ]; a_label_for "data"; + ] + [ txt "Is this file compressed?" ]; + ]; + ]; + ]; + div + ~a:[ a_class [ "my-6" ] ] + [ + button + ~a: + [ + a_id "create-block-button"; + a_onclick "createVolume()"; + a_class + [ + "py-3 px-3 rounded bg-primary-500 hover:bg-primary-800 \ + w-full text-gray-50 font-semibold"; + ]; + ] + [ txt "Create Volume" ]; + ]; + ]) From aca1cf5f9b9a169edcc5c6c3808ae4ec342d36ad Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 08:09:44 +0100 Subject: [PATCH 07/19] display a view button but have it disabled for unused volumes --- assets/style.css | 2 +- volume_index.ml | 19 ++++++++++++++++--- 2 files changed, 17 insertions(+), 4 deletions(-) diff --git a/assets/style.css b/assets/style.css index 56497af1..afc68187 100755 --- a/assets/style.css +++ b/assets/style.css @@ -1 +1 @@ -/*! tailwindcss v3.4.4 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb}:after,:before{--tw-content:""}:host,html{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:initial}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:initial;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:initial}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dd,dl,figure,h1,h2,h3,h4,h5,h6,hr,p,pre{margin:0}fieldset{margin:0}fieldset,legend{padding:0}menu,ol,ul{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}[role=button],button{cursor:pointer}:disabled{cursor:default}audio,canvas,embed,iframe,img,object,svg,video{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]{display:none}*,::backdrop,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:#3b82f680;--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }.container{width:100%}@media (min-width:640px){.container{max-width:640px}}@media (min-width:768px){.container{max-width:768px}}@media (min-width:1024px){.container{max-width:1024px}}@media (min-width:1280px){.container{max-width:1280px}}@media (min-width:1536px){.container{max-width:1536px}}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border-width:0}.pointer-events-none{pointer-events:none}.pointer-events-auto{pointer-events:auto}.visible{visibility:visible}.static{position:static}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.inset-0{inset:0}.inset-1{inset:.25rem}.-inset-x-12{left:-3rem;right:-3rem}.inset-x-1{left:.25rem;right:.25rem}.inset-x-4{left:1rem;right:1rem}.-bottom-0{bottom:0}.-bottom-0\.5{bottom:-.125rem}.-bottom-2{bottom:-.5rem}.-bottom-6{bottom:-1.5rem}.-left-1\/4{left:-25%}.-left-2{left:-.5rem}.-top-1\/4{top:-25%}.-top-4{top:-1rem}.-top-6{top:-1.5rem}.bottom-0{bottom:0}.bottom-10{bottom:2.5rem}.bottom-6{bottom:1.5rem}.left-0{left:0}.left-1\/2{left:50%}.left-8{left:2rem}.right-4{right:1rem}.top-0{top:0}.top-1\/4{top:25%}.z-30{z-index:30}.z-50{z-index:50}.z-\[-1\]{z-index:-1}.z-\[500\]{z-index:500}.col-span-1{grid-column:span 1/span 1}.col-span-10{grid-column:span 10/span 10}.col-span-2{grid-column:span 2/span 2}.col-span-3{grid-column:span 3/span 3}.col-span-4{grid-column:span 4/span 4}.col-span-7{grid-column:span 7/span 7}.-m-1{margin:-.25rem}.-m-1\.5{margin:-.375rem}.m-auto{margin:auto}.mx-auto{margin-left:auto;margin-right:auto}.my-1{margin-top:.25rem;margin-bottom:.25rem}.my-10{margin-top:2.5rem;margin-bottom:2.5rem}.my-2{margin-top:.5rem;margin-bottom:.5rem}.my-3{margin-top:.75rem;margin-bottom:.75rem}.my-4{margin-top:1rem;margin-bottom:1rem}.my-5{margin-top:1.25rem;margin-bottom:1.25rem}.my-6{margin-top:1.5rem;margin-bottom:1.5rem}.-mb-12{margin-bottom:-3rem}.-mb-6{margin-bottom:-1.5rem}.-ml-2{margin-left:-.5rem}.-mr-2{margin-right:-.5rem}.-mr-2\.5{margin-right:-.625rem}.-mt-1{margin-top:-.25rem}.-mt-24{margin-top:-6rem}.-mt-3{margin-top:-.75rem}.mb-0{margin-bottom:0}.mb-14{margin-bottom:3.5rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-5{margin-bottom:1.25rem}.mb-7{margin-bottom:1.75rem}.mb-8{margin-bottom:2rem}.mb-9{margin-bottom:2.25rem}.mb-auto{margin-bottom:auto}.mb-px{margin-bottom:1px}.ml-1{margin-left:.25rem}.ml-3{margin-left:.75rem}.ml-5{margin-left:1.25rem}.mr-2{margin-right:.5rem}.mr-3{margin-right:.75rem}.mt-1{margin-top:.25rem}.mt-1\.5{margin-top:.375rem}.mt-10{margin-top:2.5rem}.mt-12{margin-top:3rem}.mt-16{margin-top:4rem}.mt-2{margin-top:.5rem}.mt-4{margin-top:1rem}.mt-8{margin-top:2rem}.mt-auto{margin-top:auto}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.table{display:table}.grid{display:grid}.hidden{display:none}.size-2{width:.5rem;height:.5rem}.size-2\.5{width:.625rem;height:.625rem}.h-12{height:3rem}.h-2{height:.5rem}.h-2\.5{height:.625rem}.h-24{height:6rem}.h-32{height:8rem}.h-4{height:1rem}.h-5{height:1.25rem}.h-6{height:1.5rem}.h-8{height:2rem}.h-96{height:24rem}.h-\[150\%\]{height:150%}.h-full{height:100%}.h-px{height:1px}.max-h-screen{max-height:100vh}.min-h-screen{min-height:100vh}.w-0{width:0}.w-11{width:2.75rem}.w-14{width:3.5rem}.w-16{width:4rem}.w-32{width:8rem}.w-4{width:1rem}.w-5{width:1.25rem}.w-6{width:1.5rem}.w-60{width:15rem}.w-\[150\%\]{width:150%}.w-fit{width:-moz-fit-content;width:fit-content}.w-full{width:100%}.w-max{width:-moz-max-content;width:max-content}.w-px{width:1px}.min-w-full{min-width:100%}.max-w-3xl{max-width:48rem}.max-w-5xl{max-width:64rem}.max-w-6xl{max-width:72rem}.max-w-7xl{max-width:80rem}.max-w-\[94\%\]{max-width:94%}.max-w-full{max-width:100%}.max-w-lg{max-width:32rem}.max-w-md{max-width:28rem}.max-w-none{max-width:none}.max-w-sm{max-width:24rem}.max-w-xl{max-width:36rem}.max-w-4xl{max-width:56rem}.max-w-2xl{max-width:42rem}.flex-none{flex:none}.shrink-0{flex-shrink:0}.table-auto{table-layout:auto}.border-collapse{border-collapse:collapse}.origin-bottom{transform-origin:bottom}.-translate-x-1\/2{--tw-translate-x:-50%}.-rotate-3,.-translate-x-1\/2{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-rotate-3{--tw-rotate:-3deg}.rotate-3{--tw-rotate:3deg}.rotate-3,.scale-0{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-0{--tw-scale-x:0;--tw-scale-y:0}.scale-100{--tw-scale-x:1;--tw-scale-y:1}.scale-100,.scale-50{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-50{--tw-scale-x:.5;--tw-scale-y:.5}.scale-x-0{--tw-scale-x:0}.scale-x-0,.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform-gpu{transform:translate3d(var(--tw-translate-x),var(--tw-translate-y),0) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes spin{to{transform:rotate(1turn)}}.animate-spin{animation:spin 1s linear infinite}.cursor-pointer{cursor:pointer}.list-none{list-style-type:none}.appearance-none{-webkit-appearance:none;-moz-appearance:none;appearance:none}.grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.flex-row{flex-direction:row}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-start{align-items:flex-start}.items-end{align-items:flex-end}.items-center{align-items:center}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-items-center{justify-items:center}.gap-0{gap:0}.gap-0\.5{gap:.125rem}.gap-10{gap:2.5rem}.gap-12{gap:3rem}.gap-2{gap:.5rem}.gap-20{gap:5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-8{gap:2rem}.gap-x-2{-moz-column-gap:.5rem;column-gap:.5rem}.gap-x-8{-moz-column-gap:2rem;column-gap:2rem}.gap-y-16{row-gap:4rem}.gap-y-4{row-gap:1rem}.space-x-1>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.25rem*var(--tw-space-x-reverse));margin-left:calc(.25rem*(1 - var(--tw-space-x-reverse)))}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.5rem*var(--tw-space-x-reverse));margin-left:calc(.5rem*(1 - var(--tw-space-x-reverse)))}.space-x-20>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(5rem*var(--tw-space-x-reverse));margin-left:calc(5rem*(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1rem*var(--tw-space-x-reverse));margin-left:calc(1rem*(1 - var(--tw-space-x-reverse)))}.space-x-5>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1.25rem*var(--tw-space-x-reverse));margin-left:calc(1.25rem*(1 - var(--tw-space-x-reverse)))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem*var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.75rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem*var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem*var(--tw-space-y-reverse))}.space-y-5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.25rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.25rem*var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem*var(--tw-space-y-reverse))}.divide-y>:not([hidden])~:not([hidden]){--tw-divide-y-reverse:0;border-top-width:calc(1px*(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px*var(--tw-divide-y-reverse))}.divide-gray-200>:not([hidden])~:not([hidden]){--tw-divide-opacity:1;border-color:rgb(229 231 235/var(--tw-divide-opacity))}.justify-self-start{justify-self:start}.justify-self-end{justify-self:end}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-x-auto{overflow-x:auto}.whitespace-normal{white-space:normal}.whitespace-nowrap{white-space:nowrap}.text-wrap{text-wrap:wrap}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:1rem}.rounded-3xl{border-radius:1.5rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-md{border-radius:.375rem}.rounded-xl{border-radius:.75rem}.rounded-l-\[20px\]{border-top-left-radius:20px;border-bottom-left-radius:20px}.rounded-r-\[20px\]{border-top-right-radius:20px;border-bottom-right-radius:20px}.rounded-r-\[8px\]{border-top-right-radius:8px;border-bottom-right-radius:8px}.rounded-t-full{border-top-left-radius:9999px;border-top-right-radius:9999px}.border{border-width:1px}.border-x-0{border-left-width:0;border-right-width:0}.border-b{border-bottom-width:1px}.border-b-0{border-bottom-width:0}.border-b-2{border-bottom-width:2px}.border-l-0{border-left-width:0}.border-r-0{border-right-width:0}.border-t{border-top-width:1px}.border-t-0{border-top-width:0}.border-none{border-style:none}.border-\[\#D7DFE9\]{--tw-border-opacity:1;border-color:rgb(215 223 233/var(--tw-border-opacity))}.border-gray-300{--tw-border-opacity:1;border-color:rgb(209 213 219/var(--tw-border-opacity))}.border-gray-400{--tw-border-opacity:1;border-color:rgb(156 163 175/var(--tw-border-opacity))}.border-primary-200{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.border-primary-400{--tw-border-opacity:1;border-color:rgb(78 179 161/var(--tw-border-opacity))}.border-primary-500{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.border-primary-600{--tw-border-opacity:1;border-color:rgb(40 121 110/var(--tw-border-opacity))}.border-primary-700{--tw-border-opacity:1;border-color:rgb(35 98 90/var(--tw-border-opacity))}.border-secondary-500{--tw-border-opacity:1;border-color:rgb(255 78 51/var(--tw-border-opacity))}.border-transparent{border-color:#0000}.border-y-secondary-300{--tw-border-opacity:1;border-top-color:rgb(255 170 157/var(--tw-border-opacity));border-bottom-color:rgb(255 170 157/var(--tw-border-opacity))}.bg-black{--tw-bg-opacity:1;background-color:rgb(0 0 0/var(--tw-bg-opacity))}.bg-black\/20{background-color:#0003}.bg-gray-100{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}.bg-gray-50{--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity))}.bg-gray-500\/25{background-color:#6b728040}.bg-primary-100{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.bg-primary-200{--tw-bg-opacity:1;background-color:rgb(171 228 214/var(--tw-bg-opacity))}.bg-primary-300{--tw-bg-opacity:1;background-color:rgb(122 206 189/var(--tw-bg-opacity))}.bg-primary-50{--tw-bg-opacity:1;background-color:rgb(243 250 249/var(--tw-bg-opacity))}.bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.bg-primary-800{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.bg-primary-950{--tw-bg-opacity:1;background-color:rgb(13 38 37/var(--tw-bg-opacity))}.bg-secondary-300{--tw-bg-opacity:1;background-color:rgb(255 170 157/var(--tw-bg-opacity))}.bg-secondary-500{--tw-bg-opacity:1;background-color:rgb(255 78 51/var(--tw-bg-opacity))}.bg-secondary-700{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.bg-transparent{background-color:initial}.bg-opacity-0{--tw-bg-opacity:0}.bg-gradient-to-b{background-image:linear-gradient(to bottom,var(--tw-gradient-stops))}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.bg-gradient-to-t{background-image:linear-gradient(to top,var(--tw-gradient-stops))}.from-gray-100{--tw-gradient-from:#f3f4f6 var(--tw-gradient-from-position);--tw-gradient-to:#f3f4f600 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-900\/50{--tw-gradient-from:#1f423e80 var(--tw-gradient-from-position);--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-950{--tw-gradient-from:#0d2625 var(--tw-gradient-from-position);--tw-gradient-to:#0d262500 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-transparent{--tw-gradient-from:#0000 var(--tw-gradient-from-position);--tw-gradient-to:#0000 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-10\%{--tw-gradient-from-position:10%}.via-primary-900{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-primary-900\/50{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e80 var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-10\%{--tw-gradient-via-position:10%}.to-primary-950{--tw-gradient-to:#0d2625 var(--tw-gradient-to-position)}.to-primary-950\/50{--tw-gradient-to:#0d262580 var(--tw-gradient-to-position)}.to-transparent{--tw-gradient-to:#0000 var(--tw-gradient-to-position)}.to-90\%{--tw-gradient-to-position:90%}.bg-\[length\:100px_auto\]{background-size:100px auto}.bg-cover{background-size:cover}.bg-clip-padding{background-clip:padding-box}.bg-center{background-position:50%}.fill-primary-400{fill:#4eb3a1}.object-contain{-o-object-fit:contain;object-fit:contain}.object-cover{-o-object-fit:cover;object-fit:cover}.object-bottom{-o-object-position:bottom;object-position:bottom}.p-0{padding:0}.p-1{padding:.25rem}.p-1\.5{padding:.375rem}.p-10{padding:2.5rem}.p-16{padding:4rem}.p-2{padding:.5rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.p-6{padding:1.5rem}.px-10{padding-left:2.5rem;padding-right:2.5rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-40{padding-left:10rem;padding-right:10rem}.px-5{padding-left:1.25rem;padding-right:1.25rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.px-7{padding-left:1.75rem;padding-right:1.75rem}.py-0{padding-top:0;padding-bottom:0}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-10{padding-top:2.5rem;padding-bottom:2.5rem}.py-16{padding-top:4rem;padding-bottom:4rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-2\.5{padding-top:.625rem;padding-bottom:.625rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-5{padding-top:1.25rem;padding-bottom:1.25rem}.py-6{padding-top:1.5rem;padding-bottom:1.5rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pb-12{padding-bottom:3rem}.pb-16{padding-bottom:4rem}.pb-3{padding-bottom:.75rem}.pb-3\.5{padding-bottom:.875rem}.pb-32{padding-bottom:8rem}.pb-8{padding-bottom:2rem}.pl-3{padding-left:.75rem}.pl-4{padding-left:1rem}.pr-3{padding-right:.75rem}.pr-7{padding-right:1.75rem}.pr-\[10px\]{padding-right:10px}.pt-12{padding-top:3rem}.pt-24{padding-top:6rem}.pt-4{padding-top:1rem}.text-left{text-align:left}.text-center{text-align:center}.text-right{text-align:right}.text-start{text-align:start}.align-middle{vertical-align:middle}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-5xl{font-size:3rem;line-height:1}.text-7xl{font-size:4.5rem;line-height:1}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-\[450\]{font-weight:450}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.lowercase{text-transform:lowercase}.leading-6{line-height:1.5rem}.leading-none{line-height:1}.leading-tight{line-height:1.25}.tracking-tight{letter-spacing:-.025em}.tracking-wide{letter-spacing:.025em}.tracking-wider{letter-spacing:.05em}.text-gray-100{--tw-text-opacity:1;color:rgb(243 244 246/var(--tw-text-opacity))}.text-gray-200{--tw-text-opacity:1;color:rgb(229 231 235/var(--tw-text-opacity))}.text-gray-50{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.text-gray-500{--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity))}.text-gray-600{--tw-text-opacity:1;color:rgb(75 85 99/var(--tw-text-opacity))}.text-gray-700{--tw-text-opacity:1;color:rgb(55 65 81/var(--tw-text-opacity))}.text-gray-800{--tw-text-opacity:1;color:rgb(31 41 55/var(--tw-text-opacity))}.text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.text-primary-300{--tw-text-opacity:1;color:rgb(122 206 189/var(--tw-text-opacity))}.text-primary-400{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.text-primary-50{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.text-primary-500{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.text-primary-600{--tw-text-opacity:1;color:rgb(40 121 110/var(--tw-text-opacity))}.text-primary-800{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.text-primary-900{--tw-text-opacity:1;color:rgb(31 66 62/var(--tw-text-opacity))}.text-primary-950{--tw-text-opacity:1;color:rgb(13 38 37/var(--tw-text-opacity))}.text-secondary-300{--tw-text-opacity:1;color:rgb(255 170 157/var(--tw-text-opacity))}.text-secondary-50{--tw-text-opacity:1;color:rgb(255 243 241/var(--tw-text-opacity))}.text-secondary-500{--tw-text-opacity:1;color:rgb(255 78 51/var(--tw-text-opacity))}.text-secondary-700{--tw-text-opacity:1;color:rgb(200 38 13/var(--tw-text-opacity))}.accent-primary-500{accent-color:#369c8c}.opacity-0{opacity:0}.opacity-100{opacity:1}.opacity-50{opacity:.5}.shadow{--tw-shadow:0 1px 3px 0 #0000001a,0 1px 2px -1px #0000001a;--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color)}.shadow,.shadow-lg{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px #0000001a,0 4px 6px -4px #0000001a;--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.shadow-md{--tw-shadow:0 4px 6px -1px #0000001a,0 2px 4px -2px #0000001a;--tw-shadow-colored:0 4px 6px -1px var(--tw-shadow-color),0 2px 4px -2px var(--tw-shadow-color)}.shadow-md,.shadow-sm{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 2px 0 #0000000d;--tw-shadow-colored:0 1px 2px 0 var(--tw-shadow-color)}.shadow-gray-800\/5{--tw-shadow-color:#1f29370d;--tw-shadow:var(--tw-shadow-colored)}.outline-0{outline-width:0}.ring-1{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.ring-inset{--tw-ring-inset:inset}.ring-gray-800\/\[\.075\]{--tw-ring-color:rgba(31,41,55,.075)}.ring-primary-100{--tw-ring-opacity:1;--tw-ring-color:rgb(213 242 235/var(--tw-ring-opacity))}.ring-primary-200{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.blur{--tw-blur:blur(8px)}.blur,.blur-3xl{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.blur-3xl{--tw-blur:blur(64px)}.brightness-0{--tw-brightness:brightness(0)}.brightness-0,.drop-shadow{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.drop-shadow{--tw-drop-shadow:drop-shadow(0 1px 2px #0000001a) drop-shadow(0 1px 1px #0000000f)}.drop-shadow-sm{--tw-drop-shadow:drop-shadow(0 1px 1px #0000000d)}.drop-shadow-sm,.invert{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.invert{--tw-invert:invert(100%)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur-md{--tw-backdrop-blur:blur(12px)}.backdrop-blur-md,.backdrop-blur-sm{-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-sm{--tw-backdrop-blur:blur(4px)}.backdrop-blur-xl{--tw-backdrop-blur:blur(24px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.delay-100{transition-delay:.1s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.duration-500{transition-duration:.5s}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}.\[-webkit-mask-image\:linear-gradient\(to_bottom\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(180deg,#fff 75%,#fff0)}.\[-webkit-mask-image\:linear-gradient\(to_top\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(0deg,#fff 75%,#fff0)}.after\:absolute:after{content:var(--tw-content);position:absolute}.after\:bottom-0:after{content:var(--tw-content);bottom:0}.after\:left-\[0\.0625rem\]:after{content:var(--tw-content);left:.0625rem}.after\:top-0:after{content:var(--tw-content);top:0}.after\:my-auto:after{content:var(--tw-content);margin-top:auto;margin-bottom:auto}.after\:h-5:after{content:var(--tw-content);height:1.25rem}.after\:w-5:after{content:var(--tw-content);width:1.25rem}.after\:rounded-full:after{content:var(--tw-content);border-radius:9999px}.after\:bg-gray-600:after{content:var(--tw-content);--tw-bg-opacity:1;background-color:rgb(75 85 99/var(--tw-bg-opacity))}.after\:transition-all:after{content:var(--tw-content);transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.after\:content-\[\'\'\]:after{--tw-content:"";content:var(--tw-content)}.hover\:isolate:hover{isolation:isolate}.hover\:border-primary-200:hover{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.hover\:border-transparent:hover{border-color:#0000}.hover\:bg-gray-200:hover{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.hover\:bg-primary-100:hover{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.hover\:bg-primary-700:hover{--tw-bg-opacity:1;background-color:rgb(35 98 90/var(--tw-bg-opacity))}.hover\:bg-primary-800:hover{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.hover\:bg-secondary-100:hover{--tw-bg-opacity:1;background-color:rgb(255 227 223/var(--tw-bg-opacity))}.hover\:bg-secondary-700:hover{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.hover\:bg-opacity-50:hover{--tw-bg-opacity:0.5}.hover\:font-bold:hover{font-weight:700}.hover\:text-gray-50:hover{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.hover\:text-primary-400:hover{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.hover\:text-primary-50:hover{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.hover\:text-primary-500:hover{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.hover\:text-primary-700:hover{--tw-text-opacity:1;color:rgb(35 98 90/var(--tw-text-opacity))}.hover\:text-primary-800:hover{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.hover\:opacity-75:hover{opacity:.75}.focus\:isolate:focus{isolation:isolate}.focus\:border-primary-300:focus{--tw-border-opacity:1;border-color:rgb(122 206 189/var(--tw-border-opacity))}.focus\:border-primary-500:focus{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.focus\:border-transparent:focus{border-color:#0000}.focus\:bg-opacity-50:focus{--tw-bg-opacity:0.5}.focus\:text-primary-800:focus{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.focus\:outline-none:focus{outline:2px solid #0000;outline-offset:2px}.focus\:ring-\[1px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-\[1px\]:focus,.focus\:ring-\[3px\]:focus{box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.focus\:ring-\[3px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-primary-200:focus{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.focus-visible\:outline:focus-visible{outline-style:solid}.focus-visible\:outline-2:focus-visible{outline-width:2px}.focus-visible\:outline-offset-2:focus-visible{outline-offset:2px}.focus-visible\:outline-black:focus-visible{outline-color:#000}.active\:opacity-100:active{opacity:1}.active\:outline-offset-0:active{outline-offset:0}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:opacity-50:disabled{opacity:.5}.group\/btn:hover .group-hover\/btn\:w-2{width:.5rem}.group\/btn:hover .group-hover\/btn\:w-2\.5{width:.625rem}.group:hover .group-hover\:scale-100{--tw-scale-x:1;--tw-scale-y:1}.group:hover .group-hover\:scale-100,.group:hover .group-hover\:scale-x-100{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.group:hover .group-hover\:scale-x-100{--tw-scale-x:1}.group:hover .group-hover\:opacity-100,.group\/btn:hover .group-hover\/btn\:opacity-100{opacity:1}.peer:checked~.peer-checked\:bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.peer:checked~.peer-checked\:text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.peer:checked~.peer-checked\:after\:translate-x-5:after{content:var(--tw-content);--tw-translate-x:1.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.peer:focus~.peer-focus\:outline{outline-style:solid}.peer:focus~.peer-focus\:outline-2{outline-width:2px}.peer:focus~.peer-focus\:outline-offset-2{outline-offset:2px}.peer:focus~.peer-focus\:outline-gray-800{outline-color:#1f2937}.peer:focus:checked~.peer-focus\:peer-checked\:outline-primary-500{outline-color:#369c8c}.peer:active~.peer-active\:outline-offset-0{outline-offset:0}.peer:disabled~.peer-disabled\:cursor-not-allowed{cursor:not-allowed}.peer:disabled~.peer-disabled\:opacity-70{opacity:.7}@media (prefers-reduced-motion:reduce){.motion-reduce\:transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}}@media (min-width:640px){.sm\:left-14{left:3.5rem}.sm\:w-auto{width:auto}.sm\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.sm\:flex-row{flex-direction:row}.sm\:items-center{align-items:center}.sm\:px-6{padding-left:1.5rem;padding-right:1.5rem}.sm\:pb-40{padding-bottom:10rem}.sm\:pl-20{padding-left:5rem}.sm\:text-3xl{font-size:1.875rem;line-height:2.25rem}.sm\:text-5xl{font-size:3rem;line-height:1}.sm\:text-sm{font-size:.875rem;line-height:1.25rem}}@media (min-width:768px){.md\:block{display:block}.md\:h-screen{height:100vh}.md\:w-16{width:4rem}.md\:w-20{width:5rem}.md\:w-24{width:6rem}.md\:max-w-2xl{max-width:42rem}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.md\:justify-end{justify-content:flex-end}.md\:gap-16{gap:4rem}.md\:px-14{padding-left:3.5rem;padding-right:3.5rem}.md\:py-10{padding-top:2.5rem;padding-bottom:2.5rem}.md\:pb-52{padding-bottom:13rem}.md\:text-3xl{font-size:1.875rem;line-height:2.25rem}.md\:text-4xl{font-size:2.25rem;line-height:2.5rem}.md\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width:1024px){.lg\:left-20{left:5rem}.lg\:order-last{order:9999}.lg\:col-span-3{grid-column:span 3/span 3}.lg\:col-span-4{grid-column:span 4/span 4}.lg\:-my-11{margin-top:-2.75rem;margin-bottom:-2.75rem}.lg\:-mr-9{margin-right:-2.25rem}.lg\:mb-0{margin-bottom:0}.lg\:mb-16{margin-bottom:4rem}.lg\:mt-0{margin-top:0}.lg\:block{display:block}.lg\:flex{display:flex}.lg\:grid{display:grid}.lg\:max-w-4xl{max-width:56rem}.lg\:max-w-7xl{max-width:80rem}.lg\:max-w-max{max-width:-moz-max-content;max-width:max-content}.lg\:max-w-xl{max-width:36rem}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-7{grid-template-columns:repeat(7,minmax(0,1fr))}.lg\:gap-5{gap:1.25rem}.lg\:p-10{padding:2.5rem}.lg\:p-8{padding:2rem}.lg\:px-8{padding-left:2rem;padding-right:2rem}.lg\:py-12{padding-top:3rem;padding-bottom:3rem}.lg\:py-24{padding-top:6rem;padding-bottom:6rem}.lg\:pb-20{padding-bottom:5rem}.lg\:pb-60{padding-bottom:15rem}.lg\:pl-28{padding-left:7rem}.lg\:pt-20{padding-top:5rem}.lg\:pt-40{padding-top:10rem}.lg\:text-4xl{font-size:2.25rem;line-height:2.5rem}.lg\:text-5xl{font-size:3rem;line-height:1}.lg\:text-6xl{font-size:3.75rem;line-height:1}.lg\:text-lg{font-size:1.125rem;line-height:1.75rem}}@media (min-width:1280px){.xl\:col-span-3{grid-column:span 3/span 3}.xl\:mx-auto{margin-left:auto;margin-right:auto}.xl\:flex{display:flex}.xl\:hidden{display:none}.xl\:w-auto{width:auto}.xl\:w-full{width:100%}.xl\:max-w-lg{max-width:32rem}.xl\:max-w-xl{max-width:36rem}.xl\:gap-16{gap:4rem}.xl\:gap-x-16{-moz-column-gap:4rem;column-gap:4rem}.xl\:px-20{padding-left:5rem;padding-right:5rem}.xl\:py-20{padding-top:5rem;padding-bottom:5rem}.xl\:py-32{padding-top:8rem}.xl\:pb-32,.xl\:py-32{padding-bottom:8rem}.xl\:pb-\[16\.5rem\]{padding-bottom:16.5rem}.xl\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (prefers-color-scheme:dark){.dark\:border-gray-700{--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}.dark\:bg-gray-900{--tw-bg-opacity:1;background-color:rgb(17 24 39/var(--tw-bg-opacity))}.dark\:text-black{--tw-text-opacity:1;color:rgb(0 0 0/var(--tw-text-opacity))}.dark\:text-gray-300{--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity))}.dark\:after\:bg-gray-300:after{content:var(--tw-content);--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity))}.peer:checked~.dark\:peer-checked\:bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.peer:focus~.dark\:peer-focus\:outline-gray-300{outline-color:#d1d5db}.peer:focus:checked~.dark\:peer-focus\:peer-checked\:outline-primary-500{outline-color:#369c8c}} +/*! tailwindcss v3.4.4 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb}:after,:before{--tw-content:""}:host,html{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:initial}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:initial;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:initial}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dd,dl,figure,h1,h2,h3,h4,h5,h6,hr,p,pre{margin:0}fieldset{margin:0}fieldset,legend{padding:0}menu,ol,ul{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}[role=button],button{cursor:pointer}:disabled{cursor:default}audio,canvas,embed,iframe,img,object,svg,video{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]{display:none}*,::backdrop,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:#3b82f680;--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }.container{width:100%}@media (min-width:640px){.container{max-width:640px}}@media (min-width:768px){.container{max-width:768px}}@media (min-width:1024px){.container{max-width:1024px}}@media (min-width:1280px){.container{max-width:1280px}}@media (min-width:1536px){.container{max-width:1536px}}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border-width:0}.pointer-events-none{pointer-events:none}.pointer-events-auto{pointer-events:auto}.visible{visibility:visible}.static{position:static}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.inset-0{inset:0}.inset-1{inset:.25rem}.-inset-x-12{left:-3rem;right:-3rem}.inset-x-1{left:.25rem;right:.25rem}.inset-x-4{left:1rem;right:1rem}.-bottom-0{bottom:0}.-bottom-0\.5{bottom:-.125rem}.-bottom-2{bottom:-.5rem}.-bottom-6{bottom:-1.5rem}.-left-1\/4{left:-25%}.-left-2{left:-.5rem}.-top-1\/4{top:-25%}.-top-4{top:-1rem}.-top-6{top:-1.5rem}.bottom-0{bottom:0}.bottom-10{bottom:2.5rem}.bottom-6{bottom:1.5rem}.left-0{left:0}.left-1\/2{left:50%}.left-8{left:2rem}.right-4{right:1rem}.top-0{top:0}.top-1\/4{top:25%}.z-30{z-index:30}.z-50{z-index:50}.z-\[-1\]{z-index:-1}.z-\[500\]{z-index:500}.col-span-1{grid-column:span 1/span 1}.col-span-10{grid-column:span 10/span 10}.col-span-2{grid-column:span 2/span 2}.col-span-3{grid-column:span 3/span 3}.col-span-4{grid-column:span 4/span 4}.col-span-7{grid-column:span 7/span 7}.-m-1{margin:-.25rem}.-m-1\.5{margin:-.375rem}.m-auto{margin:auto}.mx-auto{margin-left:auto;margin-right:auto}.my-1{margin-top:.25rem;margin-bottom:.25rem}.my-10{margin-top:2.5rem;margin-bottom:2.5rem}.my-2{margin-top:.5rem;margin-bottom:.5rem}.my-3{margin-top:.75rem;margin-bottom:.75rem}.my-4{margin-top:1rem;margin-bottom:1rem}.my-5{margin-top:1.25rem;margin-bottom:1.25rem}.my-6{margin-top:1.5rem;margin-bottom:1.5rem}.-mb-12{margin-bottom:-3rem}.-mb-6{margin-bottom:-1.5rem}.-ml-2{margin-left:-.5rem}.-mr-2{margin-right:-.5rem}.-mr-2\.5{margin-right:-.625rem}.-mt-1{margin-top:-.25rem}.-mt-24{margin-top:-6rem}.-mt-3{margin-top:-.75rem}.mb-0{margin-bottom:0}.mb-14{margin-bottom:3.5rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-5{margin-bottom:1.25rem}.mb-7{margin-bottom:1.75rem}.mb-8{margin-bottom:2rem}.mb-9{margin-bottom:2.25rem}.mb-auto{margin-bottom:auto}.mb-px{margin-bottom:1px}.ml-1{margin-left:.25rem}.ml-3{margin-left:.75rem}.ml-5{margin-left:1.25rem}.mr-2{margin-right:.5rem}.mr-3{margin-right:.75rem}.mt-1{margin-top:.25rem}.mt-1\.5{margin-top:.375rem}.mt-10{margin-top:2.5rem}.mt-12{margin-top:3rem}.mt-16{margin-top:4rem}.mt-2{margin-top:.5rem}.mt-4{margin-top:1rem}.mt-8{margin-top:2rem}.mt-auto{margin-top:auto}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.table{display:table}.grid{display:grid}.hidden{display:none}.size-2{width:.5rem;height:.5rem}.size-2\.5{width:.625rem;height:.625rem}.h-12{height:3rem}.h-2{height:.5rem}.h-2\.5{height:.625rem}.h-24{height:6rem}.h-32{height:8rem}.h-4{height:1rem}.h-6{height:1.5rem}.h-8{height:2rem}.h-96{height:24rem}.h-\[150\%\]{height:150%}.h-full{height:100%}.h-px{height:1px}.max-h-screen{max-height:100vh}.min-h-screen{min-height:100vh}.w-0{width:0}.w-11{width:2.75rem}.w-14{width:3.5rem}.w-16{width:4rem}.w-32{width:8rem}.w-4{width:1rem}.w-6{width:1.5rem}.w-60{width:15rem}.w-\[150\%\]{width:150%}.w-fit{width:-moz-fit-content;width:fit-content}.w-full{width:100%}.w-max{width:-moz-max-content;width:max-content}.w-px{width:1px}.min-w-full{min-width:100%}.max-w-3xl{max-width:48rem}.max-w-5xl{max-width:64rem}.max-w-6xl{max-width:72rem}.max-w-7xl{max-width:80rem}.max-w-\[94\%\]{max-width:94%}.max-w-full{max-width:100%}.max-w-lg{max-width:32rem}.max-w-md{max-width:28rem}.max-w-none{max-width:none}.max-w-sm{max-width:24rem}.max-w-xl{max-width:36rem}.flex-none{flex:none}.shrink-0{flex-shrink:0}.table-auto{table-layout:auto}.border-collapse{border-collapse:collapse}.origin-bottom{transform-origin:bottom}.-translate-x-1\/2{--tw-translate-x:-50%}.-rotate-3,.-translate-x-1\/2{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-rotate-3{--tw-rotate:-3deg}.rotate-3{--tw-rotate:3deg}.rotate-3,.scale-0{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-0{--tw-scale-x:0;--tw-scale-y:0}.scale-100{--tw-scale-x:1;--tw-scale-y:1}.scale-100,.scale-50{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-50{--tw-scale-x:.5;--tw-scale-y:.5}.scale-x-0{--tw-scale-x:0}.scale-x-0,.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform-gpu{transform:translate3d(var(--tw-translate-x),var(--tw-translate-y),0) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes spin{to{transform:rotate(1turn)}}.animate-spin{animation:spin 1s linear infinite}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.list-none{list-style-type:none}.appearance-none{-webkit-appearance:none;-moz-appearance:none;appearance:none}.grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.flex-row{flex-direction:row}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-start{align-items:flex-start}.items-end{align-items:flex-end}.items-center{align-items:center}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-items-center{justify-items:center}.gap-0{gap:0}.gap-0\.5{gap:.125rem}.gap-10{gap:2.5rem}.gap-12{gap:3rem}.gap-20{gap:5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-8{gap:2rem}.gap-x-2{-moz-column-gap:.5rem;column-gap:.5rem}.gap-x-8{-moz-column-gap:2rem;column-gap:2rem}.gap-y-16{row-gap:4rem}.gap-y-4{row-gap:1rem}.space-x-1>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.25rem*var(--tw-space-x-reverse));margin-left:calc(.25rem*(1 - var(--tw-space-x-reverse)))}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.5rem*var(--tw-space-x-reverse));margin-left:calc(.5rem*(1 - var(--tw-space-x-reverse)))}.space-x-20>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(5rem*var(--tw-space-x-reverse));margin-left:calc(5rem*(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1rem*var(--tw-space-x-reverse));margin-left:calc(1rem*(1 - var(--tw-space-x-reverse)))}.space-x-5>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(1.25rem*var(--tw-space-x-reverse));margin-left:calc(1.25rem*(1 - var(--tw-space-x-reverse)))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem*var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.75rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem*var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem*var(--tw-space-y-reverse))}.space-y-5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.25rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.25rem*var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem*var(--tw-space-y-reverse))}.divide-y>:not([hidden])~:not([hidden]){--tw-divide-y-reverse:0;border-top-width:calc(1px*(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px*var(--tw-divide-y-reverse))}.divide-gray-200>:not([hidden])~:not([hidden]){--tw-divide-opacity:1;border-color:rgb(229 231 235/var(--tw-divide-opacity))}.justify-self-start{justify-self:start}.justify-self-end{justify-self:end}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-x-auto{overflow-x:auto}.whitespace-normal{white-space:normal}.whitespace-nowrap{white-space:nowrap}.text-wrap{text-wrap:wrap}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:1rem}.rounded-3xl{border-radius:1.5rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-md{border-radius:.375rem}.rounded-xl{border-radius:.75rem}.rounded-l-\[20px\]{border-top-left-radius:20px;border-bottom-left-radius:20px}.rounded-r-\[20px\]{border-top-right-radius:20px;border-bottom-right-radius:20px}.rounded-r-\[8px\]{border-top-right-radius:8px;border-bottom-right-radius:8px}.rounded-t-full{border-top-left-radius:9999px;border-top-right-radius:9999px}.border{border-width:1px}.border-x-0{border-left-width:0;border-right-width:0}.border-b{border-bottom-width:1px}.border-b-0{border-bottom-width:0}.border-b-2{border-bottom-width:2px}.border-l-0{border-left-width:0}.border-r-0{border-right-width:0}.border-t-0{border-top-width:0}.border-none{border-style:none}.border-\[\#D7DFE9\]{--tw-border-opacity:1;border-color:rgb(215 223 233/var(--tw-border-opacity))}.border-gray-300{--tw-border-opacity:1;border-color:rgb(209 213 219/var(--tw-border-opacity))}.border-gray-400{--tw-border-opacity:1;border-color:rgb(156 163 175/var(--tw-border-opacity))}.border-primary-200{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.border-primary-400{--tw-border-opacity:1;border-color:rgb(78 179 161/var(--tw-border-opacity))}.border-primary-500{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.border-primary-600{--tw-border-opacity:1;border-color:rgb(40 121 110/var(--tw-border-opacity))}.border-primary-700{--tw-border-opacity:1;border-color:rgb(35 98 90/var(--tw-border-opacity))}.border-secondary-500{--tw-border-opacity:1;border-color:rgb(255 78 51/var(--tw-border-opacity))}.border-transparent{border-color:#0000}.border-y-secondary-300{--tw-border-opacity:1;border-top-color:rgb(255 170 157/var(--tw-border-opacity));border-bottom-color:rgb(255 170 157/var(--tw-border-opacity))}.bg-black\/20{background-color:#0003}.bg-gray-100{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}.bg-gray-50{--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity))}.bg-gray-500\/25{background-color:#6b728040}.bg-primary-100{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.bg-primary-200{--tw-bg-opacity:1;background-color:rgb(171 228 214/var(--tw-bg-opacity))}.bg-primary-300{--tw-bg-opacity:1;background-color:rgb(122 206 189/var(--tw-bg-opacity))}.bg-primary-50{--tw-bg-opacity:1;background-color:rgb(243 250 249/var(--tw-bg-opacity))}.bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.bg-primary-800{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.bg-primary-950{--tw-bg-opacity:1;background-color:rgb(13 38 37/var(--tw-bg-opacity))}.bg-secondary-300{--tw-bg-opacity:1;background-color:rgb(255 170 157/var(--tw-bg-opacity))}.bg-secondary-500{--tw-bg-opacity:1;background-color:rgb(255 78 51/var(--tw-bg-opacity))}.bg-secondary-700{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.bg-transparent{background-color:initial}.bg-opacity-0{--tw-bg-opacity:0}.bg-gradient-to-b{background-image:linear-gradient(to bottom,var(--tw-gradient-stops))}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.bg-gradient-to-t{background-image:linear-gradient(to top,var(--tw-gradient-stops))}.from-gray-100{--tw-gradient-from:#f3f4f6 var(--tw-gradient-from-position);--tw-gradient-to:#f3f4f600 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-900\/50{--tw-gradient-from:#1f423e80 var(--tw-gradient-from-position);--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-primary-950{--tw-gradient-from:#0d2625 var(--tw-gradient-from-position);--tw-gradient-to:#0d262500 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-transparent{--tw-gradient-from:#0000 var(--tw-gradient-from-position);--tw-gradient-to:#0000 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-10\%{--tw-gradient-from-position:10%}.via-primary-900{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-primary-900\/50{--tw-gradient-to:#1f423e00 var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),#1f423e80 var(--tw-gradient-via-position),var(--tw-gradient-to)}.via-10\%{--tw-gradient-via-position:10%}.to-primary-950{--tw-gradient-to:#0d2625 var(--tw-gradient-to-position)}.to-primary-950\/50{--tw-gradient-to:#0d262580 var(--tw-gradient-to-position)}.to-transparent{--tw-gradient-to:#0000 var(--tw-gradient-to-position)}.to-90\%{--tw-gradient-to-position:90%}.bg-\[length\:100px_auto\]{background-size:100px auto}.bg-cover{background-size:cover}.bg-clip-padding{background-clip:padding-box}.bg-center{background-position:50%}.fill-primary-400{fill:#4eb3a1}.object-contain{-o-object-fit:contain;object-fit:contain}.object-cover{-o-object-fit:cover;object-fit:cover}.object-bottom{-o-object-position:bottom;object-position:bottom}.p-0{padding:0}.p-1{padding:.25rem}.p-1\.5{padding:.375rem}.p-10{padding:2.5rem}.p-16{padding:4rem}.p-2{padding:.5rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.p-6{padding:1.5rem}.px-10{padding-left:2.5rem;padding-right:2.5rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-40{padding-left:10rem;padding-right:10rem}.px-5{padding-left:1.25rem;padding-right:1.25rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.px-7{padding-left:1.75rem;padding-right:1.75rem}.py-0{padding-top:0;padding-bottom:0}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-10{padding-top:2.5rem;padding-bottom:2.5rem}.py-16{padding-top:4rem;padding-bottom:4rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-2\.5{padding-top:.625rem;padding-bottom:.625rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-5{padding-top:1.25rem;padding-bottom:1.25rem}.py-6{padding-top:1.5rem;padding-bottom:1.5rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pb-12{padding-bottom:3rem}.pb-16{padding-bottom:4rem}.pb-3{padding-bottom:.75rem}.pb-3\.5{padding-bottom:.875rem}.pb-32{padding-bottom:8rem}.pb-8{padding-bottom:2rem}.pl-3{padding-left:.75rem}.pl-4{padding-left:1rem}.pr-3{padding-right:.75rem}.pr-7{padding-right:1.75rem}.pr-\[10px\]{padding-right:10px}.pt-12{padding-top:3rem}.pt-24{padding-top:6rem}.pt-4{padding-top:1rem}.text-left{text-align:left}.text-center{text-align:center}.text-right{text-align:right}.text-start{text-align:start}.align-middle{vertical-align:middle}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-5xl{font-size:3rem;line-height:1}.text-7xl{font-size:4.5rem;line-height:1}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-\[450\]{font-weight:450}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.lowercase{text-transform:lowercase}.leading-6{line-height:1.5rem}.leading-none{line-height:1}.leading-tight{line-height:1.25}.tracking-tight{letter-spacing:-.025em}.tracking-wide{letter-spacing:.025em}.tracking-wider{letter-spacing:.05em}.text-gray-100{--tw-text-opacity:1;color:rgb(243 244 246/var(--tw-text-opacity))}.text-gray-200{--tw-text-opacity:1;color:rgb(229 231 235/var(--tw-text-opacity))}.text-gray-300{--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity))}.text-gray-50{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.text-gray-500{--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity))}.text-gray-600{--tw-text-opacity:1;color:rgb(75 85 99/var(--tw-text-opacity))}.text-gray-700{--tw-text-opacity:1;color:rgb(55 65 81/var(--tw-text-opacity))}.text-gray-800{--tw-text-opacity:1;color:rgb(31 41 55/var(--tw-text-opacity))}.text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.text-primary-300{--tw-text-opacity:1;color:rgb(122 206 189/var(--tw-text-opacity))}.text-primary-400{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.text-primary-50{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.text-primary-500{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.text-primary-600{--tw-text-opacity:1;color:rgb(40 121 110/var(--tw-text-opacity))}.text-primary-800{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.text-primary-900{--tw-text-opacity:1;color:rgb(31 66 62/var(--tw-text-opacity))}.text-primary-950{--tw-text-opacity:1;color:rgb(13 38 37/var(--tw-text-opacity))}.text-secondary-300{--tw-text-opacity:1;color:rgb(255 170 157/var(--tw-text-opacity))}.text-secondary-50{--tw-text-opacity:1;color:rgb(255 243 241/var(--tw-text-opacity))}.text-secondary-500{--tw-text-opacity:1;color:rgb(255 78 51/var(--tw-text-opacity))}.text-secondary-700{--tw-text-opacity:1;color:rgb(200 38 13/var(--tw-text-opacity))}.accent-primary-500{accent-color:#369c8c}.opacity-0{opacity:0}.opacity-100{opacity:1}.opacity-50{opacity:.5}.shadow{--tw-shadow:0 1px 3px 0 #0000001a,0 1px 2px -1px #0000001a;--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color)}.shadow,.shadow-lg{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px #0000001a,0 4px 6px -4px #0000001a;--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.shadow-md{--tw-shadow:0 4px 6px -1px #0000001a,0 2px 4px -2px #0000001a;--tw-shadow-colored:0 4px 6px -1px var(--tw-shadow-color),0 2px 4px -2px var(--tw-shadow-color)}.shadow-md,.shadow-sm{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 2px 0 #0000000d;--tw-shadow-colored:0 1px 2px 0 var(--tw-shadow-color)}.shadow-gray-800\/5{--tw-shadow-color:#1f29370d;--tw-shadow:var(--tw-shadow-colored)}.outline-0{outline-width:0}.ring-1{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.ring-inset{--tw-ring-inset:inset}.ring-gray-800\/\[\.075\]{--tw-ring-color:rgba(31,41,55,.075)}.ring-primary-100{--tw-ring-opacity:1;--tw-ring-color:rgb(213 242 235/var(--tw-ring-opacity))}.ring-primary-200{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.blur{--tw-blur:blur(8px)}.blur,.blur-3xl{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.blur-3xl{--tw-blur:blur(64px)}.brightness-0{--tw-brightness:brightness(0)}.brightness-0,.drop-shadow{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.drop-shadow{--tw-drop-shadow:drop-shadow(0 1px 2px #0000001a) drop-shadow(0 1px 1px #0000000f)}.drop-shadow-sm{--tw-drop-shadow:drop-shadow(0 1px 1px #0000000d)}.drop-shadow-sm,.invert{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.invert{--tw-invert:invert(100%)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur-md{--tw-backdrop-blur:blur(12px)}.backdrop-blur-md,.backdrop-blur-sm{-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-sm{--tw-backdrop-blur:blur(4px)}.backdrop-blur-xl{--tw-backdrop-blur:blur(24px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.delay-100{transition-delay:.1s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.duration-500{transition-duration:.5s}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}.\[-webkit-mask-image\:linear-gradient\(to_bottom\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(180deg,#fff 75%,#fff0)}.\[-webkit-mask-image\:linear-gradient\(to_top\2c rgba\(255\2c 255\2c 255\2c 1\)_75\%\2c rgba\(255\2c 255\2c 255\2c 0\)\)\]{-webkit-mask-image:linear-gradient(0deg,#fff 75%,#fff0)}.after\:absolute:after{content:var(--tw-content);position:absolute}.after\:bottom-0:after{content:var(--tw-content);bottom:0}.after\:left-\[0\.0625rem\]:after{content:var(--tw-content);left:.0625rem}.after\:top-0:after{content:var(--tw-content);top:0}.after\:my-auto:after{content:var(--tw-content);margin-top:auto;margin-bottom:auto}.after\:h-5:after{content:var(--tw-content);height:1.25rem}.after\:w-5:after{content:var(--tw-content);width:1.25rem}.after\:rounded-full:after{content:var(--tw-content);border-radius:9999px}.after\:bg-gray-600:after{content:var(--tw-content);--tw-bg-opacity:1;background-color:rgb(75 85 99/var(--tw-bg-opacity))}.after\:transition-all:after{content:var(--tw-content);transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.after\:content-\[\'\'\]:after{--tw-content:"";content:var(--tw-content)}.hover\:isolate:hover{isolation:isolate}.hover\:border-primary-200:hover{--tw-border-opacity:1;border-color:rgb(171 228 214/var(--tw-border-opacity))}.hover\:border-transparent:hover{border-color:#0000}.hover\:bg-gray-200:hover{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.hover\:bg-primary-100:hover{--tw-bg-opacity:1;background-color:rgb(213 242 235/var(--tw-bg-opacity))}.hover\:bg-primary-700:hover{--tw-bg-opacity:1;background-color:rgb(35 98 90/var(--tw-bg-opacity))}.hover\:bg-primary-800:hover{--tw-bg-opacity:1;background-color:rgb(32 79 74/var(--tw-bg-opacity))}.hover\:bg-secondary-100:hover{--tw-bg-opacity:1;background-color:rgb(255 227 223/var(--tw-bg-opacity))}.hover\:bg-secondary-700:hover{--tw-bg-opacity:1;background-color:rgb(200 38 13/var(--tw-bg-opacity))}.hover\:bg-opacity-50:hover{--tw-bg-opacity:0.5}.hover\:font-bold:hover{font-weight:700}.hover\:text-gray-50:hover{--tw-text-opacity:1;color:rgb(249 250 251/var(--tw-text-opacity))}.hover\:text-primary-400:hover{--tw-text-opacity:1;color:rgb(78 179 161/var(--tw-text-opacity))}.hover\:text-primary-50:hover{--tw-text-opacity:1;color:rgb(243 250 249/var(--tw-text-opacity))}.hover\:text-primary-500:hover{--tw-text-opacity:1;color:rgb(54 156 140/var(--tw-text-opacity))}.hover\:text-primary-700:hover{--tw-text-opacity:1;color:rgb(35 98 90/var(--tw-text-opacity))}.hover\:text-primary-800:hover{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.hover\:opacity-75:hover{opacity:.75}.focus\:isolate:focus{isolation:isolate}.focus\:border-primary-300:focus{--tw-border-opacity:1;border-color:rgb(122 206 189/var(--tw-border-opacity))}.focus\:border-primary-500:focus{--tw-border-opacity:1;border-color:rgb(54 156 140/var(--tw-border-opacity))}.focus\:border-transparent:focus{border-color:#0000}.focus\:bg-opacity-50:focus{--tw-bg-opacity:0.5}.focus\:text-primary-800:focus{--tw-text-opacity:1;color:rgb(32 79 74/var(--tw-text-opacity))}.focus\:outline-none:focus{outline:2px solid #0000;outline-offset:2px}.focus\:ring-\[1px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-\[1px\]:focus,.focus\:ring-\[3px\]:focus{box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.focus\:ring-\[3px\]:focus{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-primary-200:focus{--tw-ring-opacity:1;--tw-ring-color:rgb(171 228 214/var(--tw-ring-opacity))}.focus-visible\:outline:focus-visible{outline-style:solid}.focus-visible\:outline-2:focus-visible{outline-width:2px}.focus-visible\:outline-offset-2:focus-visible{outline-offset:2px}.focus-visible\:outline-black:focus-visible{outline-color:#000}.active\:opacity-100:active{opacity:1}.active\:outline-offset-0:active{outline-offset:0}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:opacity-50:disabled{opacity:.5}.group\/btn:hover .group-hover\/btn\:w-2{width:.5rem}.group\/btn:hover .group-hover\/btn\:w-2\.5{width:.625rem}.group:hover .group-hover\:scale-100{--tw-scale-x:1;--tw-scale-y:1}.group:hover .group-hover\:scale-100,.group:hover .group-hover\:scale-x-100{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.group:hover .group-hover\:scale-x-100{--tw-scale-x:1}.group:hover .group-hover\:opacity-100,.group\/btn:hover .group-hover\/btn\:opacity-100{opacity:1}.peer:checked~.peer-checked\:bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.peer:checked~.peer-checked\:text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.peer:checked~.peer-checked\:after\:translate-x-5:after{content:var(--tw-content);--tw-translate-x:1.25rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.peer:focus~.peer-focus\:outline{outline-style:solid}.peer:focus~.peer-focus\:outline-2{outline-width:2px}.peer:focus~.peer-focus\:outline-offset-2{outline-offset:2px}.peer:focus~.peer-focus\:outline-gray-800{outline-color:#1f2937}.peer:focus:checked~.peer-focus\:peer-checked\:outline-primary-500{outline-color:#369c8c}.peer:active~.peer-active\:outline-offset-0{outline-offset:0}.peer:disabled~.peer-disabled\:cursor-not-allowed{cursor:not-allowed}.peer:disabled~.peer-disabled\:opacity-70{opacity:.7}@media (prefers-reduced-motion:reduce){.motion-reduce\:transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}}@media (min-width:640px){.sm\:left-14{left:3.5rem}.sm\:w-auto{width:auto}.sm\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.sm\:items-center{align-items:center}.sm\:px-6{padding-left:1.5rem;padding-right:1.5rem}.sm\:pb-40{padding-bottom:10rem}.sm\:pl-20{padding-left:5rem}.sm\:text-3xl{font-size:1.875rem;line-height:2.25rem}.sm\:text-5xl{font-size:3rem;line-height:1}.sm\:text-sm{font-size:.875rem;line-height:1.25rem}}@media (min-width:768px){.md\:block{display:block}.md\:h-screen{height:100vh}.md\:w-16{width:4rem}.md\:w-20{width:5rem}.md\:w-24{width:6rem}.md\:max-w-2xl{max-width:42rem}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.md\:gap-16{gap:4rem}.md\:px-14{padding-left:3.5rem;padding-right:3.5rem}.md\:py-10{padding-top:2.5rem;padding-bottom:2.5rem}.md\:pb-52{padding-bottom:13rem}.md\:text-3xl{font-size:1.875rem;line-height:2.25rem}.md\:text-4xl{font-size:2.25rem;line-height:2.5rem}.md\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width:1024px){.lg\:left-20{left:5rem}.lg\:order-last{order:9999}.lg\:col-span-3{grid-column:span 3/span 3}.lg\:col-span-4{grid-column:span 4/span 4}.lg\:-my-11{margin-top:-2.75rem;margin-bottom:-2.75rem}.lg\:-mr-9{margin-right:-2.25rem}.lg\:mb-0{margin-bottom:0}.lg\:mb-16{margin-bottom:4rem}.lg\:mt-0{margin-top:0}.lg\:block{display:block}.lg\:flex{display:flex}.lg\:grid{display:grid}.lg\:max-w-4xl{max-width:56rem}.lg\:max-w-7xl{max-width:80rem}.lg\:max-w-max{max-width:-moz-max-content;max-width:max-content}.lg\:max-w-xl{max-width:36rem}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-7{grid-template-columns:repeat(7,minmax(0,1fr))}.lg\:gap-5{gap:1.25rem}.lg\:p-10{padding:2.5rem}.lg\:p-8{padding:2rem}.lg\:px-8{padding-left:2rem;padding-right:2rem}.lg\:py-12{padding-top:3rem;padding-bottom:3rem}.lg\:py-24{padding-top:6rem;padding-bottom:6rem}.lg\:pb-20{padding-bottom:5rem}.lg\:pb-60{padding-bottom:15rem}.lg\:pl-28{padding-left:7rem}.lg\:pt-20{padding-top:5rem}.lg\:pt-40{padding-top:10rem}.lg\:text-4xl{font-size:2.25rem;line-height:2.5rem}.lg\:text-5xl{font-size:3rem;line-height:1}.lg\:text-6xl{font-size:3.75rem;line-height:1}.lg\:text-lg{font-size:1.125rem;line-height:1.75rem}}@media (min-width:1280px){.xl\:col-span-3{grid-column:span 3/span 3}.xl\:mx-auto{margin-left:auto;margin-right:auto}.xl\:flex{display:flex}.xl\:hidden{display:none}.xl\:w-auto{width:auto}.xl\:w-full{width:100%}.xl\:max-w-lg{max-width:32rem}.xl\:max-w-xl{max-width:36rem}.xl\:gap-16{gap:4rem}.xl\:gap-x-16{-moz-column-gap:4rem;column-gap:4rem}.xl\:px-20{padding-left:5rem;padding-right:5rem}.xl\:py-20{padding-top:5rem;padding-bottom:5rem}.xl\:py-32{padding-top:8rem}.xl\:pb-32,.xl\:py-32{padding-bottom:8rem}.xl\:pb-\[16\.5rem\]{padding-bottom:16.5rem}.xl\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (prefers-color-scheme:dark){.dark\:border-gray-700{--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}.dark\:bg-gray-900{--tw-bg-opacity:1;background-color:rgb(17 24 39/var(--tw-bg-opacity))}.dark\:text-gray-300{--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity))}.dark\:after\:bg-gray-300:after{content:var(--tw-content);--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity))}.peer:checked~.dark\:peer-checked\:bg-primary-500{--tw-bg-opacity:1;background-color:rgb(54 156 140/var(--tw-bg-opacity))}.peer:focus~.dark\:peer-focus\:outline-gray-300{outline-color:#d1d5db}.peer:focus:checked~.dark\:peer-focus\:peer-checked\:outline-primary-500{outline-color:#369c8c}} diff --git a/volume_index.ml b/volume_index.ml index 051d8d04..afae4e64 100644 --- a/volume_index.ml +++ b/volume_index.ml @@ -387,13 +387,26 @@ let volume_index_layout volumes policy = ] [ txt "View" ] else - span + button ~a: [ + a_disabled (); a_class - [ "gap-x-2 py-1 px-2" ]; + [ + "inline-flex \ + cursor-not-allowed \ + items-center gap-x-2 \ + text-sm font-semibold \ + rounded border \ + disabled border-1 \ + py-1 px-2 \ + border-gray-400 \ + text-gray-300 \ + disabled:opacity-50 \ + disabled:pointer-events-none"; + ]; ] - []); + [ txt "View" ]); button ~a: [ From 2477959f1d4fc881776e9a5b035641edf8f7d2de Mon Sep 17 00:00:00 2001 From: PixieDust <111846546+PizieDust@users.noreply.github.com> Date: Fri, 8 Nov 2024 12:25:40 +0100 Subject: [PATCH 08/19] Update assets/main.js Co-authored-by: Hannes Mehnert --- assets/main.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/assets/main.js b/assets/main.js index 7385b11e..16621aea 100644 --- a/assets/main.js +++ b/assets/main.js @@ -551,7 +551,7 @@ async function deleteVolume(block_name) { if (data.status === 200) { formAlert.classList.remove("hidden", "text-secondary-500"); formAlert.classList.add("text-primary-500"); - formAlert.textContent = "Succesfully deleted"; + formAlert.textContent = "Successfully deleted"; postAlert("bg-primary-300", "Volume deleted succesfully"); setTimeout(() => window.location.reload(), 1000); buttonLoading(deleteButton, false, "Delete") From 96435ca97e9bd8e00a82fa944c1ab1919fa3075a Mon Sep 17 00:00:00 2001 From: PixieDust <111846546+PizieDust@users.noreply.github.com> Date: Fri, 8 Nov 2024 12:25:53 +0100 Subject: [PATCH 09/19] Update unikernel.ml Co-authored-by: Hannes Mehnert --- unikernel.ml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index 3a37255f..c25ae5ec 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -226,8 +226,8 @@ struct let ct = Multipart_form.Content_type.of_string (content_type ^ "\r\n") in match ct with | Error (`Msg msg) -> - Logs.warn (fun m -> m "couldn't content-type: %S" msg); - Error (`Msg ("couldn't content-type:" ^ msg)) |> Lwt.return + Logs.warn (fun m -> m "couldn't parse content-type %s: %S" content_type msg); + Error (`Msg ("couldn't parse content-type " ^ content_type ^ ": " ^ msg)) |> Lwt.return | Ok ct -> ( match Multipart_form.of_string_to_list data ct with | Error (`Msg msg) -> From 0a433f179de8729d0ea6cb8b089f3fc26c9dbfe0 Mon Sep 17 00:00:00 2001 From: PixieDust <111846546+PizieDust@users.noreply.github.com> Date: Fri, 8 Nov 2024 12:26:00 +0100 Subject: [PATCH 10/19] Update unikernel.ml Co-authored-by: Hannes Mehnert --- unikernel.ml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index c25ae5ec..142b4e2f 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -231,8 +231,8 @@ struct | Ok ct -> ( match Multipart_form.of_string_to_list data ct with | Error (`Msg msg) -> - Logs.warn (fun m -> m "couldn't multipart: %s" msg); - Error (`Msg ("Couldn't multipart: " ^ msg)) |> Lwt.return + Logs.warn (fun m -> m "couldn't decode multipart data: %s" msg); + Error (`Msg ("Couldn't decode multipart data: " ^ msg)) |> Lwt.return | Ok (m, assoc) -> Ok (m, assoc) |> Lwt.return) let sign_up reqd = From 239eb1217542e8be2b6d5236cc3585c48ee9e40e Mon Sep 17 00:00:00 2001 From: PixieDust <111846546+PizieDust@users.noreply.github.com> Date: Fri, 8 Nov 2024 12:26:17 +0100 Subject: [PATCH 11/19] Update unikernel.ml Co-authored-by: Hannes Mehnert --- unikernel.ml | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index 142b4e2f..5e304186 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -950,8 +950,7 @@ struct ~data:"Couldn't find unikernel name in json" `Bad_request let unikernel_create albatross reqd (user : User_model.user) = - read_multipart_data reqd >>= fun result -> - match result with + read_multipart_data reqd >>= function | Error (`Msg msg) -> Middleware.http_response reqd ~title:"Error" ~data:("Couldn't multipart: " ^ msg) From c90cd3e56be804db8acc4dda2a977d9adda22610 Mon Sep 17 00:00:00 2001 From: "Automated ocamlformat GitHub action, developed by robur.coop" Date: Fri, 8 Nov 2024 11:29:15 +0000 Subject: [PATCH 12/19] formatted code --- unikernel.ml | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index 5e304186..de62a0d8 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -226,13 +226,17 @@ struct let ct = Multipart_form.Content_type.of_string (content_type ^ "\r\n") in match ct with | Error (`Msg msg) -> - Logs.warn (fun m -> m "couldn't parse content-type %s: %S" content_type msg); - Error (`Msg ("couldn't parse content-type " ^ content_type ^ ": " ^ msg)) |> Lwt.return + Logs.warn (fun m -> + m "couldn't parse content-type %s: %S" content_type msg); + Error + (`Msg ("couldn't parse content-type " ^ content_type ^ ": " ^ msg)) + |> Lwt.return | Ok ct -> ( match Multipart_form.of_string_to_list data ct with | Error (`Msg msg) -> Logs.warn (fun m -> m "couldn't decode multipart data: %s" msg); - Error (`Msg ("Couldn't decode multipart data: " ^ msg)) |> Lwt.return + Error (`Msg ("Couldn't decode multipart data: " ^ msg)) + |> Lwt.return | Ok (m, assoc) -> Ok (m, assoc) |> Lwt.return) let sign_up reqd = From 8a4df934ba0df4b934e446e45ae11d8e0783de01 Mon Sep 17 00:00:00 2001 From: PixieDust <111846546+PizieDust@users.noreply.github.com> Date: Fri, 8 Nov 2024 13:38:02 +0100 Subject: [PATCH 13/19] Update unikernel.ml Co-authored-by: Hannes Mehnert --- unikernel.ml | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index de62a0d8..70804eba 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1224,11 +1224,7 @@ struct reply); []) >>= fun blocks -> - let policy = - match Albatross.policy ~domain:user.name albatross with - | Ok p -> p - | Error _ -> None - in + let policy = Result.to_option (Albatross.policy ~domain:user.name albatross) in let now = Ptime.v (P.now_d_ps ()) in generate_csrf_token store user now reqd >>= function | Ok csrf -> From b7be7de5a7fff85bfaec21d619a624a8129d1708 Mon Sep 17 00:00:00 2001 From: "Automated ocamlformat GitHub action, developed by robur.coop" Date: Fri, 8 Nov 2024 12:41:05 +0000 Subject: [PATCH 14/19] formatted code --- unikernel.ml | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/unikernel.ml b/unikernel.ml index 70804eba..d1b02873 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1224,7 +1224,9 @@ struct reply); []) >>= fun blocks -> - let policy = Result.to_option (Albatross.policy ~domain:user.name albatross) in + let policy = + Result.to_option (Albatross.policy ~domain:user.name albatross) + in let now = Ptime.v (P.now_d_ps ()) in generate_csrf_token store user now reqd >>= function | Ok csrf -> From 3e9eca0837c7a8552e3234543c3ade90acf1b135 Mon Sep 17 00:00:00 2001 From: Hannes Mehnert Date: Fri, 8 Nov 2024 15:56:39 +0100 Subject: [PATCH 15/19] Update unikernel.ml --- unikernel.ml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/unikernel.ml b/unikernel.ml index d1b02873..a7139e1c 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1225,7 +1225,7 @@ struct []) >>= fun blocks -> let policy = - Result.to_option (Albatross.policy ~domain:user.name albatross) + Result.fold ~ok:Fun.id ~error:(fun _ -> None) (Albatross.policy ~domain:user.name albatross) in let now = Ptime.v (P.now_d_ps ()) in generate_csrf_token store user now reqd >>= function From 6b20fc4d1335332f502bfa39e7287cfd9d9ed55d Mon Sep 17 00:00:00 2001 From: "Automated ocamlformat GitHub action, developed by robur.coop" Date: Fri, 8 Nov 2024 14:59:42 +0000 Subject: [PATCH 16/19] formatted code --- unikernel.ml | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/unikernel.ml b/unikernel.ml index a7139e1c..fc543ec3 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1225,7 +1225,9 @@ struct []) >>= fun blocks -> let policy = - Result.fold ~ok:Fun.id ~error:(fun _ -> None) (Albatross.policy ~domain:user.name albatross) + Result.fold ~ok:Fun.id + ~error:(fun _ -> None) + (Albatross.policy ~domain:user.name albatross) in let now = Ptime.v (P.now_d_ps ()) in generate_csrf_token store user now reqd >>= function From de852fdad634f47955273ceb94811d4a4e01d1b5 Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 16:36:53 +0100 Subject: [PATCH 17/19] block and unikernel name checks --- assets/main.js | 22 ++++++++++++++++++---- 1 file changed, 18 insertions(+), 4 deletions(-) diff --git a/assets/main.js b/assets/main.js index 16621aea..821646a1 100644 --- a/assets/main.js +++ b/assets/main.js @@ -173,7 +173,7 @@ async function deployUnikernel() { const binary = document.getElementById("unikernel-binary").files[0]; const molly_csrf = document.getElementById("molly-csrf").value; const formAlert = document.getElementById("form-alert"); - if (!name || !binary) { + if (!isValidName(name) || !binary) { formAlert.classList.remove("hidden", "text-primary-500"); formAlert.classList.add("text-secondary-500"); formAlert.textContent = "Please fill in the required data" @@ -579,7 +579,7 @@ async function createVolume() { const block_compressed = document.getElementById("block_compressed").checked; const block_data = document.getElementById("block_data").files[0]; try { - if (block_name === "") { + if (!isValidName(block_name)) { formAlert.classList.remove("hidden", "text-primary-500"); formAlert.classList.add("text-secondary-500"); formAlert.textContent = "Please enter a name for this volume" @@ -614,9 +614,9 @@ async function createVolume() { if (data.status === 200) { formAlert.classList.remove("hidden", "text-secondary-500"); formAlert.classList.add("text-primary-500"); - formAlert.textContent = "Succesfully deleted"; + formAlert.textContent = "Succesfully created"; postAlert("bg-primary-300", "Volume created succesfully"); - setTimeout(() => window.location.reload(), 000); + setTimeout(() => window.location.reload(), 1000); buttonLoading(createButton, false, "Create volume") } else { formAlert.classList.remove("hidden", "text-primary-500"); @@ -632,3 +632,17 @@ async function createVolume() { buttonLoading(createButton, false, "Create volume") } } + +function isValidName(s) { + const length = s.length; + if (length === 0 || length >= 64) return false; + if (s[0] === '-') return false; + for (let i = 0; i < length; i++) { + const char = s[i]; + if (!(/[a-zA-Z0-9.-]/).test(char)) { + return false; + } + } + return true; +} + From ab22fe969b288937af2b891326a515e5c1025e1e Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 18:20:17 +0100 Subject: [PATCH 18/19] type checking user inputted fields --- assets/main.js | 108 ++++++++++++++++++++++++++----------------------- unikernel.ml | 91 ++++++++++++++++++++++++----------------- 2 files changed, 111 insertions(+), 88 deletions(-) diff --git a/assets/main.js b/assets/main.js index 821646a1..09317edb 100644 --- a/assets/main.js +++ b/assets/main.js @@ -578,53 +578,59 @@ async function createVolume() { const formAlert = document.getElementById("form-alert"); const block_compressed = document.getElementById("block_compressed").checked; const block_data = document.getElementById("block_data").files[0]; + + if (!isValidName(block_name)) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = "Please enter a name for this volume" + buttonLoading(createButton, false, "Create volume") + return; + } + if (Number(block_size) < 1) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = "Volume size must be 1MB or greater." + buttonLoading(createButton, false, "Create volume") + return; + } + if (data_toggle && !block_data) { + formAlert.classList.remove("hidden", "text-primary-500"); + formAlert.classList.add("text-secondary-500"); + formAlert.textContent = "You must upload a file else switch 'Dumb data to this volume' off" + buttonLoading(createButton, false, "Create volume") + return; + } + try { - if (!isValidName(block_name)) { - formAlert.classList.remove("hidden", "text-primary-500"); - formAlert.classList.add("text-secondary-500"); - formAlert.textContent = "Please enter a name for this volume" + buttonLoading(createButton, true, "Creating...") + let formData = new FormData(); + let json_data = JSON.stringify( + { + "block_name": block_name, + "block_size": Number(block_size), + "block_compressed": block_compressed, + "molly_csrf": molly_csrf + }) + formData.append("block_data", block_data) + formData.append("json_data", json_data) + const response = await fetch("/api/volume/create", { + method: 'POST', + body: formData + }) + const data = await response.json(); + if (data.status === 200) { + formAlert.classList.remove("hidden", "text-secondary-500"); + formAlert.classList.add("text-primary-500"); + formAlert.textContent = "Succesfully created"; + postAlert("bg-primary-300", "Volume created succesfully"); + setTimeout(() => window.location.reload(), 1000); buttonLoading(createButton, false, "Create volume") - } - else if (Number(block_size) < 1) { + } else { formAlert.classList.remove("hidden", "text-primary-500"); formAlert.classList.add("text-secondary-500"); - formAlert.textContent = "Volume size must be 1MB or greater." + formAlert.textContent = data.data buttonLoading(createButton, false, "Create volume") } - else { - buttonLoading(createButton, true, "Creating...") - let formData = new FormData(); - formData.append("block_name", block_name); - formData.append("block_size", Number(block_size)) - if (data_toggle && !block_data) { - formAlert.classList.remove("hidden", "text-primary-500"); - formAlert.classList.add("text-secondary-500"); - formAlert.textContent = "You must upload a file else switch 'Dumb data to this volume' off" - buttonLoading(createButton, false, "Create volume") - return; - } - formData.append("block_compressed", block_compressed) - formData.append("block_data", block_data) - formData.append("molly_csrf", molly_csrf) - const response = await fetch("/api/volume/create", { - method: 'POST', - body: formData - }) - const data = await response.json(); - if (data.status === 200) { - formAlert.classList.remove("hidden", "text-secondary-500"); - formAlert.classList.add("text-primary-500"); - formAlert.textContent = "Succesfully created"; - postAlert("bg-primary-300", "Volume created succesfully"); - setTimeout(() => window.location.reload(), 1000); - buttonLoading(createButton, false, "Create volume") - } else { - formAlert.classList.remove("hidden", "text-primary-500"); - formAlert.classList.add("text-secondary-500"); - formAlert.textContent = data.data - buttonLoading(createButton, false, "Create volume") - } - } } catch (error) { formAlert.classList.remove("hidden", "text-primary-500"); formAlert.classList.add("text-secondary-500"); @@ -634,15 +640,15 @@ async function createVolume() { } function isValidName(s) { - const length = s.length; - if (length === 0 || length >= 64) return false; - if (s[0] === '-') return false; - for (let i = 0; i < length; i++) { - const char = s[i]; - if (!(/[a-zA-Z0-9.-]/).test(char)) { - return false; - } - } - return true; + const length = s.length; + if (length === 0 || length >= 64) return false; + if (s[0] === '-') return false; + for (let i = 0; i < length; i++) { + const char = s[i]; + if (!(/[a-zA-Z0-9.-]/).test(char)) { + return false; + } + } + return true; } diff --git a/unikernel.ml b/unikernel.ml index fc543ec3..eba8a578 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1282,43 +1282,60 @@ struct `Bad_request | Ok (m, assoc) -> ( let m, _r = to_map ~assoc m in - match - ( Map.find_opt "block_name" m, - Map.find_opt "block_size" m, - Map.find_opt "block_data" m, - Map.find_opt "block_compressed" m, - Map.find_opt "molly_csrf" m ) - with - | ( Some (_, block_name), - Some (_, block_size), - Some (_, block_data), - Some (_, block_compressed), - Some (_, form_csrf_token) ) -> - (let now = Ptime.v (P.now_d_ps ()) in - Middleware.csrf_verification user now form_csrf_token (fun reqd -> - Albatross.query albatross ~domain:user.name ~name:block_name - (`Block_cmd - (`Block_add - ( int_of_string block_size, - bool_of_string block_compressed, - Some block_data ))) - >>= function - | Error msg -> - Logs.err (fun m -> m "Error querying albatross: %s" msg); - Middleware.http_response reqd ~title:"Error" - ~data:("Error querying albatross: " ^ msg) - `Internal_server_error - | Ok (_hdr, res) -> ( - match Albatross_json.res res with - | Ok res -> - Middleware.http_response reqd ~title:"Success" - ~data:(Yojson.Basic.to_string res) - `OK - | Error (`String res) -> - Middleware.http_response reqd ~title:"Error" - ~data:(Yojson.Basic.to_string (`String res)) - `Internal_server_error))) - reqd + match (Map.find_opt "json_data" m, Map.find_opt "block_data" m) with + | Some (_, json_data), Some (_, block_data) -> ( + let json = + try Ok (Yojson.Basic.from_string json_data) + with Yojson.Json_error s -> Error (`Msg s) + in + match json with + | Ok (`Assoc json_dict) -> ( + match + Utils.Json. + ( get "block_name" json_dict, + get "block_size" json_dict, + get "block_compressed" json_dict, + get "molly_csrf" json_dict ) + with + | ( Some (`String block_name), + Some (`Int block_size), + Some (`Bool block_compressed), + Some (`String form_csrf) ) -> + let now = Ptime.v (P.now_d_ps ()) in + Middleware.csrf_verification user now form_csrf + (fun reqd -> + Albatross.query albatross ~domain:user.name + ~name:block_name + (`Block_cmd + (`Block_add + (block_size, block_compressed, Some block_data))) + >>= function + | Error msg -> + Logs.err (fun m -> + m "Error querying albatross: %s" msg); + Middleware.http_response reqd ~title:"Error" + ~data:("Error querying albatross: " ^ msg) + `Internal_server_error + | Ok (_hdr, res) -> ( + match Albatross_json.res res with + | Ok res -> + Middleware.http_response reqd ~title:"Success" + ~data:(Yojson.Basic.to_string res) + `OK + | Error (`String res) -> + Middleware.http_response reqd ~title:"Error" + ~data:(Yojson.Basic.to_string (`String res)) + `Internal_server_error)) + reqd + | _ -> + Middleware.http_response reqd ~title:"Error" + ~data: + (Fmt.str "Register: Unexpected fields. Got %s" + (Yojson.Basic.to_string (`Assoc json_dict))) + `Bad_request) + | _ -> + Middleware.http_response reqd ~title:"Error" + ~data:"Register account: expected a dictionary" `Bad_request) | _ -> Logs.warn (fun m -> m "couldn't find fields"); Middleware.http_response reqd ~title:"Error" From a17e51c157844297905322a5c518f371e2d33c80 Mon Sep 17 00:00:00 2001 From: PizieDust Date: Fri, 8 Nov 2024 18:23:39 +0100 Subject: [PATCH 19/19] renames --- unikernel.ml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/unikernel.ml b/unikernel.ml index eba8a578..8e378cbb 100644 --- a/unikernel.ml +++ b/unikernel.ml @@ -1330,12 +1330,12 @@ struct | _ -> Middleware.http_response reqd ~title:"Error" ~data: - (Fmt.str "Register: Unexpected fields. Got %s" + (Fmt.str "Create volume: Unexpected fields. Got %s" (Yojson.Basic.to_string (`Assoc json_dict))) `Bad_request) | _ -> Middleware.http_response reqd ~title:"Error" - ~data:"Register account: expected a dictionary" `Bad_request) + ~data:"Create volume: expected a dictionary" `Bad_request) | _ -> Logs.warn (fun m -> m "couldn't find fields"); Middleware.http_response reqd ~title:"Error"