From 921dd28f7499e46d78bd39d9ac7630efc50b64e8 Mon Sep 17 00:00:00 2001 From: Alex Auvolat Date: Tue, 1 Feb 2022 18:23:36 +0100 Subject: [PATCH] add blog post --- content/blog/2022-fosdem.md | 2 + content/blog/2022-who-we-are.md | 168 ++++++++++++++++++++++++++++++++ static/style.css | 1 + templates/download.html | 2 +- 4 files changed, 172 insertions(+), 1 deletion(-) create mode 100644 content/blog/2022-who-we-are.md create mode 100644 static/style.css diff --git a/content/blog/2022-fosdem.md b/content/blog/2022-fosdem.md index 0fbc453..8f1f24e 100644 --- a/content/blog/2022-fosdem.md +++ b/content/blog/2022-fosdem.md @@ -8,6 +8,8 @@ On next Sunday, Febuary 6th, 2022, we will be there to present Garage.* +--- + In 2000, a belgian free software activist going by the name of Raphael Baudin set out to create a small event for free software developpers in Brussels. This event quickly became the "Free and Open Source Developers' European Meeting", diff --git a/content/blog/2022-who-we-are.md b/content/blog/2022-who-we-are.md new file mode 100644 index 0000000..2e3f10e --- /dev/null +++ b/content/blog/2022-who-we-are.md @@ -0,0 +1,168 @@ ++++ +title="Who we are and what we're doing" +date=2022-02-01 ++++ + +*Deuxfleurs is a non-profit based in France that aims to defend and promote +individual freedom and rights on the Internet. In their quest to build a +decentralized, resilient self-hosting infrastructure, they have found that +currently existing software is often ill suited to such a particular deployment +scenario. In the context of data storage, Garage was built to provide a highly +available data store that exploits redundancy over different geographical +locations, and does its best to not be too impacted by network latencies.* + + + +--- + +Hello! We are Deuxfleurs, a non-profit based in France working to promote +self-hosting and small-scale hosting. + +What does that mean? Well, we figured that big tech monopolies such as Google, +Facebook or Amazon today hold disproportionate power and are becoming quite +dangerous to us, citizens of the Internet. They know everything we are doing, +saying, and even thinking, and they are not making good use of that +information. The interests of these companies are those of the capitalist +elite: they are mostly interested in making huge profits by exploiting the +Earth's precious resources, producing, advertising and selling us massive +amounts of stuff we don't need. They don't truly care about the needs of the +people, nor do they care that planetary destruction is under way because of +them. + +Big tech monopolies are in a particularly strong position to influence our +behaviors, consciously or not, because we rely on them for selecting the online +content we read, watch, or listen to. Advertising is omnipresent, and because +they know us so well, they can subvert us into thinking that a mindless +consumer society is what we truly want, whereas we most likely would choose +otherwise if we had the chance to think by ourselves. + +We don't want that. That's not what the Internet is for. Freedom is freedom +from influence: the ability to do things by oneself, for oneself, on one's own +terms. Self-hosting is both the means by which we reclaim this freedom on the +Internet – by not using services of big tech monopolies and thus removing +ourselves from their influence – and the result of applying our critical +thinking and our technical abilities to build the Internet that suits us. + +Self-hosting means that we don't use cloud services. Instead, we store our +personal data on computers that we own, which we run at home. We build local +communities to share the services that we run with non-technical people. We +communicate with other groups that do the same (or, sometimes, that don't) +thanks to standard protocols such as HTTP, e-mail, or Matrix, that allow a +global community to exist outside of big tech monopolies. + +### Self-hosting is a hard problem + +As I said, self-hosting means running our own hardware at home, and providing +24/7 Internet services from there. We have many reasons for doing this. One is +because this is the only way we can truly control who has access to our data. +Another one is that it helps us be aware of the physical substrate of which the +Internet is made: making the Internet run has an environmental cost which we +want to evaluate and keep under control. The physical hardware also gives us a +sense of community, calling to mind all of the people that could currently be +connected and making use of our services, and reminding us of the purpose for +which we are doing this. + +If you have a home, you know that bad things can happen there too. The power +grid is not infallible, neither is your Internet connection. Fires and floods +happen. And the computers we are running can themselves crash at any moment, +for any number of reasons. Self-hosted solutions today are often not equipped +to face such challenges, and might suffer from unavailability or data loss +as a consequence. + +If we want to grow our communities, and attract more people that might be +sympathetic to our vision of the world, we need a baseline of quality for the +services we provide. Users can tolerate some flaws or imperfections, in the +name of defending and promoting their ideals, but if the services are +catastrophic, being unavailable at critical times, or losing users' precious +data, the compromise is much harder to make and people will be tempted to go +back to a comfortable lifestyle bestowed by big tech companies. + +Fixing availability, making services reliable even when hosted at unreliable +locations or on unreliable hardware, is one of the main objectives of +Deuxfleurs, and in particular of the project Garage which we are building. + +### Distributed systems to the rescue + +Distributed systems, or distributed computing, is a set of techniques that can +be applied to make computer services more reliable, by making them run on +several computers at once. It so happens that a few of us have studied +distributed systems, which helps a lot (some of us even have PhDs!) + +The following concepts of distributed computing are particularly relevant to +us: + +- **Crash tolerance** is when a service that runs on several computers at once + can continue operating normally even when one (or a small number) of the + computers stops working. + +- **Geo-distribution** is when the computers that make up a distributed system + are not all located in the same facility. Ideally, they would even be spread + over different cities, so that outages affecting one region do not prevent + the rest of the system from working. + +We set out to apply these concepts at Deuxfleurs to build our infrastructure, +in order to provide services that are replicated over several machines in several +geographical locations, so that we are able to provide good availability guarantees +to our users. We try to use as most as possible software packages that already +existed and are freely available, for example the Linux operating system +and the HashiCorp suite (Nomad and Consul). + +Unfortunately, in the domain of distributed data storage, the available options +weren't entirely satisfactory in our case, which is why we launched the +development of our own solution: Garage. We will talk more in other blog +posts about why Garage is better suited to us than alternative options. In this +post, I will simply try to give a high-level overview of what Garage is. + +### What is Garage, exactly? + +Garage is a distributed storage solution, that automatically replicates your +data on several servers. Garage takes into account the geographical location +of servers, and ensures that copies of your data are located at different +locations when possible for maximal redundancy, a unique feature in the +landscape of distributed storage systems. + +Garage implements the Amazon S3 protocol, a de-facto standard that makes it +compatible with a large variety of existing software. For instance it can be +used as a storage back-end for many self-hosted web applications such as +NextCloud, Matrix, Mastodon, Peertube, and many others, replacing the local +file system of a server by a distributed storage layer. Garage can also be +used to synchronize your files or store your backups with utilities such as +Rclone or Restic. Last but not least, Garage can be used to host static +websites, such as the one you are currently reading, which is served directly +by the Garage cluster we host at Deuxfleurs. + +Garage leverages the theory of distributed systems, and in particular +*Conflict-free Replicated Data Types* (CRDTs in short), a set of mathematical +tools that help us write distributed software that runs faster, by avoiding +some kinds of unnecessary chit-chat between servers. In a future blog post, +we will show how this allow us to significantly outperform Minio, our closest +competitor (another self-hostable implementation of the S3 protocol). + +On the side of software engineering, we are committed to making Garage +a tool that is reliable, lightweight, and easy to administrate. +Garage is written in the Rust programming language, which helps us ensure +the stability and safety of the software, and allows us to build software +that is fast and uses little memory. + +### Conclusion + +The current version of Garage is version 0.6, which is a *beta* release. +This means that it hasn't yet been tested by many people, and we might have +ignored some edge cases in which it would not perform as expected. + +However, we are already actively using Garage at Deuxfleurs for many uses, and +it is working exceptionally well for us. We are currently using it to store +backups of personal files, to store the media files that we send and receive +over the Matrix network, as well as to host a small but increasing number of +static websites. Our current deployment hosts about 200 000 files spread in 50 +buckets, for a total size of slightly above 500 GB. These number can seem small +when compared to the datasets you could expect your typical cloud provider to +be handling, however these sizes are fairly typical of the small-scale +self-hosted deployments we are targeting, and our Garage cluster is in no way +nearing its capacity limit. + +Today, we are proudly releasing Garage's new website, with updated +documentation pages. Poke around to try to understand how the software works, +and try installing your own instance! Your feedback is precious to us, and we +would be glad to hear back from you on our issue tracker, by e-mail, or on our +Matrix channel (`#garage:deuxfleurs.fr`). diff --git a/static/style.css b/static/style.css new file mode 100644 index 0000000..b4f6eb0 --- /dev/null +++ b/static/style.css @@ -0,0 +1 @@ +/*! tailwindcss v3.0.7 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid}:after,:before{--tw-content:""}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,-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Arial,Noto Sans,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr[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-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-size:100%;line-height:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}[type=button],[type=reset],[type=submit],button{-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}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input:-ms-input-placeholder,textarea:-ms-input-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}*,:after,:before{--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-transform:translateX(var(--tw-translate-x)) translateY(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));--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity));--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000}.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}}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.-bottom-8{bottom:-2rem}.left-1\/2{left:50%}.top-0{top:0}.-left-12{left:-3rem}.right-0\.5{right:.125rem}.right-0{right:0}.left-0{left:0}.bottom-1\/2{bottom:50%}.top-12{top:3rem}.top-20{top:5rem}.top-4{top:1rem}.right-4{right:1rem}.z-40{z-index:40}.z-50{z-index:50}.col-span-full{grid-column:1/-1}.col-span-4{grid-column:span 4/span 4}.col-span-1{grid-column:span 1/span 1}.mx-auto{margin-left:auto;margin-right:auto}.my-12{margin-top:3rem;margin-bottom:3rem}.my-2{margin-bottom:.5rem}.mt-2,.my-2{margin-top:.5rem}.mb-4{margin-bottom:1rem}.mb-10{margin-bottom:2.5rem}.mt-8{margin-top:2rem}.mt-4{margin-top:1rem}.mb-2{margin-bottom:.5rem}.mt-12{margin-top:3rem}.mr-2{margin-right:.5rem}.ml-2{margin-left:.5rem}.mb-1{margin-bottom:.25rem}.mt-0\.5{margin-top:.125rem}.mt-0{margin-top:0}.block{display:block}.inline-block{display:inline-block}.inline{display:inline}.flex{display:flex}.inline-flex{display:inline-flex}.grid{display:grid}.hidden{display:none}.h-4{height:1rem}.h-8{height:2rem}.h-6{height:1.5rem}.h-1{height:.25rem}.h-2{height:.5rem}.h-20{height:5rem}.h-10{height:2.5rem}.h-16{height:4rem}.h-full{height:100%}.h-0\.5{height:.125rem}.h-0{height:0}.h-auto{height:auto}.h-5{height:1.25rem}.h-7{height:1.75rem}.max-h-96{max-height:24rem}.w-full{width:100%}.w-96{width:24rem}.w-4{width:1rem}.w-16{width:4rem}.w-8{width:2rem}.w-6{width:1.5rem}.w-2{width:.5rem}.w-max{width:-webkit-max-content;width:-moz-max-content;width:max-content}.w-20{width:5rem}.w-2\/3{width:66.666667%}.w-10{width:2.5rem}.w-80{width:20rem}.w-1\/2{width:50%}.w-5{width:1.25rem}.w-7{width:1.75rem}.w-5\/6{width:83.333333%}.max-w-4xl{max-width:56rem}.max-w-7xl{max-width:80rem}.max-w-5xl{max-width:64rem}.origin-top-left{transform-origin:top left}.origin-top-right{transform-origin:top right}.-translate-x-1\/2{--tw-translate-x:-50%;transform:var(--tw-transform)}.translate-y-1\/2{--tw-translate-y:50%;transform:var(--tw-transform)}.-rotate-45{--tw-rotate:-45deg}.-rotate-45,.rotate-90{transform:var(--tw-transform)}.rotate-90{--tw-rotate:90deg}.transform{transform:var(--tw-transform)}@-webkit-keyframes pulse{50%{opacity:.5}}@keyframes pulse{50%{opacity:.5}}.animate-pulse{-webkit-animation:pulse 2s cubic-bezier(.4,0,.6,1) infinite;animation:pulse 2s cubic-bezier(.4,0,.6,1) infinite}.cursor-pointer{cursor:pointer}.select-none{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.list-none{list-style-type:none}.list-decimal{list-style-type:decimal}.grid-cols-1{grid-template-columns:repeat(1,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}.items-center{align-items:center}.justify-start{justify-content:flex-start}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-around{justify-content:space-around}.gap-6{gap:1.5rem}.gap-x-32{-moz-column-gap:8rem;column-gap:8rem}.gap-x-0{-moz-column-gap:0;column-gap:0}.gap-y-24{row-gap:6rem}.gap-x-12{-moz-column-gap:3rem;column-gap:3rem}.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-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-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-y-20>: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-12>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(3rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(3rem*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-1\.5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.375rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.375rem*var(--tw-space-y-reverse))}.space-y-1>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.25rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.25rem*var(--tw-space-y-reverse))}.space-y-8>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(2rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(2rem*var(--tw-space-y-reverse))}.space-x-8>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(2rem*var(--tw-space-x-reverse));margin-left:calc(2rem*(1 - var(--tw-space-x-reverse)))}.space-y-0\.5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.125rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.125rem*var(--tw-space-y-reverse))}.space-y-0>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(0px*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(0px*var(--tw-space-y-reverse))}.space-x-0>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(0px*var(--tw-space-x-reverse));margin-left:calc(0px*(1 - var(--tw-space-x-reverse)))}.space-x-1\.5>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(.375rem*var(--tw-space-x-reverse));margin-left:calc(.375rem*(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)))}.overflow-hidden{overflow:hidden}.overflow-x-hidden{overflow-x:hidden}.overflow-y-scroll{overflow-y:scroll}.whitespace-nowrap{white-space:nowrap}.rounded{border-radius:.25rem}.rounded-full{border-radius:9999px}.rounded-md{border-radius:.375rem}.rounded-2xl{border-radius:1rem}.rounded-lg{border-radius:.5rem}.rounded-r{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.rounded-l-lg{border-top-left-radius:.5rem;border-bottom-left-radius:.5rem}.rounded-tl{border-top-left-radius:.25rem}.rounded-tl-lg{border-top-left-radius:.5rem}.rounded-bl-lg{border-bottom-left-radius:.5rem}.border{border-width:1px}.border-8{border-width:8px}.border-l-4{border-left-width:4px}.border-b-2{border-bottom-width:2px}.border-b{border-bottom-width:1px}.border-t{border-top-width:1px}.border-l{border-left-width:1px}.border-r{border-right-width:1px}.border-garage-orange{--tw-border-opacity:1;border-color:rgb(255 147 41/var(--tw-border-opacity))}.border-gray-200{--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity))}.border-gray-400{--tw-border-opacity:1;border-color:rgb(156 163 175/var(--tw-border-opacity))}.border-gray-300{--tw-border-opacity:1;border-color:rgb(209 213 219/var(--tw-border-opacity))}.border-orange-400{--tw-border-opacity:1;border-color:rgb(251 146 60/var(--tw-border-opacity))}.border-garage-gray{--tw-border-opacity:1;border-color:rgb(78 78 78/var(--tw-border-opacity))}.border-transparent{border-color:transparent}.border-opacity-100{--tw-border-opacity:1}.border-opacity-20{--tw-border-opacity:0.2}.border-opacity-0{--tw-border-opacity:0}.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-300{--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity))}.bg-white{--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity))}.bg-garage-orange{--tw-bg-opacity:1;background-color:rgb(255 147 41/var(--tw-bg-opacity))}.bg-gray-200{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.bg-garage-gray{--tw-bg-opacity:1;background-color:rgb(78 78 78/var(--tw-bg-opacity))}.bg-yellow-200{--tw-bg-opacity:1;background-color:rgb(254 240 138/var(--tw-bg-opacity))}.bg-transparent{background-color:initial}.bg-opacity-10{--tw-bg-opacity:0.1}.bg-opacity-20{--tw-bg-opacity:0.2}.bg-opacity-50{--tw-bg-opacity:0.5}.bg-gradient-to-bl{background-image:linear-gradient(to bottom left,var(--tw-gradient-stops))}.bg-gradient-to-tl{background-image:linear-gradient(to top left,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.bg-gradient-to-b{background-image:linear-gradient(to bottom,var(--tw-gradient-stops))}.from-gray-50{--tw-gradient-from:#f9fafb;--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to,rgba(249,250,251,0))}.from-garage-orange{--tw-gradient-from:#ff9329;--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to,rgba(255,147,41,0))}.from-gray-400{--tw-gradient-from:#9ca3af;--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to,rgba(156,163,175,0))}.from-gray-100{--tw-gradient-from:#f3f4f6;--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to,rgba(243,244,246,0))}.from-yellow-100{--tw-gradient-from:#fef9c3;--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to,hsla(55,97%,88%,0))}.via-gray-50{--tw-gradient-stops:var(--tw-gradient-from),#f9fafb,var(--tw-gradient-to,rgba(249,250,251,0))}.via-orange-500{--tw-gradient-stops:var(--tw-gradient-from),#f97316,var(--tw-gradient-to,rgba(249,115,22,0))}.via-gray-500{--tw-gradient-stops:var(--tw-gradient-from),#6b7280,var(--tw-gradient-to,hsla(220,9%,46%,0))}.to-gray-100{--tw-gradient-to:#f3f4f6}.to-orange-300{--tw-gradient-to:#fdba74}.to-gray-400{--tw-gradient-to:#9ca3af}.to-transparent{--tw-gradient-to:transparent}.bg-size-200{background-size:200% 200%}.bg-pos-0{background-position:0 0}.p-2{padding:.5rem}.p-12{padding:3rem}.p-4{padding:1rem}.p-1\.5{padding:.375rem}.p-1{padding:.25rem}.p-0\.5{padding:.125rem}.p-0{padding:0}.p-3{padding:.75rem}.p-2\.5{padding:.625rem}.py-48{padding-top:12rem;padding-bottom:12rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.px-4{padding-left:1rem;padding-right:1rem}.py-8{padding-top:2rem;padding-bottom:2rem}.px-8{padding-left:2rem;padding-right:2rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-12{padding-top:3rem;padding-bottom:3rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-1\.5{padding-top:.375rem;padding-bottom:.375rem}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-0\.5{padding-top:.125rem;padding-bottom:.125rem}.px-1\.5{padding-left:.375rem;padding-right:.375rem}.py-0{padding-top:0;padding-bottom:0}.px-1{padding-left:.25rem;padding-right:.25rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-5{padding-left:1.25rem;padding-right:1.25rem}.py-20{padding-top:5rem;padding-bottom:5rem}.px-0{padding-left:0;padding-right:0}.px-3{padding-left:.75rem;padding-right:.75rem}.pt-4{padding-top:1rem}.pb-4{padding-bottom:1rem}.pl-4{padding-left:1rem}.pl-2{padding-left:.5rem}.pb-3{padding-bottom:.75rem}.text-center{text-align:center}.text-justify{text-align:justify}.text-2xl{font-size:1.5rem;line-height:2rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-base{font-size:1rem;line-height:1.5rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-xs{font-size:.75rem;line-height:1rem}.font-semibold{font-weight:600}.font-light{font-weight:300}.font-bold{font-weight:700}.font-extrabold{font-weight:800}.font-normal{font-weight:400}.font-thin{font-weight:100}.uppercase{text-transform:uppercase}.italic{font-style:italic}.not-italic{font-style:normal}.leading-10{line-height:2.5rem}.leading-5{line-height:1.25rem}.tracking-wide{letter-spacing:.025em}.tracking-tighter{letter-spacing:-.05em}.text-garage-orange{--tw-text-opacity:1;color:rgb(255 147 41/var(--tw-text-opacity))}.text-garage-gray{--tw-text-opacity:1;color:rgb(78 78 78/var(--tw-text-opacity))}.text-gray-900{--tw-text-opacity:1;color:rgb(17 24 39/var(--tw-text-opacity))}.text-gray-600{--tw-text-opacity:1;color:rgb(75 85 99/var(--tw-text-opacity))}.text-gray-500{--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity))}.text-white{--tw-text-opacity:1;color:rgb(255 255 255/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-400{--tw-text-opacity:1;color:rgb(156 163 175/var(--tw-text-opacity))}.underline{-webkit-text-decoration-line:underline;text-decoration-line:underline}.shadow-sm{--tw-shadow:0 1px 2px 0 rgba(0,0,0,.05);--tw-shadow-colored:0 1px 2px 0 var(--tw-shadow-color)}.shadow,.shadow-sm{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow{--tw-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px -1px rgba(0,0,0,.1);--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color)}.shadow-lg{--tw-shadow:0 10px 15px -3px rgba(0,0,0,.1),0 4px 6px -4px rgba(0,0,0,.1);--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.shadow-inner,.shadow-lg{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-inner{--tw-shadow:inset 0 2px 4px 0 rgba(0,0,0,.05);--tw-shadow-colored:inset 0 2px 4px 0 var(--tw-shadow-color)}.shadow-md{--tw-shadow:0 4px 6px -1px rgba(0,0,0,.1),0 2px 4px -2px rgba(0,0,0,.1);--tw-shadow-colored:0 4px 6px -1px var(--tw-shadow-color),0 2px 4px -2px var(--tw-shadow-color)}.shadow-2xl,.shadow-md{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-2xl{--tw-shadow:0 25px 50px -12px rgba(0,0,0,.25);--tw-shadow-colored:0 25px 50px -12px var(--tw-shadow-color)}.outline-2{outline-width:2px}.outline-garage-orange{outline-color:#ff9329}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition{transition-property:color,background-color,border-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-text-decoration-color,-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-text-decoration-color,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.duration-300{transition-duration:.3s}.duration-500{transition-duration:.5s}.duration-150{transition-duration:.15s}#navMenuToggleBtn{position:absolute;left:-100vw}#navMenuToggleBtn:checked+label{background:#fff;padding:.375rem;border-radius:.5rem;--tw-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px -1px rgba(0,0,0,.1);--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}#navMenuToggleBtn:checked~#navMenu{display:block}#map-container{background-image:url(/images/map.svg);background-repeat:no-repeat;background-position:0;background-size:cover;background-attachment:fixed;height:70vh}@media screen and (max-width:1280px){#map-container{background-position:50%;background-attachment:scroll;height:50vh}}@media screen and (max-width:768px){#map-container{background-position:25% 50%;background-size:auto;background-attachment:scroll;height:42vh}}#global_toc .deploySubMenu{position:absolute;left:-100vw}#global_toc .deploySubMenu:checked~.subMenu{display:none}#global_toc .deploySubMenu:checked+label .arrow{transform:rotate(0deg)}#documentation-section .page-content{line-height:1.9rem}#documentation-section .page-content p+p{margin-top:1.15rem}#documentation-section .page-content h1,#documentation-section .page-content h2,#documentation-section .page-content h3{margin-top:2.85rem}#documentation-section .page-content h4,#documentation-section .page-content h5,#documentation-section .page-content h6{margin-top:2rem;font-weight:600}#documentation-section .page-content ul{list-style-type:"➡";padding-left:1.25rem}#documentation-section .page-content ol{padding-left:1.25rem}#documentation-section .page-content ol li,#documentation-section .page-content ul li{padding-left:.95rem}#documentation-section .page-content ol li ol,#documentation-section .page-content ol li ul,#documentation-section .page-content ul li ol,#documentation-section .page-content ul li ul{padding-left:0}#documentation-section .page-content ol li ul,#documentation-section .page-content ul li ul{list-style-type:"➜"}@media screen and (max-width:768px){#documentation-section .page-content p{font-size:.95rem}#documentation-section .page-content p+p{margin-top:.75rem}#documentation-section .page-content code{font-size:.9rem}#documentation-section .page-content pre{overflow-x:scroll;font-size:.95rem}#documentation-section .page-content ul{padding-left:.25rem}#documentation-section .page-content ul li{font-size:.95rem;padding-left:.75rem}#documentation-section .page-content ul li a{font-size:.95rem}}#blogpost-section .page-content p+#continue-reading+p,#blogpost-section .page-content p+p{margin-top:2rem}#blogpost-section .page-content #continue-reading{margin:2rem 0}#blogpost-section .page-content ul{list-style-type:"➡";padding-left:1.25rem;margin:2rem 0}#blogpost-section .page-content hr{width:10px;height:10px;border-radius:50%;background:#ff9329;margin:35px auto}#blogpost-section .page-content ul li{padding-left:.95rem}#blogpost-section img{display:inline-block;margin:2.75rem 0}article .page-content{font-size:1rem;line-height:1.5rem}@media (min-width:768px){article .page-content{font-size:1.125rem;line-height:1.75rem}}article pre{margin-top:1rem;margin-bottom:1rem;white-space:pre-wrap;border-radius:.375rem;border-left-width:4px;--tw-border-opacity:1;border-color:rgb(78 78 78/var(--tw-border-opacity));padding:1rem;--tw-shadow:0 10px 15px -3px rgba(0,0,0,.1),0 4px 6px -4px rgba(0,0,0,.1);--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}article table{margin-top:1rem;margin-bottom:1rem;min-width:100%}article table>: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));--tw-divide-opacity:1;border-color:rgb(229 231 235/var(--tw-divide-opacity))}article table{overflow:hidden;border-bottom-width:1px;--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity));--tw-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px -1px rgba(0,0,0,.1);--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}@media (min-width:640px){article table{border-radius:.5rem}}article table thead{--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity))}article table tbody>: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));--tw-divide-opacity:1;border-color:rgb(229 231 235/var(--tw-divide-opacity))}article table tbody{--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity))}article table thead tr th{padding:.75rem 1.5rem;text-align:left;font-size:.75rem;line-height:1rem;font-weight:500;text-transform:uppercase;letter-spacing:.05em;--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity))}article table tbody tr td{white-space:nowrap;padding:1rem 1.5rem}article blockquote{margin-top:.5rem;margin-bottom:.5rem;border-left-width:4px;--tw-border-opacity:1;border-color:rgb(255 147 41/var(--tw-border-opacity));padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem}article ul{margin-left:1rem;list-style-type:disc;padding-top:1rem;padding-bottom:1rem}article ul ul{list-style-type:disc}article ul ol,article ul ul{margin-left:1rem;padding-top:0;padding-bottom:0}article ul ol{list-style-type:decimal}article ol{padding-top:1rem;padding-bottom:1rem}article ol,article ol ol{margin-left:1rem;list-style-type:decimal}article ol ol,article ol ul{padding-top:0;padding-bottom:0}article ol ul{margin-left:1rem;list-style-type:disc}li>code,li>strong>code,p>code,p>strong>code{border-radius:.25rem;--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity));padding:.125rem .25rem;font-size:1rem;line-height:1.5rem;font-weight:600;--tw-text-opacity:1;color:rgb(78 78 78/var(--tw-text-opacity));--tw-shadow:inset 0 2px 4px 0 rgba(0,0,0,.05);--tw-shadow-colored:inset 0 2px 4px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.page-content>h1{font-size:1.5rem;line-height:2rem}.page-content>h1,.page-content>h2{margin-top:1.5rem;margin-bottom:1rem;font-weight:600;--tw-text-opacity:1;color:rgb(31 41 55/var(--tw-text-opacity))}.page-content>h2{font-size:1.25rem;line-height:1.75rem}.page-content>h3{margin-top:1.5rem;margin-bottom:1rem;font-size:1.125rem;line-height:1.75rem;font-weight:600;--tw-text-opacity:1;color:rgb(31 41 55/var(--tw-text-opacity))}.page-content{--tw-text-opacity:1;color:rgb(78 78 78/var(--tw-text-opacity))}.page-content a{border-bottom-width:1px;--tw-border-opacity:1;border-color:rgb(255 147 41/var(--tw-border-opacity));font-weight:600;--tw-text-opacity:1;color:rgb(255 147 41/var(--tw-text-opacity));transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.5s}.page-content a:hover{--tw-border-opacity:1;border-color:rgb(239 68 68/var(--tw-border-opacity));--tw-text-opacity:1;color:rgb(239 68 68/var(--tw-text-opacity))}.page-content img{margin-top:2rem;margin-bottom:2rem;display:inline-block;border-radius:.25rem;--tw-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px -1px rgba(0,0,0,.1);--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.is-active{--tw-text-opacity:1;color:rgb(255 147 41/var(--tw-text-opacity))}.hover\:rounded-full:hover{border-radius:9999px}.hover\:border-orange-400:hover{--tw-border-opacity:1;border-color:rgb(251 146 60/var(--tw-border-opacity))}.hover\:border-garage-orange:hover{--tw-border-opacity:1;border-color:rgb(255 147 41/var(--tw-border-opacity))}.hover\:border-gray-500:hover{--tw-border-opacity:1;border-color:rgb(107 114 128/var(--tw-border-opacity))}.hover\:bg-gray-100:hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}.hover\:bg-orange-300:hover{--tw-bg-opacity:1;background-color:rgb(253 186 116/var(--tw-bg-opacity))}.hover\:bg-garage-orange:hover{--tw-bg-opacity:1;background-color:rgb(255 147 41/var(--tw-bg-opacity))}.hover\:bg-garage-gray:hover{--tw-bg-opacity:1;background-color:rgb(78 78 78/var(--tw-bg-opacity))}.hover\:bg-white:hover{--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity))}.hover\:bg-opacity-20:hover{--tw-bg-opacity:0.2}.hover\:bg-pos-100:hover{background-position:100% 100%}.hover\:text-garage-orange:hover{--tw-text-opacity:1;color:rgb(255 147 41/var(--tw-text-opacity))}.hover\:text-orange-600:hover{--tw-text-opacity:1;color:rgb(234 88 12/var(--tw-text-opacity))}.hover\:underline:hover{-webkit-text-decoration-line:underline;text-decoration-line:underline}.hover\:shadow-lg:hover{--tw-shadow:0 10px 15px -3px rgba(0,0,0,.1),0 4px 6px -4px rgba(0,0,0,.1);--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.hover\:shadow-lg:hover,.hover\:shadow-xl:hover{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.hover\:shadow-xl:hover{--tw-shadow:0 20px 25px -5px rgba(0,0,0,.1),0 8px 10px -6px rgba(0,0,0,.1);--tw-shadow-colored:0 20px 25px -5px var(--tw-shadow-color),0 8px 10px -6px var(--tw-shadow-color)}.hover\:shadow:hover{--tw-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px -1px rgba(0,0,0,.1);--tw-shadow-colored:0 1px 3px 0 var(--tw-shadow-color),0 1px 2px -1px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.focus\:bg-white:focus{--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity))}.focus\:outline:focus{outline-style:solid}.group:hover .group-hover\:w-8{width:2rem}.group:hover .group-hover\:translate-y-2{--tw-translate-y:0.5rem;transform:var(--tw-transform)}.group:hover .group-hover\:-translate-y-1{--tw-translate-y:-0.25rem;transform:var(--tw-transform)}.group:hover .group-hover\:bg-garage-orange{--tw-bg-opacity:1;background-color:rgb(255 147 41/var(--tw-bg-opacity))}.group:hover .group-hover\:bg-garage-gray{--tw-bg-opacity:1;background-color:rgb(78 78 78/var(--tw-bg-opacity))}.group:hover .group-hover\:text-garage-gray{--tw-text-opacity:1;color:rgb(78 78 78/var(--tw-text-opacity))}.group:hover .group-hover\:text-garage-orange{--tw-text-opacity:1;color:rgb(255 147 41/var(--tw-text-opacity))}@media (min-width:768px){.md\:static{position:static}.md\:mt-0{margin-top:0}.md\:block{display:block}.md\:inline{display:inline}.md\:hidden{display:none}.md\:w-4\/5{width:80%}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.md\:flex-row{flex-direction:row}.md\:items-start{align-items:flex-start}.md\:justify-start{justify-content:flex-start}.md\:justify-end{justify-content:flex-end}.md\: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)))}.md\:space-y-0>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(0px*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(0px*var(--tw-space-y-reverse))}.md\:bg-transparent{background-color:initial}.md\:p-4{padding:1rem}.md\:px-12{padding-left:3rem;padding-right:3rem}.md\:px-0{padding-left:0;padding-right:0}.md\:px-8{padding-left:2rem;padding-right:2rem}.md\:py-0{padding-top:0;padding-bottom:0}.md\:px-3{padding-left:.75rem;padding-right:.75rem}.md\:text-base{font-size:1rem;line-height:1.5rem}.md\:text-2xl{font-size:1.5rem;line-height:2rem}.md\:tracking-normal{letter-spacing:0}.md\:shadow-none{--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}}@media (min-width:1024px){.lg\:w-1\/3{width:33.333333%}.lg\:w-3\/4{width:75%}.lg\:flex-row{flex-direction:row}.lg\:space-x-12>:not([hidden])~:not([hidden]){--tw-space-x-reverse:0;margin-right:calc(3rem*var(--tw-space-x-reverse));margin-left:calc(3rem*(1 - var(--tw-space-x-reverse)))}.lg\:px-8{padding-left:2rem;padding-right:2rem}.lg\:text-2xl{font-size:1.5rem;line-height:2rem}}@media (min-width:1280px){.xl\:col-span-3{grid-column:span 3/span 3}.xl\:w-max{width:-webkit-max-content;width:-moz-max-content;width:max-content}.xl\:w-2\/3{width:66.666667%}.xl\:grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.xl\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.xl\:gap-x-12{-moz-column-gap:3rem;column-gap:3rem}.xl\:p-8{padding:2rem}.xl\:px-12{padding-left:3rem;padding-right:3rem}.xl\:px-0{padding-left:0;padding-right:0}.xl\:px-4{padding-left:1rem;padding-right:1rem}.xl\:text-3xl{font-size:1.875rem;line-height:2.25rem}}@media (min-width:1536px){.\32xl\:block{display:block}.\32xl\:hidden{display:none}.\32xl\:w-1\/2{width:50%}.\32xl\:px-0{padding-left:0;padding-right:0}} \ No newline at end of file diff --git a/templates/download.html b/templates/download.html index 43d786c..914c1a2 100644 --- a/templates/download.html +++ b/templates/download.html @@ -7,7 +7,7 @@ {% block content %}
-

Download

+

Download Garage